⚠️ This lesson is retired and might contain outdated information.

Create a native desktop system menu with the Electron Menu module

Cameron Nokes
InstructorCameron Nokes
Share this video with your friends

Social Share Links

Send Tweet
Published 7 years ago
Updated a year ago

A main menu is a hallmark of native desktop applications. Electron makes it easy to create native menus and menu items in JavaScript.

We’ll go over creating a menu template using Menu.buildFromTemplate and then setting it as the main application menu via Menu.setApplicationMenu.

[00:00] We've already set up our main process JavaScript, it's creating a browser window, loading an index.html and then showing it when they're ready to show event has been emitted. Now we're going to add a native main menu to our application.

[00:13] By main menu we mean this. This is Atom's main menu, but we want to add something like this for our application. To do this, let's create a new file called mainmenu.js. Let's require app and menu from Electron.

[00:33] The typical application menus on Windows and Mac are quite a bit different, so from the get-go we're going to check if we're running on Windows. In Electron applications, it's not uncommon to have special handling for each platform.

[00:46] There's nothing special Electron going on here, this is the typical Node.js way you'd check for it. We're going to export a function that will call setMainMenu. The first thing we're going to do is we'll set up an array here called template, and we're going to define the far-left menu, also called the application menu sometimes on Mac.

[01:10] It's always the name of the application, on Windows it's usually just File is the most far-left one. We'll just add a quick menu item to it. Electron menus are defined as arrays of objects, so for our application menu we'll have an object here, and for our label, we'll check if it's Windows.

[01:32] Because if it is Windows we want to call file, if it is a Mac, what we're going to do is we're going to call app.getName. This is an Electron method that dynamically returns the name of your application as defined in your package.json.

[01:48] Then for its menu items, we pass this submenu property an array of objects. If we label here, we'll check if it's Windows. Windows generally calls it exit, and Mac calls it quit, and by convention Mac also passes the application name.

[02:13] For every menu item, Electron allows us to define an accelerator, which is just another word for a hot key for your menu item action. We'll define an accelerator here, and we'll say if it's Windows, the standard hot key for quitting an app is alt-F4, on Mac we're going to do command or ctrl+Q.

[02:36] We do command or ctrl here just in case maybe somebody's in a weird keyboard, or they're on a Linux machine, and basically command will be used on Mac, and anywhere command is not available, ctrl will be used instead for this hot key.

[02:54] This string format is an Electron defined format that you can read about more in the API documentation for menus. It's pretty intuitive and straightforward. Next we'll define a click handler. This function will be called when either the accelerator is activated from the keyboard, or when they manually click on the menu item.

[03:17] In here we'll call the quit method. We have our template done, we have a single top menu item defined with a single menu item underneath it, so now we need to set it on the application, the main menu.

[03:29] First, we need to call buildFromTemplate, and we'll pass it our template. This will take this data structure and convert it to the data structures that Electron needs to actually construct your menu.

[03:45] Then we'll pass the output of that to menu.setApplicationMenu, which will actually set your application's menu. Now let's require this in our main.js, jump in here, and we'll require and set mainMenu. We'll want to call this in our onReady handler for our app, because it won't work before then.

[04:05] OK, let's run our npm start script and check it out, npm start. We can see this is our application menu which on Mac is just going to say Electron, and whenever you're running Electron in development, this is always going to say Electron regardless of what you have in your package.json.

[04:22] When you actually package and compile your app into an executable file, this will have your actual application name here. This is our menu item that we defined, we can see that it's working.

[04:34] This is our label that we defined, and the hot key that we defined as well. If we click on that now, we can see that it quits our application. Let's start it again, and we'll try the command+Q short cut, and that works as well. Now let's add an edit menu to our main menu. Back here in our template we're going to add another object here, call this edit. Then add our submenu array here, edit menus are petty standard cross-platform, so we won't need any cross-platform checks here.

[05:09] Typically edit menus contain items like copy, paste, etc., instead of manually reimplementing all of that functionality, Electron gives us the ability to specify a role on a menu item, to designate a standard menu item without any custom functionality.

[05:24] Let's add some of these role menu items. First one we'll do role is undo, and then we'll do redo. Typically, at this point in a menu item there is a separator. This won't do anything, it will just be a visual line that separates the undo and redo from the rest of the menu items.

[05:46] Then we'll add cut, copy, paste, and select all. There's lots of different roles that you can specify for menu items. I encourage you to check out Electron's documentation to see them all. A few of them are platform-specific, most are general for all of them.

[06:03] OK, let's add a text input to our index.html so that we can try out all these different edit menu items. I'll just correct the misspelling separator then run npm start to check out our edit menu.

[06:17] Here's the edit menu that we defined, and we can see that Electron filled in the details based on the role that we provided. For example, we have undo here, and it's command+Z, on Windows it' going to be ctrl-Z. Redo on Windows it's going to be ctrl-Y, on Mac it's shift-command+Z.

[06:36] Electron has taken care of all the platform specific details here. Also, this is provided by default from MacOS. Let's try out our edit menu functionality here in this input. Do test, command+A selects all, and do command+C to copy it. Command+V paste, we can see it's working in that input.

[06:57] This is a basic overview of what's possible with a menu API. There's a lot more it can do including nested submenus, checkbox menu items, custom icons and more. Take a look at the Electron docs on the menu API to see what's possible.

James Barlow
James Barlow
~ 7 years ago

On Win32, with Electron 1.6.11, a default native menu is added to the app based on the example template in the docs:

https://electron.atom.io/docs/api/menu/#examples

rajeshkw
rajeshkw
~ 7 years ago

This is some crazy new stuff that i got to learn today!! Thanks Egghead!

Elgs
Elgs
~ 7 years ago

Could the process.platform be 'win64' on Windows 64bit systems?

Ady
Ady
~ 7 years ago

just to be clear it seems like you are using main.js and index.js interchangeably, is that the case?? Thank for a clear and concise course otherwise so far

Salinger MS
Salinger MS
~ 6 years ago

I have an error 'ReferenceError : isWindows is not defined

---- never mind got, minor problem with my code

Markdown supported.
Become a member to join the discussionEnroll Today