contextmenuService.ts 7.6 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.
 *--------------------------------------------------------------------------------------------*/

J
João Moreno 已提交
6
import { IAction, IActionRunner, ActionRunner, WorkbenchActionExecutedEvent, WorkbenchActionExecutedClassification, Separator } from 'vs/base/common/actions';
7
import * as dom from 'vs/base/browser/dom';
8
import { IContextMenuService, IContextViewService } from 'vs/platform/contextview/browser/contextView';
J
Johannes Rieken 已提交
9 10
import { ITelemetryService } from 'vs/platform/telemetry/common/telemetry';
import { IKeybindingService } from 'vs/platform/keybinding/common/keybinding';
11
import { getZoomFactor } from 'vs/base/browser/browser';
B
Benjamin Pasero 已提交
12
import { unmnemonicLabel } from 'vs/base/common/labels';
M
Matt Bierner 已提交
13
import { Event, Emitter } from 'vs/base/common/event';
14
import { INotificationService } from 'vs/platform/notification/common/notification';
15
import { IContextMenuDelegate, ContextSubMenu, IContextMenuEvent } from 'vs/base/browser/contextmenu';
16
import { once } from 'vs/base/common/functional';
B
Benjamin Pasero 已提交
17
import { Disposable } from 'vs/base/common/lifecycle';
18
import { IContextMenuItem } from 'vs/base/parts/contextmenu/common/contextmenu';
19
import { popup } from 'vs/base/parts/contextmenu/electron-sandbox/contextmenu';
20 21 22 23 24 25 26
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';
S
SteVen Batten 已提交
27
import { stripCodicons } from 'vs/base/common/codicons';
B
Benjamin Pasero 已提交
28 29

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

31
	declare readonly _serviceBrand: undefined;
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
		}

		// 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 {

66
	declare readonly _serviceBrand: undefined;
67

B
Benjamin Pasero 已提交
68
	private _onDidContextMenu = this._register(new Emitter<void>());
69
	readonly onDidContextMenu: Event<void> = 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
			const menu = this.createMenu(delegate, actions, onHide);
			const anchor = delegate.getAnchor();

B
Benjamin Pasero 已提交
93 94 95
			let x: number;
			let y: number;

96
			const zoom = getZoomFactor();
97
			if (dom.isHTMLElement(anchor)) {
98
				const elementPosition = dom.getDomNodePagePosition(anchor);
99 100 101

				x = elementPosition.left;
				y = elementPosition.top + elementPosition.height;
S
SteVen Batten 已提交
102 103 104 105 106 107 108

				// Shift macOS menus by a few pixels below elements
				// to account for extra padding on top of native menu
				// https://github.com/microsoft/vscode/issues/84231
				if (isMacintosh) {
					y += 4 / zoom;
				}
109
			} else {
B
Benjamin Pasero 已提交
110
				const pos: { x: number; y: number; } = anchor;
111 112
				x = pos.x + 1; /* prevent first item from being selected automatically under mouse */
				y = pos.y;
113
			}
114 115 116 117 118 119 120

			x *= zoom;
			y *= zoom;

			popup(menu, {
				x: Math.floor(x),
				y: Math.floor(y),
R
Rob Lourens 已提交
121
				positioningItem: delegate.autoSelectFirstItem ? 0 : undefined,
122
			}, () => onHide());
123
		}
124
	}
E
Erich Gamma 已提交
125

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

129 130
		return entries.map(entry => this.createMenuItem(delegate, entry, actionRunner, onHide));
	}
I
isidor 已提交
131

132 133 134 135
	private createMenuItem(delegate: IContextMenuDelegate, entry: IAction | ContextSubMenu, actionRunner: IActionRunner, onHide: () => void): IContextMenuItem {

		// Separator
		if (entry instanceof Separator) {
136
			return { type: 'separator' };
137 138 139 140 141
		}

		// Submenu
		if (entry instanceof ContextSubMenu) {
			return {
S
SteVen Batten 已提交
142
				label: unmnemonicLabel(stripCodicons(entry.label)).trim(),
143
				submenu: this.createMenu(delegate, entry.entries, onHide)
144
			};
145 146 147 148
		}

		// Normal Menu Item
		else {
B
Benjamin Pasero 已提交
149 150 151 152 153 154 155 156 157
			let type: 'radio' | 'checkbox' | undefined = undefined;
			if (!!entry.checked) {
				if (typeof delegate.getCheckedActionsRepresentation === 'function') {
					type = delegate.getCheckedActionsRepresentation(entry);
				} else {
					type = 'checkbox';
				}
			}

158
			const item: IContextMenuItem = {
S
SteVen Batten 已提交
159
				label: unmnemonicLabel(stripCodicons(entry.label)).trim(),
B
Benjamin Pasero 已提交
160 161
				checked: !!entry.checked,
				type,
162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183
				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}]`;
184 185
					}
				}
I
isidor 已提交
186 187
			}

188 189
			return item;
		}
I
isidor 已提交
190 191
	}

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

S
SteVen Batten 已提交
195
		const context = delegate.getActionsContext ? delegate.getActionsContext(event) : undefined;
E
Erich Gamma 已提交
196

197 198 199 200 201 202 203 204
		const runnable = actionRunner.run(actionToRun, context);
		if (runnable) {
			try {
				await runnable;
			} catch (error) {
				this.notificationService.error(error);
			}
		}
E
Erich Gamma 已提交
205
	}
206
}
207

B
Benjamin Pasero 已提交
208
registerSingleton(IContextMenuService, ContextMenuService, true);