contextmenuService.ts 7.1 KB
Newer Older
E
Erich Gamma 已提交
1 2 3 4 5
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

6
import { IAction, IActionRunner, ActionRunner, WorkbenchActionExecutedEvent, WorkbenchActionExecutedClassification } from 'vs/base/common/actions';
J
Johannes Rieken 已提交
7
import { Separator } from 'vs/base/browser/ui/actionbar/actionbar';
8
import * as dom from 'vs/base/browser/dom';
9
import { IContextMenuService, IContextViewService } from 'vs/platform/contextview/browser/contextView';
J
Johannes Rieken 已提交
10 11
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding';
12
import { webFrame } from 'electron';
B
Benjamin Pasero 已提交
13
import { unmnemonicLabel } from 'vs/base/common/labels';
M
Matt Bierner 已提交
14
import { Event, Emitter } from 'vs/base/common/event';
15
import { INotificationService } from 'vs/platform/notification/common/notification';
16
import { IContextMenuDelegate, ContextSubMenu, IContextMenuEvent } from 'vs/base/browser/contextmenu';
17
import { once } from 'vs/base/common/functional';
B
Benjamin Pasero 已提交
18
import { Disposable } from 'vs/base/common/lifecycle';
19 20
import { IContextMenuItem } from 'vs/base/parts/contextmenu/common/contextmenu';
import { popup } from 'vs/base/parts/contextmenu/electron-browser/contextmenu';
21 22 23 24 25 26 27
import { getTitleBarStyle } from 'vs/platform/windows/common/windows';
import { isMacintosh } from 'vs/base/common/platform';
import { IConfigurationService } from 'vs/platform/configuration/common/configuration';
import { IEnvironmentService } from 'vs/platform/environment/common/environment';
import { ContextMenuService as HTMLContextMenuService } from 'vs/platform/contextview/browser/contextMenuService';
import { IThemeService } from 'vs/platform/theme/common/themeService';
import { registerSingleton } from 'vs/platform/instantiation/common/extensions';
B
Benjamin Pasero 已提交
28 29

export class ContextMenuService extends Disposable implements IContextMenuService {
E
Erich Gamma 已提交
30

B
Benjamin Pasero 已提交
31
	_serviceBrand: any;
32

33 34 35 36 37 38 39 40 41 42 43
	get onDidContextMenu(): Event<void> { return this.impl.onDidContextMenu; }

	private impl: IContextMenuService;

	constructor(
		@INotificationService notificationService: INotificationService,
		@ITelemetryService telemetryService: ITelemetryService,
		@IKeybindingService keybindingService: IKeybindingService,
		@IConfigurationService configurationService: IConfigurationService,
		@IEnvironmentService environmentService: IEnvironmentService,
		@IContextViewService contextViewService: IContextViewService,
B
Benjamin Pasero 已提交
44
		@IThemeService themeService: IThemeService
45 46 47 48 49
	) {
		super();

		// Custom context menu: Linux/Windows if custom title is enabled
		if (!isMacintosh && getTitleBarStyle(configurationService, environmentService) === 'custom') {
B
Benjamin Pasero 已提交
50
			this.impl = new HTMLContextMenuService(telemetryService, notificationService, contextViewService, keybindingService, themeService);
51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67
		}

		// Native context menu: otherwise
		else {
			this.impl = new NativeContextMenuService(notificationService, telemetryService, keybindingService);
		}
	}

	showContextMenu(delegate: IContextMenuDelegate): void {
		this.impl.showContextMenu(delegate);
	}
}

class NativeContextMenuService extends Disposable implements IContextMenuService {

	_serviceBrand: any;

B
Benjamin Pasero 已提交
68 69
	private _onDidContextMenu = this._register(new Emitter<void>());
	get onDidContextMenu(): Event<void> { return this._onDidContextMenu.event; }
70

71
	constructor(
72 73 74
		@INotificationService private readonly notificationService: INotificationService,
		@ITelemetryService private readonly telemetryService: ITelemetryService,
		@IKeybindingService private readonly keybindingService: IKeybindingService
75
	) {
B
Benjamin Pasero 已提交
76
		super();
E
Erich Gamma 已提交
77 78
	}

B
Benjamin Pasero 已提交
79
	showContextMenu(delegate: IContextMenuDelegate): void {
80 81 82 83
		const actions = delegate.getActions();
		if (actions.length) {
			const onHide = once(() => {
				if (delegate.onHide) {
M
Matt Bierner 已提交
84
					delegate.onHide(false);
85 86
				}

87 88
				this._onDidContextMenu.fire();
			});
89

90 91 92 93 94 95 96 97 98 99
			const menu = this.createMenu(delegate, actions, onHide);
			const anchor = delegate.getAnchor();
			let x: number, y: number;

			if (dom.isHTMLElement(anchor)) {
				let elementPosition = dom.getDomNodePagePosition(anchor);

				x = elementPosition.left;
				y = elementPosition.top + elementPosition.height;
			} else {
B
Benjamin Pasero 已提交
100
				const pos: { x: number; y: number; } = anchor;
101 102
				x = pos.x + 1; /* prevent first item from being selected automatically under mouse */
				y = pos.y;
103
			}
104 105 106 107 108 109 110 111

			let zoom = webFrame.getZoomFactor();
			x *= zoom;
			y *= zoom;

			popup(menu, {
				x: Math.floor(x),
				y: Math.floor(y),
R
Rob Lourens 已提交
112
				positioningItem: delegate.autoSelectFirstItem ? 0 : undefined,
113 114 115
				onHide: () => onHide()
			});
		}
116
	}
E
Erich Gamma 已提交
117

118
	private createMenu(delegate: IContextMenuDelegate, entries: ReadonlyArray<IAction | ContextSubMenu>, onHide: () => void): IContextMenuItem[] {
119
		const actionRunner = delegate.actionRunner || new ActionRunner();
I
isidor 已提交
120

121 122
		return entries.map(entry => this.createMenuItem(delegate, entry, actionRunner, onHide));
	}
I
isidor 已提交
123

124 125 126 127
	private createMenuItem(delegate: IContextMenuDelegate, entry: IAction | ContextSubMenu, actionRunner: IActionRunner, onHide: () => void): IContextMenuItem {

		// Separator
		if (entry instanceof Separator) {
128
			return { type: 'separator' };
129 130 131 132 133 134 135
		}

		// Submenu
		if (entry instanceof ContextSubMenu) {
			return {
				label: unmnemonicLabel(entry.label),
				submenu: this.createMenu(delegate, entry.entries, onHide)
136
			};
137 138 139 140 141 142 143
		}

		// Normal Menu Item
		else {
			const item: IContextMenuItem = {
				label: unmnemonicLabel(entry.label),
				checked: !!entry.checked || !!entry.radio,
R
Rob Lourens 已提交
144
				type: !!entry.checked ? 'checkbox' : !!entry.radio ? 'radio' : undefined,
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166
				enabled: !!entry.enabled,
				click: event => {

					// To preserve pre-electron-2.x behaviour, we first trigger
					// the onHide callback and then the action.
					// Fixes https://github.com/Microsoft/vscode/issues/45601
					onHide();

					// Run action which will close the menu
					this.runAction(actionRunner, entry, delegate, event);
				}
			};

			const keybinding = !!delegate.getKeyBinding ? delegate.getKeyBinding(entry) : this.keybindingService.lookupKeybinding(entry.id);
			if (keybinding) {
				const electronAccelerator = keybinding.getElectronAccelerator();
				if (electronAccelerator) {
					item.accelerator = electronAccelerator;
				} else {
					const label = keybinding.getLabel();
					if (label) {
						item.label = `${item.label} [${label}]`;
167 168
					}
				}
I
isidor 已提交
169 170
			}

171 172
			return item;
		}
I
isidor 已提交
173 174
	}

175
	private async runAction(actionRunner: IActionRunner, actionToRun: IAction, delegate: IContextMenuDelegate, event: IContextMenuEvent): Promise<void> {
176
		this.telemetryService.publicLog2<WorkbenchActionExecutedEvent, WorkbenchActionExecutedClassification>('workbenchActionExecuted', { id: actionToRun.id, from: 'contextMenu' });
E
Erich Gamma 已提交
177

178
		const context = delegate.getActionsContext ? delegate.getActionsContext(event) : event;
E
Erich Gamma 已提交
179

180 181 182 183 184 185 186 187
		const runnable = actionRunner.run(actionToRun, context);
		if (runnable) {
			try {
				await runnable;
			} catch (error) {
				this.notificationService.error(error);
			}
		}
E
Erich Gamma 已提交
188
	}
189
}
190 191

registerSingleton(IContextMenuService, ContextMenuService, true);