123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330 |
- 'use strict'
- const {BrowserWindow, MenuItem, webContents} = require('electron')
- const EventEmitter = require('events').EventEmitter
- const v8Util = process.atomBinding('v8_util')
- const bindings = process.atomBinding('menu')
- // Automatically generated radio menu item's group id.
- var nextGroupId = 0
- // Search between separators to find a radio menu item and return its group id,
- // otherwise generate a group id.
- var generateGroupId = function (items, pos) {
- var i, item, j, k, ref1, ref2, ref3
- if (pos > 0) {
- for (i = j = ref1 = pos - 1; ref1 <= 0 ? j <= 0 : j >= 0; i = ref1 <= 0 ? ++j : --j) {
- item = items[i]
- if (item.type === 'radio') {
- return item.groupId
- }
- if (item.type === 'separator') {
- break
- }
- }
- } else if (pos < items.length) {
- for (i = k = ref2 = pos, ref3 = items.length - 1; ref2 <= ref3 ? k <= ref3 : k >= ref3; i = ref2 <= ref3 ? ++k : --k) {
- item = items[i]
- if (item.type === 'radio') {
- return item.groupId
- }
- if (item.type === 'separator') {
- break
- }
- }
- }
- return ++nextGroupId
- }
- // Returns the index of item according to |id|.
- var indexOfItemById = function (items, id) {
- var i, item, j, len
- for (i = j = 0, len = items.length; j < len; i = ++j) {
- item = items[i]
- if (item.id === id) {
- return i
- }
- }
- return -1
- }
- // Returns the index of where to insert the item according to |position|.
- var indexToInsertByPosition = function (items, position) {
- var insertIndex
- if (!position) {
- return items.length
- }
- const [query, id] = position.split('=')
- insertIndex = indexOfItemById(items, id)
- if (insertIndex === -1 && query !== 'endof') {
- console.warn("Item with id '" + id + "' is not found")
- return items.length
- }
- switch (query) {
- case 'after':
- insertIndex++
- break
- case 'endof':
- // If the |id| doesn't exist, then create a new group with the |id|.
- if (insertIndex === -1) {
- items.push({
- id: id,
- type: 'separator'
- })
- insertIndex = items.length - 1
- }
- // Find the end of the group.
- insertIndex++
- while (insertIndex < items.length && items[insertIndex].type !== 'separator') {
- insertIndex++
- }
- }
- return insertIndex
- }
- const Menu = bindings.Menu
- Object.setPrototypeOf(Menu.prototype, EventEmitter.prototype)
- Menu.prototype._init = function () {
- this.commandsMap = {}
- this.groupsMap = {}
- this.items = []
- this.delegate = {
- isCommandIdChecked: (commandId) => {
- var command = this.commandsMap[commandId]
- return command != null ? command.checked : undefined
- },
- isCommandIdEnabled: (commandId) => {
- var command = this.commandsMap[commandId]
- return command != null ? command.enabled : undefined
- },
- isCommandIdVisible: (commandId) => {
- var command = this.commandsMap[commandId]
- return command != null ? command.visible : undefined
- },
- getAcceleratorForCommandId: (commandId, useDefaultAccelerator) => {
- const command = this.commandsMap[commandId]
- if (command == null) return
- if (command.accelerator != null) return command.accelerator
- if (useDefaultAccelerator) return command.getDefaultRoleAccelerator()
- },
- getIconForCommandId: (commandId) => {
- var command = this.commandsMap[commandId]
- return command != null ? command.icon : undefined
- },
- executeCommand: (event, commandId) => {
- const command = this.commandsMap[commandId]
- if (command == null) return
- command.click(event, BrowserWindow.getFocusedWindow(), webContents.getFocusedWebContents())
- },
- menuWillShow: () => {
- // Make sure radio groups have at least one menu item seleted.
- var checked, group, id, j, len, radioItem, ref1
- ref1 = this.groupsMap
- for (id in ref1) {
- group = ref1[id]
- checked = false
- for (j = 0, len = group.length; j < len; j++) {
- radioItem = group[j]
- if (!radioItem.checked) {
- continue
- }
- checked = true
- break
- }
- if (!checked) {
- v8Util.setHiddenValue(group[0], 'checked', true)
- }
- }
- }
- }
- }
- Menu.prototype.popup = function (window, x, y, positioningItem) {
- let asyncPopup
- // menu.popup(x, y, positioningItem)
- if (window != null && (typeof window !== 'object' || window.constructor !== BrowserWindow)) {
- // Shift.
- positioningItem = y
- y = x
- x = window
- window = null
- }
- // menu.popup(window, {})
- if (x != null && typeof x === 'object') {
- const options = x
- x = options.x
- y = options.y
- positioningItem = options.positioningItem
- asyncPopup = options.async
- }
- // Default to showing in focused window.
- if (window == null) window = BrowserWindow.getFocusedWindow()
- // Default to showing under mouse location.
- if (typeof x !== 'number') x = -1
- if (typeof y !== 'number') y = -1
- // Default to not highlighting any item.
- if (typeof positioningItem !== 'number') positioningItem = -1
- // Default to synchronous for backwards compatibility.
- if (typeof asyncPopup !== 'boolean') asyncPopup = false
- this.popupAt(window, x, y, positioningItem, asyncPopup)
- }
- Menu.prototype.closePopup = function (window) {
- if (window == null || window.constructor !== BrowserWindow) {
- window = BrowserWindow.getFocusedWindow()
- }
- if (window != null) {
- this.closePopupAt(window.id)
- }
- }
- Menu.prototype.append = function (item) {
- return this.insert(this.getItemCount(), item)
- }
- Menu.prototype.insert = function (pos, item) {
- var base, name
- if ((item != null ? item.constructor : void 0) !== MenuItem) {
- throw new TypeError('Invalid item')
- }
- switch (item.type) {
- case 'normal':
- this.insertItem(pos, item.commandId, item.label)
- break
- case 'checkbox':
- this.insertCheckItem(pos, item.commandId, item.label)
- break
- case 'separator':
- this.insertSeparator(pos)
- break
- case 'submenu':
- this.insertSubMenu(pos, item.commandId, item.label, item.submenu)
- break
- case 'radio':
- // Grouping radio menu items.
- item.overrideReadOnlyProperty('groupId', generateGroupId(this.items, pos))
- if ((base = this.groupsMap)[name = item.groupId] == null) {
- base[name] = []
- }
- this.groupsMap[item.groupId].push(item)
- // Setting a radio menu item should flip other items in the group.
- v8Util.setHiddenValue(item, 'checked', item.checked)
- Object.defineProperty(item, 'checked', {
- enumerable: true,
- get: function () {
- return v8Util.getHiddenValue(item, 'checked')
- },
- set: () => {
- var j, len, otherItem, ref1
- ref1 = this.groupsMap[item.groupId]
- for (j = 0, len = ref1.length; j < len; j++) {
- otherItem = ref1[j]
- if (otherItem !== item) {
- v8Util.setHiddenValue(otherItem, 'checked', false)
- }
- }
- return v8Util.setHiddenValue(item, 'checked', true)
- }
- })
- this.insertRadioItem(pos, item.commandId, item.label, item.groupId)
- }
- if (item.sublabel != null) {
- this.setSublabel(pos, item.sublabel)
- }
- if (item.icon != null) {
- this.setIcon(pos, item.icon)
- }
- if (item.role != null) {
- this.setRole(pos, item.role)
- }
- // Make menu accessable to items.
- item.overrideReadOnlyProperty('menu', this)
- // Remember the items.
- this.items.splice(pos, 0, item)
- this.commandsMap[item.commandId] = item
- }
- // Force menuWillShow to be called
- Menu.prototype._callMenuWillShow = function () {
- if (this.delegate != null) {
- this.delegate.menuWillShow()
- }
- this.items.forEach(function (item) {
- if (item.submenu != null) {
- item.submenu._callMenuWillShow()
- }
- })
- }
- var applicationMenu = null
- Menu.setApplicationMenu = function (menu) {
- if (!(menu === null || menu.constructor === Menu)) {
- throw new TypeError('Invalid menu')
- }
- // Keep a reference.
- applicationMenu = menu
- if (process.platform === 'darwin') {
- if (menu === null) {
- return
- }
- menu._callMenuWillShow()
- bindings.setApplicationMenu(menu)
- } else {
- BrowserWindow.getAllWindows().forEach(function (window) {
- window.setMenu(menu)
- })
- }
- }
- Menu.getApplicationMenu = function () {
- return applicationMenu
- }
- Menu.sendActionToFirstResponder = bindings.sendActionToFirstResponder
- Menu.buildFromTemplate = function (template) {
- var insertIndex, item, j, k, len, len1, menu, menuItem, positionedTemplate
- if (!Array.isArray(template)) {
- throw new TypeError('Invalid template for Menu')
- }
- positionedTemplate = []
- insertIndex = 0
- for (j = 0, len = template.length; j < len; j++) {
- item = template[j]
- if (item.position) {
- insertIndex = indexToInsertByPosition(positionedTemplate, item.position)
- } else {
- // If no |position| is specified, insert after last item.
- insertIndex++
- }
- positionedTemplate.splice(insertIndex, 0, item)
- }
- menu = new Menu()
- for (k = 0, len1 = positionedTemplate.length; k < len1; k++) {
- item = positionedTemplate[k]
- if (typeof item !== 'object') {
- throw new TypeError('Invalid template for MenuItem')
- }
- menuItem = new MenuItem(item)
- menu.append(menuItem)
- }
- return menu
- }
- module.exports = Menu
|