menubar.ts 30.9 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, Event as 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 26 27

const telemetryFrom = 'menu';

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

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

interface IMenuItemWithKeybinding {
	userSettingsLabel?: string;
}

42 43
export class Menubar {

44
	private static readonly lastKnownMenubarStorageKey = 'lastKnownMenubarData';
45

B
Benjamin Pasero 已提交
46
	private willShutdown: boolean;
47
	private appMenuInstalled: boolean;
48
	private closedLastWindow: boolean;
49 50

	private menuUpdater: RunOnceScheduler;
51 52 53 54 55
	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[];
56

57
	private menubarMenus: { [id: string]: IMenubarMenu };
58

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

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

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

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

78 79
		this.menubarMenus = Object.create(null);
		this.keybindings = Object.create(null);
80

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

		this.addFallbackHandlers();
86

87 88
		this.closedLastWindow = false;

89 90
		this.oldMenus = [];

91 92 93 94 95
		this.install();

		this.registerListeners();
	}

96 97 98 99 100 101 102 103 104 105 106 107 108
	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;
			}
		}
	}

109
	private addFallbackHandlers(): void {
110

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

		// Recent Menu Items
118
		this.fallbackMenuHandlers['workbench.action.clearRecentFiles'] = () => this.workspacesHistoryMainService.clearRecentlyOpened();
119 120

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

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

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

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

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

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

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

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

175 176 177 178
		return enableMenuBarMnemonics;
	}

	private get currentEnableNativeTabs(): boolean {
S
SteVen Batten 已提交
179 180 181 182
		if (!isMacintosh) {
			return false;
		}

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

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

194
		// Save off new menu and keybindings
195
		this.stateService.setItem(Menubar.lastKnownMenubarStorageKey, menubarData);
196

197 198 199 200 201 202 203 204 205 206 207 208 209 210 211
		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 已提交
212
		if (!this.willShutdown) {
213
			setTimeout(() => {
B
Benjamin Pasero 已提交
214
				if (!this.willShutdown) {
215 216 217 218 219 220
					this.install();
				}
			}, 10 /* delay this because there is an issue with updating a menu when it is open */);
		}
	}

221 222 223 224 225 226 227
	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)) {
228
			this.closedLastWindow = e.newCount === 0;
229 230 231
			this.scheduleUpdateMenu();
		}
	}
232 233

	private install(): void {
234 235 236 237 238 239
		// 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);
		}
240

241 242 243 244 245 246 247
		// 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;
		}

248 249 250 251
		// Menus
		const menubar = new Menu();

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

260
		// Mac: Dock
261 262 263 264
		if (isMacintosh && !this.appMenuInstalled) {
			this.appMenuInstalled = true;

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

			app.dock.setMenu(dockMenu);
		}

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

274 275
		this.setMenuById(fileMenu, 'File');
		menubar.append(fileMenuItem);
276

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

281 282
		this.setMenuById(editMenu, 'Edit');
		menubar.append(editMenuItem);
283 284 285 286

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

288 289
		this.setMenuById(selectionMenu, 'Selection');
		menubar.append(selectionMenuItem);
290 291

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

295 296
		this.setMenuById(viewMenu, 'View');
		menubar.append(viewMenuItem);
297

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

302 303
		this.setMenuById(gotoMenu, 'Go');
		menubar.append(gotoMenuItem);
304

305 306 307 308
		// Debug
		const debugMenu = new Menu();
		const debugMenuItem = new MenuItem({ label: this.mnemonicLabel(nls.localize({ key: 'mDebug', comment: ['&& denotes a mnemonic'] }, "&&Debug")), submenu: debugMenu });

309 310
		this.setMenuById(debugMenu, 'Debug');
		menubar.append(debugMenuItem);
311

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

316 317
		this.setMenuById(terminalMenu, 'Terminal');
		menubar.append(terminalMenuItem);
318

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

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

331 332 333 334
		// 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' });

335 336
		this.setMenuById(helpMenu, 'Help');
		menubar.append(helpMenuItem);
337

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

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

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

		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 });
		}

359 360 361
		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' });
362
		const hideOthers = new MenuItem({ label: nls.localize('mHideOthers', "Hide Others"), role: 'hideothers', accelerator: 'Command+Alt+H' });
363 364 365
		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: () => {
B
Benjamin Pasero 已提交
366
				if (
367 368
					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)
M
Matt Bierner 已提交
369
					this.windowsMainService.getLastActiveWindow()!.isMinimized()	// allow to quit when window has no focus but is minimized (https://github.com/Microsoft/vscode/issues/63000)
B
Benjamin Pasero 已提交
370 371
				) {
					this.windowsMainService.quit();
372 373 374 375 376 377
				}
			}
		}));

		const actions = [about];
		actions.push(...checkForUpdates);
S
SteVen Batten 已提交
378 379 380 381 382 383 384 385

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

386 387 388 389 390 391 392 393 394 395 396 397 398 399
		actions.push(...[
			__separator__(),
			services,
			__separator__(),
			hide,
			hideOthers,
			showAll,
			__separator__(),
			quit
		]);

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

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

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

413 414 415
			case 'Window':
				if (isMacintosh) {
					return (this.windowsMainService.getWindowCount() === 0 && this.closedLastWindow) || !!this.menubarMenus;
S
SteVen Batten 已提交
416
				}
S
SteVen Batten 已提交
417

418
			default:
419
				return this.windowsMainService.getWindowCount() > 0 && (!!this.menubarMenus && !!this.menubarMenus[menuId]);
420 421 422 423
		}
	}


424
	private setMenu(menu: Menu, items: Array<MenubarMenuItem>) {
425 426
		items.forEach((item: MenubarMenuItem) => {
			if (isMenubarMenuItemSeparator(item)) {
427
				menu.append(__separator__());
428 429 430 431 432
			} 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);
433
			} else if (isMenubarMenuItemUriAction(item)) {
434
				menu.append(this.createOpenRecentMenuItem(item.uri, item.label, item.id));
435
			} else if (isMenubarMenuItemAction(item)) {
436
				if (item.id === 'workbench.action.showAboutDialog') {
S
SteVen Batten 已提交
437
					this.insertCheckForUpdatesItems(menu);
438 439
				}

440 441 442 443 444 445 446 447 448
				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 {
449
						menu.append(this.createMenuItem(item.label, item.id, item.enabled === false ? false : true, !!item.checked));
450 451
					}
				} else {
452
					menu.append(this.createMenuItem(item.label, item.id, item.enabled === false ? false : true, !!item.checked));
453
				}
454
			}
455 456 457
		});
	}

458
	private setMenuById(menu: Menu, menuId: string): void {
459
		if (this.menubarMenus && this.menubarMenus[menuId]) {
S
SteVen Batten 已提交
460 461
			this.setMenu(menu, this.menubarMenus[menuId].items);
		}
462 463
	}

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

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

		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,
485
					urisToOpen: [openable],
486 487
					forceNewWindow: openInNewWindow,
					gotoLineMode: false
488 489 490
				}).length > 0;

				if (!success) {
491
					this.workspacesHistoryMainService.removeFromRecentlyOpened([revivedUri]);
492 493 494 495 496
				}
			}
		}, false));
	}

497
	private isOptionClick(event: KeyboardEvent): boolean {
M
Matt Bierner 已提交
498
		return !!(event && ((!isMacintosh && (event.ctrlKey || event.shiftKey)) || (isMacintosh && (event.metaKey || event.altKey))));
499 500
	}

501 502
	private createRoleMenuItem(label: string, commandId: string, role: any): MenuItem {
		const options: MenuItemConstructorOptions = {
503 504 505 506 507 508 509 510
			label: this.mnemonicLabel(label),
			role,
			enabled: true
		};

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

511
	private setMacWindowMenu(macWindowMenu: Menu): void {
512 513 514 515 516
		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');

517
		const nativeTabMenuItems: MenuItem[] = [];
518
		if (this.currentEnableNativeTabs) {
519 520
			nativeTabMenuItems.push(__separator__());

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

B
Benjamin Pasero 已提交
523 524 525 526
			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'));
527 528 529 530 531 532 533 534 535 536 537 538
		}

		[
			minimize,
			zoom,
			switchWindow,
			...nativeTabMenuItems,
			__separator__(),
			bringAllToFront
		].forEach(item => macWindowMenu.append(item));
	}

539
	private getUpdateMenuItems(): MenuItem[] {
540 541 542 543 544 545 546 547
		const state = this.updateService.state;

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

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

551
						const focusedWindow = BrowserWindow.getFocusedWindow();
552 553 554 555 556 557
						const context = focusedWindow ? { windowId: focusedWindow.id } : null;
						this.updateService.checkForUpdates(context);
					}, 0)
				})];

			case StateType.CheckingForUpdates:
558
				return [new MenuItem({ label: nls.localize('miCheckingForUpdates', "Checking for Updates..."), enabled: false })];
559 560 561

			case StateType.AvailableForDownload:
				return [new MenuItem({
S
SteVen Batten 已提交
562
					label: this.mnemonicLabel(nls.localize('miDownloadUpdate', "D&&ownload Available Update")), click: () => {
563 564 565 566 567 568 569 570 571
						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 已提交
572
					label: this.mnemonicLabel(nls.localize('miInstallUpdate', "Install &&Update...")), click: () => {
573 574 575 576 577 578 579 580 581 582
						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 已提交
583
					label: this.mnemonicLabel(nls.localize('miRestartToUpdate', "Restart to &&Update")), click: () => {
584 585 586 587 588 589 590
						this.reportMenuActionTelemetry('RestartToUpdate');
						this.updateService.quitAndInstall();
					}
				})];
		}
	}

591
	private static _menuItemIsTriggeredViaKeybinding(event: KeyboardEvent, userSettingsLabel: string): boolean {
592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617
		// 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
		);
	}

618 619 620
	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 {
621
		const label = this.mnemonicLabel(arg1);
622
		const click: () => void = (typeof arg2 === 'function') ? arg2 : (menuItem: MenuItem & IMenuItemWithKeybinding, win: BrowserWindow, event: Event) => {
A
Alex Dima 已提交
623
			const userSettingsLabel = menuItem ? menuItem.userSettingsLabel : null;
624 625 626 627 628
			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
			}

629
			if (userSettingsLabel && Menubar._menuItemIsTriggeredViaKeybinding(event, userSettingsLabel)) {
630 631 632 633
				this.runActionInRenderer({ type: 'keybinding', userSettingsLabel });
			} else {
				this.runActionInRenderer({ type: 'commandId', commandId });
			}
634 635 636 637
		};
		const enabled = typeof arg3 === 'boolean' ? arg3 : this.windowsMainService.getWindowCount() > 0;
		const checked = typeof arg4 === 'boolean' ? arg4 : false;

638
		const options: MenuItemConstructorOptions = {
639 640 641 642 643 644
			label,
			click,
			enabled
		};

		if (checked) {
645 646
			options.type = 'checkbox';
			options.checked = checked;
647 648
		}

M
Matt Bierner 已提交
649
		let commandId: string | undefined;
650 651 652 653 654 655
		if (typeof arg2 === 'string') {
			commandId = arg2;
		} else if (Array.isArray(arg2)) {
			commandId = arg2[0];
		}

S
SteVen Batten 已提交
656
		if (isMacintosh) {
657

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

			// 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 已提交
684 685
		}

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

689 690
	private makeContextAwareClickHandler(click: () => void, contextSpecificHandlers: IMenuItemClickHandler): () => void {
		return () => {
691

692
			// No Active Window
693
			const activeWindow = BrowserWindow.getFocusedWindow();
694 695 696 697 698
			if (!activeWindow) {
				return contextSpecificHandlers.inNoWindow();
			}

			// DevTools focused
699 700
			if (activeWindow.webContents.isDevToolsFocused()) {
				return contextSpecificHandlers.inDevTools(activeWindow.webContents.devToolsWebContents);
701 702 703 704 705 706 707
			}

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

708
	private runActionInRenderer(invocation: IMenuItemInvocation): void {
709 710 711
		// 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 已提交
712 713
		// Still allow to run when the last active window is minimized though for
		// https://github.com/Microsoft/vscode/issues/63000
714 715
		let activeBrowserWindow = BrowserWindow.getFocusedWindow();
		if (!activeBrowserWindow) {
B
Benjamin Pasero 已提交
716
			const lastActiveWindow = this.windowsMainService.getLastActiveWindow();
M
Matt Bierner 已提交
717
			if (lastActiveWindow && lastActiveWindow.isMinimized()) {
718
				activeBrowserWindow = lastActiveWindow.win;
B
Benjamin Pasero 已提交
719 720 721
			}
		}

722
		const activeWindow = activeBrowserWindow ? this.windowsMainService.getWindowById(activeBrowserWindow.id) : undefined;
723
		if (activeWindow) {
724 725
			this.logService.trace('menubar#runActionInRenderer', invocation);

B
Benjamin Pasero 已提交
726 727
			if (isMacintosh && !this.environmentService.isBuilt && !activeWindow.isReady) {
				if ((invocation.type === 'commandId' && invocation.commandId === 'workbench.action.toggleDevTools') || (invocation.type !== 'commandId' && invocation.userSettingsLabel === 'alt+cmd+i')) {
728 729 730 731 732
					// 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 已提交
733
			}
B
Benjamin Pasero 已提交
734

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

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

		// Apply binding if there is one
		if (binding && binding.label) {

			// if the binding is native, we can just apply it
752
			if (binding.isNative !== false) {
753
				options.accelerator = binding.label;
754
				options.userSettingsLabel = binding.userSettingsLabel;
755 756 757 758
			}

			// 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 已提交
759
			else if (typeof options.label === 'string') {
760 761 762 763 764 765 766 767 768 769 770
				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 已提交
771
			options.accelerator = undefined;
772 773 774 775 776
		}

		return options;
	}

777
	private likeAction(commandId: string, options: MenuItemConstructorOptions, setAccelerator = !options.accelerator): MenuItemConstructorOptions {
778 779 780 781 782 783 784 785 786 787 788 789 790 791 792 793 794 795 796 797 798
		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 {
799
		this.telemetryService.publicLog2<WorkbenchActionExecutedEvent, WorkbenchActionExecutedClassification>('workbenchActionExecuted', { id, from: telemetryFrom });
800 801 802 803 804 805 806
	}

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

807
function __separator__(): MenuItem {
808 809
	return new MenuItem({ type: 'separator' });
}