some tips by Kilian Valkhof

Kilian Valkhof

"UX Developer"

Partner at Fluxility

In my spare time, I like to make tools for other people, mostly developers.

(in Qt)
f.lux linux gui
(in GTK)
(in Electron!)


Cross-platform apps are nice

But make your app feel like it belongs on someone's computer.

Using shitty tools sucks

& we have enough of them already

Getting it (mostly) right is not hard

Just pay attention to the details :)

Here's some tips for Electron apps

Copy and Paste on OS X

Doesn't work

...without adding an App Menu that has them.

	if (process.platform === 'darwin') {
    var template = [{
      label: 'FromScratch',
      submenu: [{
        label: 'Quit',
        accelerator: 'CmdOrCtrl+Q',
        click: function() { app.quit(); }
    }, {
      label: 'Edit',
      submenu: [{
        label: 'Undo',
        accelerator: 'CmdOrCtrl+Z',
        selector: 'undo:'
      }, {
        label: 'Redo',
        accelerator: 'Shift+CmdOrCtrl+Z',
        selector: 'redo:'
      }, {
        type: 'separator'
      }, {
        label: 'Cut',
        accelerator: 'CmdOrCtrl+X',
        selector: 'cut:'
      }, {
        label: 'Copy',
        accelerator: 'CmdOrCtrl+C',
        selector: 'copy:'
      }, {
        label: 'Paste',
        accelerator: 'CmdOrCtrl+V',
        selector: 'paste:'
      }, {
        label: 'Select All',
        accelerator: 'CmdOrCtrl+A',
        selector: 'selectAll:'

    var osxMenu = menu.buildFromTemplate(template);

...just copy and paste this into your main.js

Specify an icon!

Or your app will look like this on Ubuntu:

Many apps get this wrong.

And it's super easy to fix:

mainWindow = new BrowserWindow({
  title: 'ElectronApp',
  icon: __dirname + '/app/assets/img/icon.png',

This also gives you a nice little icon in the topleft of your windows app chrome.

UI text is not supposed to be selectable

So prevent selection to make your app feel more native:

.my-ui-text {

Ctrl+A selects all selectable text in your app.

You need 3 icons for 3 platforms

Windows: .ICO
Linux: .PNG

Start with a 1024x1024px png

Linux, check!

to get an .ico, use icotools:

icotool -c icon.png > icon.ico

to get an .icns, use png2icns:

png2icns icon.icns icon.png

Or use a GUI tool such as img2icns (mac) or iconverticons (web, Windows, OS X)


electron-packager doesn't need the extension

					$ electron-packager . MyApp --icon=img/icon --platform=all --arch=all --version=0.36.0 --out=../dist/ --asar

White loading screens belong in browsers

Hide your app until your page has loaded:

var mainWindow = new BrowserWindow({
	title: 'ElectronApp',
	show: false,

mainWindow.webContents.on('did-finish-load', function() {;

Preserve your window dimensions

There is a special circle of hell for apps that don't do this.

Prebuilt solutions:

electron-window-state and electron-window-state-manager

Both work, have good documentation and take care of edge cases.

Get something to save your state to:

var JSONStorage = require('node-localstorage').JSONStorage;
var storageLocation = process.env[(process.platform === 'win32') ?
	'USERPROFILE' : 'HOME'] + '/.fromscratch';
global.nodeStorage = new JSONStorage(storageLocation);

When opening your app,
try to load in the state:

var windowState = {};
  try {
    windowState = global.nodeStorage.getItem('windowstate');
  } catch (err) {
    // the file is there, but corrupt. Handle appropriately.

Set your BrowserWindow dimensions:

var mainWindow = new BrowserWindow({
  title: 'ElectronApp',
  x: windowState.bounds && windowState.bounds.x || undefined,
  y: windowState.bounds && windowState.bounds.y || undefined,
  width: windowState.bounds && windowState.bounds.width || 550,
  height: windowState.bounds && windowState.bounds.height || 450,

// Restore maximised state if it is set.
// not possible via options so we do it here
if (windowState.isMaximized) {

Save your current state on move, resize and close:

['resize', 'move', 'close'].forEach(function(e) {
  mainWindow.on(e, function() {

var storeWindowState = function() {
  windowState.isMaximized = mainWindow.isMaximized();
  if (!windowState.isMaximized) {
    // only update bounds if the window isn't currently maximized
    windowState.bounds = mainWindow.getBounds();
  global.nodeStorage.setItem('windowstate', windowState);

Quick fire

For Accelerators (shortcuts) don't use Cmd or Ctrl, just use CmdOrCtrl.

Quick fire

Want to use the system font San Francisco?

body {
  font: caption;

Quick fire

For an extra native look and feel, use system colors.

Quick fire

Check out Grid Stylesheets for a contraint based layout system (like GTK, Qt or Autolayout)

Quick fire

...Though you can get very far by using calc() in your css.



I would love to get feedback, you can find me here:,
/kilianvalkhof, /kilian

Slides will end up somewhere on my blog or on Twitter. Or both!