menubar.ts 31.3 KB
Newer Older
1 2 3 4 5 6
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

import * as nls from 'vs/nls';
7
import { isMacintosh, language } from 'vs/base/common/platform';
8
import { IEnvironmentService } from 'vs/platform/environment/common/environment';
9
import { app, shell, Menu, MenuItem, BrowserWindow, MenuItemConstructorOptions, WebContents, Event, KeyboardEvent } from 'electron';
10
import { OpenContext, IRunActionInWindowRequest, getTitleBarStyle, IRunKeybindingInWindowRequest, IWindowOpenable } from 'vs/platform/windows/common/windows';
11
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
12 13
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
import { IUpdateService, StateType } from 'vs/platform/update/common/update';
14
import product from 'vs/platform/product/common/product';
15
import { RunOnceScheduler } from 'vs/base/common/async';
16
import { ILogService } from 'vs/platform/log/common/log';
17
import { mnemonicMenuLabel as baseMnemonicLabel } from 'vs/base/common/labels';
18
import { IWindowsMainService, IWindowsCountChangedEvent } from 'vs/platform/windows/electron-main/windows';
19
import { IWorkspacesHistoryMainService } from 'vs/platform/workspaces/electron-main/workspacesHistoryMainService';
20
import { IMenubarData, IMenubarKeybinding, MenubarMenuItem, isMenubarMenuItemSeparator, isMenubarMenuItemSubmenu, isMenubarMenuItemAction, IMenubarMenu, isMenubarMenuItemUriAction } from 'vs/platform/menubar/node/menubar';
21
import { URI } from 'vs/base/common/uri';
22
import { IStateService } from 'vs/platform/state/node/state';
23
import { ILifecycleMainService } from 'vs/platform/lifecycle/electron-main/lifecycleMainService';
24
import { WorkbenchActionExecutedEvent, WorkbenchActionExecutedClassification } from 'vs/base/common/actions';
25
import { IElectronMainService } from 'vs/platform/electron/electron-main/electronMainService';
26
import { INativeEnvironmentService } from 'vs/platform/environment/node/environmentService';
27 28 29

const telemetryFrom = 'menu';

30
interface IMenuItemClickHandler {
31
	inDevTools: (contents: WebContents) => void;
32 33 34
	inNoWindow: () => void;
}

35 36 37 38 39 40 41 42 43
type IMenuItemInvocation = (
	{ type: 'commandId'; commandId: string; }
	| { type: 'keybinding'; userSettingsLabel: string; }
);

interface IMenuItemWithKeybinding {
	userSettingsLabel?: string;
}

44 45
export class Menubar {

46
	private static readonly lastKnownMenubarStorageKey = 'lastKnownMenubarData';
47

48 49
	private willShutdown: boolean | undefined;
	private appMenuInstalled: boolean | undefined;
50
	private closedLastWindow: boolean;
51 52

	private menuUpdater: RunOnceScheduler;
53 54 55 56 57
	private menuGC: RunOnceScheduler;

	// Array to keep menus around so that GC doesn't cause crash as explained in #55347
	// TODO@sbatten Remove this when fixed upstream by Electron
	private oldMenus: Menu[];
58

59
	private menubarMenus: { [id: string]: IMenubarMenu };
60

S
SteVen Batten 已提交
61
	private keybindings: { [commandId: string]: IMenubarKeybinding };
S
SteVen Batten 已提交
62

63
	private fallbackMenuHandlers: { [id: string]: (menuItem: MenuItem, browserWindow: BrowserWindow, event: Event) => void } = {};
64

65
	constructor(
66 67 68
		@IUpdateService private readonly updateService: IUpdateService,
		@IConfigurationService private readonly configurationService: IConfigurationService,
		@IWindowsMainService private readonly windowsMainService: IWindowsMainService,
69
		@IEnvironmentService private readonly environmentService: INativeEnvironmentService,
70
		@ITelemetryService private readonly telemetryService: ITelemetryService,
71
		@IWorkspacesHistoryMainService private readonly workspacesHistoryMainService: IWorkspacesHistoryMainService,
72
		@IStateService private readonly stateService: IStateService,
73
		@ILifecycleMainService private readonly lifecycleMainService: ILifecycleMainService,
74 75
		@ILogService private readonly logService: ILogService,
		@IElectronMainService private readonly electronMainService: IElectronMainService
76 77
	) {
		this.menuUpdater = new RunOnceScheduler(() => this.doUpdateMenu(), 0);
S
SteVen Batten 已提交
78

79 80
		this.menuGC = new RunOnceScheduler(() => { this.oldMenus = []; }, 10000);

81 82
		this.menubarMenus = Object.create(null);
		this.keybindings = Object.create(null);
83

B
Benjamin Pasero 已提交
84
		if (isMacintosh || getTitleBarStyle(this.configurationService, this.environmentService) === 'native') {
85 86
			this.restoreCachedMenubarData();
		}
87 88

		this.addFallbackHandlers();
89

90 91
		this.closedLastWindow = false;

92 93
		this.oldMenus = [];

94 95 96 97 98
		this.install();

		this.registerListeners();
	}

99 100 101 102 103 104 105 106 107 108 109 110 111
	private restoreCachedMenubarData() {
		const menubarData = this.stateService.getItem<IMenubarData>(Menubar.lastKnownMenubarStorageKey);
		if (menubarData) {
			if (menubarData.menus) {
				this.menubarMenus = menubarData.menus;
			}

			if (menubarData.keybindings) {
				this.keybindings = menubarData.keybindings;
			}
		}
	}

112
	private addFallbackHandlers(): void {
113

114
		// File Menu Items
115 116
		this.fallbackMenuHandlers['workbench.action.files.newUntitledFile'] = () => this.windowsMainService.openEmptyWindow(OpenContext.MENU);
		this.fallbackMenuHandlers['workbench.action.newWindow'] = () => this.windowsMainService.openEmptyWindow(OpenContext.MENU);
117 118
		this.fallbackMenuHandlers['workbench.action.files.openFileFolder'] = (menuItem, win, event) => this.electronMainService.pickFileFolderAndOpen(undefined, { forceNewWindow: this.isOptionClick(event), telemetryExtraData: { from: telemetryFrom } });
		this.fallbackMenuHandlers['workbench.action.openWorkspace'] = (menuItem, win, event) => this.electronMainService.pickWorkspaceAndOpen(undefined, { forceNewWindow: this.isOptionClick(event), telemetryExtraData: { from: telemetryFrom } });
119 120

		// Recent Menu Items
121
		this.fallbackMenuHandlers['workbench.action.clearRecentFiles'] = () => this.workspacesHistoryMainService.clearRecentlyOpened();
122 123

		// Help Menu Items
124 125 126
		const twitterUrl = product.twitterUrl;
		if (twitterUrl) {
			this.fallbackMenuHandlers['workbench.action.openTwitterUrl'] = () => this.openUrl(twitterUrl, 'openTwitterUrl');
127 128
		}

129 130 131
		const requestFeatureUrl = product.requestFeatureUrl;
		if (requestFeatureUrl) {
			this.fallbackMenuHandlers['workbench.action.openRequestFeatureUrl'] = () => this.openUrl(requestFeatureUrl, 'openUserVoiceUrl');
132 133
		}

134 135 136
		const reportIssueUrl = product.reportIssueUrl;
		if (reportIssueUrl) {
			this.fallbackMenuHandlers['workbench.action.openIssueReporter'] = () => this.openUrl(reportIssueUrl, 'openReportIssues');
137 138
		}

139 140
		const licenseUrl = product.licenseUrl;
		if (licenseUrl) {
S
SteVen Batten 已提交
141
			this.fallbackMenuHandlers['workbench.action.openLicenseUrl'] = () => {
142
				if (language) {
143 144
					const queryArgChar = licenseUrl.indexOf('?') > 0 ? '&' : '?';
					this.openUrl(`${licenseUrl}${queryArgChar}lang=${language}`, 'openLicenseUrl');
145
				} else {
146
					this.openUrl(licenseUrl, 'openLicenseUrl');
147 148 149 150
				}
			};
		}

151 152
		const privacyStatementUrl = product.privacyStatementUrl;
		if (privacyStatementUrl && licenseUrl) {
153 154
			this.fallbackMenuHandlers['workbench.action.openPrivacyStatementUrl'] = () => {
				if (language) {
155 156
					const queryArgChar = licenseUrl.indexOf('?') > 0 ? '&' : '?';
					this.openUrl(`${privacyStatementUrl}${queryArgChar}lang=${language}`, 'openPrivacyStatement');
157
				} else {
158
					this.openUrl(privacyStatementUrl, 'openPrivacyStatement');
159 160 161 162 163
				}
			};
		}
	}

164 165
	private registerListeners(): void {
		// Keep flag when app quits
166
		this.lifecycleMainService.onWillShutdown(() => this.willShutdown = true);
167 168

		// // Listen to some events from window service to update menu
169
		this.windowsMainService.onWindowsCountChanged(e => this.onWindowsCountChanged(e));
170 171 172 173 174 175 176
	}

	private get currentEnableMenuBarMnemonics(): boolean {
		let enableMenuBarMnemonics = this.configurationService.getValue<boolean>('window.enableMenuBarMnemonics');
		if (typeof enableMenuBarMnemonics !== 'boolean') {
			enableMenuBarMnemonics = true;
		}
177

178 179 180 181
		return enableMenuBarMnemonics;
	}

	private get currentEnableNativeTabs(): boolean {
S
SteVen Batten 已提交
182 183 184 185
		if (!isMacintosh) {
			return false;
		}

186 187 188 189 190 191 192
		let enableNativeTabs = this.configurationService.getValue<boolean>('window.nativeTabs');
		if (typeof enableNativeTabs !== 'boolean') {
			enableNativeTabs = false;
		}
		return enableNativeTabs;
	}

193 194 195
	updateMenu(menubarData: IMenubarData, windowId: number) {
		this.menubarMenus = menubarData.menus;
		this.keybindings = menubarData.keybindings;
S
SteVen Batten 已提交
196

197
		// Save off new menu and keybindings
198
		this.stateService.setItem(Menubar.lastKnownMenubarStorageKey, menubarData);
199

200 201 202 203 204 205 206 207 208 209 210 211 212 213 214
		this.scheduleUpdateMenu();
	}


	private scheduleUpdateMenu(): void {
		this.menuUpdater.schedule(); // buffer multiple attempts to update the menu
	}

	private doUpdateMenu(): void {

		// Due to limitations in Electron, it is not possible to update menu items dynamically. The suggested
		// workaround from Electron is to set the application menu again.
		// See also https://github.com/electron/electron/issues/846
		//
		// Run delayed to prevent updating menu while it is open
B
Benjamin Pasero 已提交
215
		if (!this.willShutdown) {
216
			setTimeout(() => {
B
Benjamin Pasero 已提交
217
				if (!this.willShutdown) {
218 219 220 221 222 223
					this.install();
				}
			}, 10 /* delay this because there is an issue with updating a menu when it is open */);
		}
	}

224 225 226 227 228 229 230
	private onWindowsCountChanged(e: IWindowsCountChangedEvent): void {
		if (!isMacintosh) {
			return;
		}

		// Update menu if window count goes from N > 0 or 0 > N to update menu item enablement
		if ((e.oldCount === 0 && e.newCount > 0) || (e.oldCount > 0 && e.newCount === 0)) {
231
			this.closedLastWindow = e.newCount === 0;
232 233 234
			this.scheduleUpdateMenu();
		}
	}
235 236

	private install(): void {
237 238 239 240 241 242
		// Store old menu in our array to avoid GC to collect the menu and crash. See #55347
		// TODO@sbatten Remove this when fixed upstream by Electron
		const oldMenu = Menu.getApplicationMenu();
		if (oldMenu) {
			this.oldMenus.push(oldMenu);
		}
243

244 245 246 247 248 249 250
		// If we don't have a menu yet, set it to null to avoid the electron menu.
		// This should only happen on the first launch ever
		if (Object.keys(this.menubarMenus).length === 0) {
			Menu.setApplicationMenu(isMacintosh ? new Menu() : null);
			return;
		}

251 252 253 254
		// Menus
		const menubar = new Menu();

		// Mac: Application
255
		let macApplicationMenuItem: MenuItem;
256 257 258 259
		if (isMacintosh) {
			const applicationMenu = new Menu();
			macApplicationMenuItem = new MenuItem({ label: product.nameShort, submenu: applicationMenu });
			this.setMacApplicationMenu(applicationMenu);
260
			menubar.append(macApplicationMenuItem);
261 262
		}

263
		// Mac: Dock
264 265 266 267
		if (isMacintosh && !this.appMenuInstalled) {
			this.appMenuInstalled = true;

			const dockMenu = new Menu();
268
			dockMenu.append(new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'miNewWindow', comment: ['&& denotes a mnemonic'] }, "New &&Window")), click: () => this.windowsMainService.openEmptyWindow(OpenContext.DOCK) }));
269 270 271 272

			app.dock.setMenu(dockMenu);
		}

273 274 275 276
		// File
		const fileMenu = new Menu();
		const fileMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mFile', comment: ['&& denotes a mnemonic'] }, "&&File")), submenu: fileMenu });

277 278
		this.setMenuById(fileMenu, 'File');
		menubar.append(fileMenuItem);
279

280 281 282 283
		// Edit
		const editMenu = new Menu();
		const editMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mEdit', comment: ['&& denotes a mnemonic'] }, "&&Edit")), submenu: editMenu });

284 285
		this.setMenuById(editMenu, 'Edit');
		menubar.append(editMenuItem);
286 287 288 289

		// Selection
		const selectionMenu = new Menu();
		const selectionMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mSelection', comment: ['&& denotes a mnemonic'] }, "&&Selection")), submenu: selectionMenu });
290

291 292
		this.setMenuById(selectionMenu, 'Selection');
		menubar.append(selectionMenuItem);
293 294

		// View
295 296 297
		const viewMenu = new Menu();
		const viewMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mView', comment: ['&& denotes a mnemonic'] }, "&&View")), submenu: viewMenu });

298 299
		this.setMenuById(viewMenu, 'View');
		menubar.append(viewMenuItem);
300

301
		// Go
302 303
		const gotoMenu = new Menu();
		const gotoMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mGoto', comment: ['&& denotes a mnemonic'] }, "&&Go")), submenu: gotoMenu });
304

305 306
		this.setMenuById(gotoMenu, 'Go');
		menubar.append(gotoMenuItem);
307

308 309
		// Debug
		const debugMenu = new Menu();
310
		const debugMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mRun', comment: ['&& denotes a mnemonic'] }, "&&Run")), submenu: debugMenu });
311

312
		this.setMenuById(debugMenu, 'Run');
313
		menubar.append(debugMenuItem);
314

315 316 317
		// Terminal
		const terminalMenu = new Menu();
		const terminalMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mTerminal', comment: ['&& denotes a mnemonic'] }, "&&Terminal")), submenu: terminalMenu });
318

319 320
		this.setMenuById(terminalMenu, 'Terminal');
		menubar.append(terminalMenuItem);
321

322
		// Mac: Window
323
		let macWindowMenuItem: MenuItem | undefined;
S
SteVen Batten 已提交
324
		if (this.shouldDrawMenu('Window')) {
325 326 327 328
			const windowMenu = new Menu();
			macWindowMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize('mWindow', "Window")), submenu: windowMenu, role: 'window' });
			this.setMacWindowMenu(windowMenu);
		}
329 330 331 332 333

		if (macWindowMenuItem) {
			menubar.append(macWindowMenuItem);
		}

334 335 336 337
		// Help
		const helpMenu = new Menu();
		const helpMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mHelp', comment: ['&& denotes a mnemonic'] }, "&&Help")), submenu: helpMenu, role: 'help' });

338 339
		this.setMenuById(helpMenu, 'Help');
		menubar.append(helpMenuItem);
340

S
SteVen Batten 已提交
341 342 343 344 345
		if (menubar.items && menubar.items.length > 0) {
			Menu.setApplicationMenu(menubar);
		} else {
			Menu.setApplicationMenu(null);
		}
346 347 348

		// Dispose of older menus after some time
		this.menuGC.schedule();
349 350
	}

351
	private setMacApplicationMenu(macApplicationMenu: Menu): void {
352
		const about = this.createMenuItem(nls.localize('mAbout', "About {0}", product.nameLong), 'workbench.action.showAboutDialog');
353
		const checkForUpdates = this.getUpdateMenuItems();
S
SteVen Batten 已提交
354 355 356 357 358 359 360 361

		let preferences;
		if (this.shouldDrawMenu('Preferences')) {
			const preferencesMenu = new Menu();
			this.setMenuById(preferencesMenu, 'Preferences');
			preferences = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'miPreferences', comment: ['&& denotes a mnemonic'] }, "&&Preferences")), submenu: preferencesMenu });
		}

362 363 364
		const servicesMenu = new Menu();
		const services = new MenuItem({ label: nls.localize('mServices', "Services"), role: 'services', submenu: servicesMenu });
		const hide = new MenuItem({ label: nls.localize('mHide', "Hide {0}", product.nameLong), role: 'hide', accelerator: 'Command+H' });
365
		const hideOthers = new MenuItem({ label: nls.localize('mHideOthers', "Hide Others"), role: 'hideOthers', accelerator: 'Command+Alt+H' });
366 367 368
		const showAll = new MenuItem({ label: nls.localize('mShowAll', "Show All"), role: 'unhide' });
		const quit = new MenuItem(this.likeAction('workbench.action.quit', {
			label: nls.localize('miQuit', "Quit {0}", product.nameLong), click: () => {
369
				const lastActiveWindow = this.windowsMainService.getLastActiveWindow();
B
Benjamin Pasero 已提交
370
				if (
B
Benjamin Pasero 已提交
371 372 373
					this.windowsMainService.getWindowCount() === 0 || 	// allow to quit when no more windows are open
					!!BrowserWindow.getFocusedWindow() ||				// allow to quit when window has focus (fix for https://github.com/Microsoft/vscode/issues/39191)
					lastActiveWindow?.isMinimized()						// allow to quit when window has no focus but is minimized (https://github.com/Microsoft/vscode/issues/63000)
B
Benjamin Pasero 已提交
374
				) {
375
					this.electronMainService.quit(undefined);
376 377 378 379 380 381
				}
			}
		}));

		const actions = [about];
		actions.push(...checkForUpdates);
S
SteVen Batten 已提交
382 383 384 385 386 387 388 389

		if (preferences) {
			actions.push(...[
				__separator__(),
				preferences
			]);
		}

390 391 392 393 394 395 396 397 398 399 400 401 402 403
		actions.push(...[
			__separator__(),
			services,
			__separator__(),
			hide,
			hideOthers,
			showAll,
			__separator__(),
			quit
		]);

		actions.forEach(i => macApplicationMenu.append(i));
	}

404
	private shouldDrawMenu(menuId: string): boolean {
S
SteVen Batten 已提交
405
		// We need to draw an empty menu to override the electron default
B
Benjamin Pasero 已提交
406
		if (!isMacintosh && getTitleBarStyle(this.configurationService, this.environmentService) === 'custom') {
S
SteVen Batten 已提交
407 408 409
			return false;
		}

410 411 412
		switch (menuId) {
			case 'File':
			case 'Help':
S
SteVen Batten 已提交
413
				if (isMacintosh) {
414 415
					return (this.windowsMainService.getWindowCount() === 0 && this.closedLastWindow) || (!!this.menubarMenus && !!this.menubarMenus[menuId]);
				}
S
SteVen Batten 已提交
416

417 418 419
			case 'Window':
				if (isMacintosh) {
					return (this.windowsMainService.getWindowCount() === 0 && this.closedLastWindow) || !!this.menubarMenus;
S
SteVen Batten 已提交
420
				}
S
SteVen Batten 已提交
421

422
			default:
423
				return this.windowsMainService.getWindowCount() > 0 && (!!this.menubarMenus && !!this.menubarMenus[menuId]);
424 425 426 427
		}
	}


428
	private setMenu(menu: Menu, items: Array<MenubarMenuItem>) {
429 430
		items.forEach((item: MenubarMenuItem) => {
			if (isMenubarMenuItemSeparator(item)) {
431
				menu.append(__separator__());
432 433 434 435 436
			} else if (isMenubarMenuItemSubmenu(item)) {
				const submenu = new Menu();
				const submenuItem = new MenuItem({ label: this.mnemonicLabel(item.label), submenu: submenu });
				this.setMenu(submenu, item.submenu.items);
				menu.append(submenuItem);
437
			} else if (isMenubarMenuItemUriAction(item)) {
438
				menu.append(this.createOpenRecentMenuItem(item.uri, item.label, item.id));
439
			} else if (isMenubarMenuItemAction(item)) {
440
				if (item.id === 'workbench.action.showAboutDialog') {
S
SteVen Batten 已提交
441
					this.insertCheckForUpdatesItems(menu);
442 443
				}

444 445 446 447 448 449 450 451 452
				if (isMacintosh) {
					if (this.windowsMainService.getWindowCount() === 0 && this.closedLastWindow) {
						// In the fallback scenario, we are either disabled or using a fallback handler
						if (this.fallbackMenuHandlers[item.id]) {
							menu.append(new MenuItem(this.likeAction(item.id, { label: this.mnemonicLabel(item.label), click: this.fallbackMenuHandlers[item.id] })));
						} else {
							menu.append(this.createMenuItem(item.label, item.id, false, item.checked));
						}
					} else {
453
						menu.append(this.createMenuItem(item.label, item.id, item.enabled === false ? false : true, !!item.checked));
454 455
					}
				} else {
456
					menu.append(this.createMenuItem(item.label, item.id, item.enabled === false ? false : true, !!item.checked));
457
				}
458
			}
459 460 461
		});
	}

462
	private setMenuById(menu: Menu, menuId: string): void {
463
		if (this.menubarMenus && this.menubarMenus[menuId]) {
S
SteVen Batten 已提交
464 465
			this.setMenu(menu, this.menubarMenus[menuId].items);
		}
466 467
	}

468
	private insertCheckForUpdatesItems(menu: Menu) {
S
SteVen Batten 已提交
469 470 471 472 473 474 475
		const updateItems = this.getUpdateMenuItems();
		if (updateItems.length) {
			updateItems.forEach(i => menu.append(i));
			menu.append(__separator__());
		}
	}

476
	private createOpenRecentMenuItem(uri: URI, label: string, commandId: string): MenuItem {
477
		const revivedUri = URI.revive(uri);
478
		const openable: IWindowOpenable =
M
Martin Aeschlimann 已提交
479 480
			(commandId === 'openRecentFile') ? { fileUri: revivedUri } :
				(commandId === 'openRecentWorkspace') ? { workspaceUri: revivedUri } : { folderUri: revivedUri };
481 482 483 484 485 486 487 488

		return new MenuItem(this.likeAction(commandId, {
			label,
			click: (menuItem, win, event) => {
				const openInNewWindow = this.isOptionClick(event);
				const success = this.windowsMainService.open({
					context: OpenContext.MENU,
					cli: this.environmentService.args,
489
					urisToOpen: [openable],
490 491
					forceNewWindow: openInNewWindow,
					gotoLineMode: false
492 493 494
				}).length > 0;

				if (!success) {
B
Benjamin Pasero 已提交
495
					this.workspacesHistoryMainService.removeRecentlyOpened([revivedUri]);
496 497 498 499 500
				}
			}
		}, false));
	}

501
	private isOptionClick(event: KeyboardEvent): boolean {
M
Matt Bierner 已提交
502
		return !!(event && ((!isMacintosh && (event.ctrlKey || event.shiftKey)) || (isMacintosh && (event.metaKey || event.altKey))));
503 504
	}

505 506
	private createRoleMenuItem(label: string, commandId: string, role: any): MenuItem {
		const options: MenuItemConstructorOptions = {
507 508 509 510 511 512 513 514
			label: this.mnemonicLabel(label),
			role,
			enabled: true
		};

		return new MenuItem(this.withKeybinding(commandId, options));
	}

515
	private setMacWindowMenu(macWindowMenu: Menu): void {
516 517 518 519 520
		const minimize = new MenuItem({ label: nls.localize('mMinimize', "Minimize"), role: 'minimize', accelerator: 'Command+M', enabled: this.windowsMainService.getWindowCount() > 0 });
		const zoom = new MenuItem({ label: nls.localize('mZoom', "Zoom"), role: 'zoom', enabled: this.windowsMainService.getWindowCount() > 0 });
		const bringAllToFront = new MenuItem({ label: nls.localize('mBringToFront', "Bring All to Front"), role: 'front', enabled: this.windowsMainService.getWindowCount() > 0 });
		const switchWindow = this.createMenuItem(nls.localize({ key: 'miSwitchWindow', comment: ['&& denotes a mnemonic'] }, "Switch &&Window..."), 'workbench.action.switchWindow');

521
		const nativeTabMenuItems: MenuItem[] = [];
522
		if (this.currentEnableNativeTabs) {
523 524
			nativeTabMenuItems.push(__separator__());

B
Benjamin Pasero 已提交
525
			nativeTabMenuItems.push(this.createMenuItem(nls.localize('mNewTab', "New Tab"), 'workbench.action.newWindowTab'));
526

B
Benjamin Pasero 已提交
527 528 529 530
			nativeTabMenuItems.push(this.createRoleMenuItem(nls.localize('mShowPreviousTab', "Show Previous Tab"), 'workbench.action.showPreviousWindowTab', 'selectPreviousTab'));
			nativeTabMenuItems.push(this.createRoleMenuItem(nls.localize('mShowNextTab', "Show Next Tab"), 'workbench.action.showNextWindowTab', 'selectNextTab'));
			nativeTabMenuItems.push(this.createRoleMenuItem(nls.localize('mMoveTabToNewWindow', "Move Tab to New Window"), 'workbench.action.moveWindowTabToNewWindow', 'moveTabToNewWindow'));
			nativeTabMenuItems.push(this.createRoleMenuItem(nls.localize('mMergeAllWindows', "Merge All Windows"), 'workbench.action.mergeAllWindowTabs', 'mergeAllWindows'));
531 532 533 534 535
		}

		[
			minimize,
			zoom,
536
			__separator__(),
537 538 539 540 541 542 543
			switchWindow,
			...nativeTabMenuItems,
			__separator__(),
			bringAllToFront
		].forEach(item => macWindowMenu.append(item));
	}

544
	private getUpdateMenuItems(): MenuItem[] {
545 546 547 548 549 550 551 552
		const state = this.updateService.state;

		switch (state.type) {
			case StateType.Uninitialized:
				return [];

			case StateType.Idle:
				return [new MenuItem({
S
SteVen Batten 已提交
553
					label: this.mnemonicLabel(nls.localize('miCheckForUpdates', "Check for &&Updates...")), click: () => setTimeout(() => {
554 555
						this.reportMenuActionTelemetry('CheckForUpdate');

J
Joao Moreno 已提交
556
						const window = this.windowsMainService.getLastActiveWindow();
J
Joao Moreno 已提交
557
						const context = window && `window:${window.id}`; // sessionId
558 559 560 561 562
						this.updateService.checkForUpdates(context);
					}, 0)
				})];

			case StateType.CheckingForUpdates:
563
				return [new MenuItem({ label: nls.localize('miCheckingForUpdates', "Checking for Updates..."), enabled: false })];
564 565 566

			case StateType.AvailableForDownload:
				return [new MenuItem({
S
SteVen Batten 已提交
567
					label: this.mnemonicLabel(nls.localize('miDownloadUpdate', "D&&ownload Available Update")), click: () => {
568 569 570 571 572 573 574 575 576
						this.updateService.downloadUpdate();
					}
				})];

			case StateType.Downloading:
				return [new MenuItem({ label: nls.localize('miDownloadingUpdate', "Downloading Update..."), enabled: false })];

			case StateType.Downloaded:
				return [new MenuItem({
S
SteVen Batten 已提交
577
					label: this.mnemonicLabel(nls.localize('miInstallUpdate', "Install &&Update...")), click: () => {
578 579 580 581 582 583 584 585 586 587
						this.reportMenuActionTelemetry('InstallUpdate');
						this.updateService.applyUpdate();
					}
				})];

			case StateType.Updating:
				return [new MenuItem({ label: nls.localize('miInstallingUpdate', "Installing Update..."), enabled: false })];

			case StateType.Ready:
				return [new MenuItem({
J
Joao Moreno 已提交
588
					label: this.mnemonicLabel(nls.localize('miRestartToUpdate', "Restart to &&Update")), click: () => {
589 590 591 592 593 594 595
						this.reportMenuActionTelemetry('RestartToUpdate');
						this.updateService.quitAndInstall();
					}
				})];
		}
	}

596
	private static _menuItemIsTriggeredViaKeybinding(event: KeyboardEvent, userSettingsLabel: string): boolean {
597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622
		// The event coming in from Electron will inform us only about the modifier keys pressed.
		// The strategy here is to check if the modifier keys match those of the keybinding,
		// since it is highly unlikely to use modifier keys when clicking with the mouse
		if (!userSettingsLabel) {
			// There is no keybinding
			return false;
		}

		let ctrlRequired = /ctrl/.test(userSettingsLabel);
		let shiftRequired = /shift/.test(userSettingsLabel);
		let altRequired = /alt/.test(userSettingsLabel);
		let metaRequired = /cmd/.test(userSettingsLabel) || /super/.test(userSettingsLabel);

		if (!ctrlRequired && !shiftRequired && !altRequired && !metaRequired) {
			// This keybinding does not use any modifier keys, so we cannot use this heuristic
			return false;
		}

		return (
			ctrlRequired === event.ctrlKey
			&& shiftRequired === event.shiftKey
			&& altRequired === event.altKey
			&& metaRequired === event.metaKey
		);
	}

623 624 625
	private createMenuItem(label: string, commandId: string | string[], enabled?: boolean, checked?: boolean): MenuItem;
	private createMenuItem(label: string, click: () => void, enabled?: boolean, checked?: boolean): MenuItem;
	private createMenuItem(arg1: string, arg2: any, arg3?: boolean, arg4?: boolean): MenuItem {
626
		const label = this.mnemonicLabel(arg1);
627
		const click: () => void = (typeof arg2 === 'function') ? arg2 : (menuItem: MenuItem & IMenuItemWithKeybinding, win: BrowserWindow, event: Event) => {
A
Alex Dima 已提交
628
			const userSettingsLabel = menuItem ? menuItem.userSettingsLabel : null;
629 630 631 632 633
			let commandId = arg2;
			if (Array.isArray(arg2)) {
				commandId = this.isOptionClick(event) ? arg2[1] : arg2[0]; // support alternative action if we got multiple action Ids and the option key was pressed while invoking
			}

634
			if (userSettingsLabel && Menubar._menuItemIsTriggeredViaKeybinding(event, userSettingsLabel)) {
635 636 637 638
				this.runActionInRenderer({ type: 'keybinding', userSettingsLabel });
			} else {
				this.runActionInRenderer({ type: 'commandId', commandId });
			}
639 640 641 642
		};
		const enabled = typeof arg3 === 'boolean' ? arg3 : this.windowsMainService.getWindowCount() > 0;
		const checked = typeof arg4 === 'boolean' ? arg4 : false;

643
		const options: MenuItemConstructorOptions = {
644 645 646 647 648 649
			label,
			click,
			enabled
		};

		if (checked) {
650 651
			options.type = 'checkbox';
			options.checked = checked;
652 653
		}

M
Matt Bierner 已提交
654
		let commandId: string | undefined;
655 656 657 658 659 660
		if (typeof arg2 === 'string') {
			commandId = arg2;
		} else if (Array.isArray(arg2)) {
			commandId = arg2[0];
		}

S
SteVen Batten 已提交
661
		if (isMacintosh) {
662

663
			// Add role for special case menu items
S
SteVen Batten 已提交
664
			if (commandId === 'editor.action.clipboardCutAction') {
665
				options.role = 'cut';
S
SteVen Batten 已提交
666
			} else if (commandId === 'editor.action.clipboardCopyAction') {
667
				options.role = 'copy';
S
SteVen Batten 已提交
668
			} else if (commandId === 'editor.action.clipboardPasteAction') {
669
				options.role = 'paste';
S
SteVen Batten 已提交
670
			}
671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688

			// Add context aware click handlers for special case menu items
			if (commandId === 'undo') {
				options.click = this.makeContextAwareClickHandler(click, {
					inDevTools: devTools => devTools.undo(),
					inNoWindow: () => Menu.sendActionToFirstResponder('undo:')
				});
			} else if (commandId === 'redo') {
				options.click = this.makeContextAwareClickHandler(click, {
					inDevTools: devTools => devTools.redo(),
					inNoWindow: () => Menu.sendActionToFirstResponder('redo:')
				});
			} else if (commandId === 'editor.action.selectAll') {
				options.click = this.makeContextAwareClickHandler(click, {
					inDevTools: devTools => devTools.selectAll(),
					inNoWindow: () => Menu.sendActionToFirstResponder('selectAll:')
				});
			}
S
SteVen Batten 已提交
689 690
		}

691 692 693
		return new MenuItem(this.withKeybinding(commandId, options));
	}

694 695
	private makeContextAwareClickHandler(click: () => void, contextSpecificHandlers: IMenuItemClickHandler): () => void {
		return () => {
696

697
			// No Active Window
698
			const activeWindow = BrowserWindow.getFocusedWindow();
699 700 701 702 703
			if (!activeWindow) {
				return contextSpecificHandlers.inNoWindow();
			}

			// DevTools focused
704 705
			if (activeWindow.webContents.isDevToolsFocused()) {
				return contextSpecificHandlers.inDevTools(activeWindow.webContents.devToolsWebContents);
706 707 708 709 710 711 712
			}

			// Finally execute command in Window
			click();
		};
	}

713
	private runActionInRenderer(invocation: IMenuItemInvocation): void {
714 715 716
		// We make sure to not run actions when the window has no focus, this helps
		// for https://github.com/Microsoft/vscode/issues/25907 and specifically for
		// https://github.com/Microsoft/vscode/issues/11928
B
Benjamin Pasero 已提交
717 718
		// Still allow to run when the last active window is minimized though for
		// https://github.com/Microsoft/vscode/issues/63000
719 720
		let activeBrowserWindow = BrowserWindow.getFocusedWindow();
		if (!activeBrowserWindow) {
B
Benjamin Pasero 已提交
721
			const lastActiveWindow = this.windowsMainService.getLastActiveWindow();
B
Benjamin Pasero 已提交
722
			if (lastActiveWindow?.isMinimized()) {
723
				activeBrowserWindow = lastActiveWindow.win;
B
Benjamin Pasero 已提交
724 725 726
			}
		}

727
		const activeWindow = activeBrowserWindow ? this.windowsMainService.getWindowById(activeBrowserWindow.id) : undefined;
728
		if (activeWindow) {
729 730
			this.logService.trace('menubar#runActionInRenderer', invocation);

B
Benjamin Pasero 已提交
731 732
			if (isMacintosh && !this.environmentService.isBuilt && !activeWindow.isReady) {
				if ((invocation.type === 'commandId' && invocation.commandId === 'workbench.action.toggleDevTools') || (invocation.type !== 'commandId' && invocation.userSettingsLabel === 'alt+cmd+i')) {
733 734 735 736 737
					// prevent this action from running twice on macOS (https://github.com/Microsoft/vscode/issues/62719)
					// we already register a keybinding in bootstrap-window.js for opening developer tools in case something
					// goes wrong and that keybinding is only removed when the application has loaded (= window ready).
					return;
				}
B
Benjamin Pasero 已提交
738
			}
B
Benjamin Pasero 已提交
739

B
Benjamin Pasero 已提交
740
			if (invocation.type === 'commandId') {
741
				activeWindow.sendWhenReady('vscode:runAction', { id: invocation.commandId, from: 'menu' } as IRunActionInWindowRequest);
B
Benjamin Pasero 已提交
742
			} else {
743
				activeWindow.sendWhenReady('vscode:runKeybinding', { userSettingsLabel: invocation.userSettingsLabel } as IRunKeybindingInWindowRequest);
B
Benjamin Pasero 已提交
744
			}
745 746
		} else {
			this.logService.trace('menubar#runActionInRenderer: no active window found', invocation);
747 748 749
		}
	}

750
	private withKeybinding(commandId: string | undefined, options: MenuItemConstructorOptions & IMenuItemWithKeybinding): MenuItemConstructorOptions {
M
Matt Bierner 已提交
751
		const binding = typeof commandId === 'string' ? this.keybindings[commandId] : undefined;
752 753

		// Apply binding if there is one
B
Benjamin Pasero 已提交
754
		if (binding?.label) {
755 756

			// if the binding is native, we can just apply it
757
			if (binding.isNative !== false) {
758
				options.accelerator = binding.label;
759
				options.userSettingsLabel = binding.userSettingsLabel;
760 761 762 763
			}

			// the keybinding is not native so we cannot show it as part of the accelerator of
			// the menu item. we fallback to a different strategy so that we always display it
M
Matt Bierner 已提交
764
			else if (typeof options.label === 'string') {
765 766 767 768 769 770 771 772 773 774 775
				const bindingIndex = options.label.indexOf('[');
				if (bindingIndex >= 0) {
					options.label = `${options.label.substr(0, bindingIndex)} [${binding.label}]`;
				} else {
					options.label = `${options.label} [${binding.label}]`;
				}
			}
		}

		// Unset bindings if there is none
		else {
R
Rob Lourens 已提交
776
			options.accelerator = undefined;
777 778 779 780 781
		}

		return options;
	}

782
	private likeAction(commandId: string, options: MenuItemConstructorOptions, setAccelerator = !options.accelerator): MenuItemConstructorOptions {
783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798 799 800 801 802 803
		if (setAccelerator) {
			options = this.withKeybinding(commandId, options);
		}

		const originalClick = options.click;
		options.click = (item, window, event) => {
			this.reportMenuActionTelemetry(commandId);
			if (originalClick) {
				originalClick(item, window, event);
			}
		};

		return options;
	}

	private openUrl(url: string, id: string): void {
		shell.openExternal(url);
		this.reportMenuActionTelemetry(id);
	}

	private reportMenuActionTelemetry(id: string): void {
804
		this.telemetryService.publicLog2<WorkbenchActionExecutedEvent, WorkbenchActionExecutedClassification>('workbenchActionExecuted', { id, from: telemetryFrom });
805 806 807 808 809 810 811
	}

	private mnemonicLabel(label: string): string {
		return baseMnemonicLabel(label, !this.currentEnableMenuBarMnemonics);
	}
}

812
function __separator__(): MenuItem {
813 814
	return new MenuItem({ type: 'separator' });
}