123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414 |
- /*---------------------------------------------------------------------------------------------
- * Copyright (c) Microsoft Corporation. All rights reserved.
- * Licensed under the MIT License. See License.txt in the project root for license information.
- *--------------------------------------------------------------------------------------------*/
- var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
- var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
- if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
- else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
- return c > 3 && r && Object.defineProperty(target, key, r), r;
- };
- var __param = (this && this.__param) || function (paramIndex, decorator) {
- return function (target, key) { decorator(target, key, paramIndex); }
- };
- var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
- function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
- return new (P || (P = Promise))(function (resolve, reject) {
- function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
- function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
- function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
- step((generator = generator.apply(thisArg, _arguments || [])).next());
- });
- };
- import { $, addDisposableListener, append, asCSSUrl, EventType, ModifierKeyEmitter, prepend } from '../../../base/browser/dom.js';
- import { StandardKeyboardEvent } from '../../../base/browser/keyboardEvent.js';
- import { ActionViewItem, BaseActionViewItem } from '../../../base/browser/ui/actionbar/actionViewItems.js';
- import { DropdownMenuActionViewItem } from '../../../base/browser/ui/dropdown/dropdownActionViewItem.js';
- import { ActionRunner, Separator, SubmenuAction } from '../../../base/common/actions.js';
- import { UILabelProvider } from '../../../base/common/keybindingLabels.js';
- import { DisposableStore, MutableDisposable, toDisposable } from '../../../base/common/lifecycle.js';
- import { isLinux, isWindows, OS } from '../../../base/common/platform.js';
- import './menuEntryActionViewItem.css';
- import { localize } from '../../../nls.js';
- import { IMenuService, MenuItemAction, SubmenuItemAction } from '../common/actions.js';
- import { IContextKeyService } from '../../contextkey/common/contextkey.js';
- import { IContextMenuService } from '../../contextview/browser/contextView.js';
- import { IInstantiationService } from '../../instantiation/common/instantiation.js';
- import { IKeybindingService } from '../../keybinding/common/keybinding.js';
- import { INotificationService } from '../../notification/common/notification.js';
- import { IStorageService } from '../../storage/common/storage.js';
- import { ThemeIcon } from '../../theme/common/themeService.js';
- export function createAndFillInActionBarActions(menu, options, target, primaryGroup, primaryMaxCount, shouldInlineSubmenu, useSeparatorsInPrimaryActions) {
- const groups = menu.getActions(options);
- const isPrimaryAction = typeof primaryGroup === 'string' ? (actionGroup) => actionGroup === primaryGroup : primaryGroup;
- // Action bars handle alternative actions on their own so the alternative actions should be ignored
- fillInActions(groups, target, false, isPrimaryAction, primaryMaxCount, shouldInlineSubmenu, useSeparatorsInPrimaryActions);
- return asDisposable(groups);
- }
- function asDisposable(groups) {
- const disposables = new DisposableStore();
- for (const [, actions] of groups) {
- for (const action of actions) {
- disposables.add(action);
- }
- }
- return disposables;
- }
- function fillInActions(groups, target, useAlternativeActions, isPrimaryAction = actionGroup => actionGroup === 'navigation', primaryMaxCount = Number.MAX_SAFE_INTEGER, shouldInlineSubmenu = () => false, useSeparatorsInPrimaryActions = false) {
- let primaryBucket;
- let secondaryBucket;
- if (Array.isArray(target)) {
- primaryBucket = target;
- secondaryBucket = target;
- }
- else {
- primaryBucket = target.primary;
- secondaryBucket = target.secondary;
- }
- const submenuInfo = new Set();
- for (const [group, actions] of groups) {
- let target;
- if (isPrimaryAction(group)) {
- target = primaryBucket;
- if (target.length > 0 && useSeparatorsInPrimaryActions) {
- target.push(new Separator());
- }
- }
- else {
- target = secondaryBucket;
- if (target.length > 0) {
- target.push(new Separator());
- }
- }
- for (let action of actions) {
- if (useAlternativeActions) {
- action = action instanceof MenuItemAction && action.alt ? action.alt : action;
- }
- const newLen = target.push(action);
- // keep submenu info for later inlining
- if (action instanceof SubmenuAction) {
- submenuInfo.add({ group, action, index: newLen - 1 });
- }
- }
- }
- // ask the outside if submenu should be inlined or not. only ask when
- // there would be enough space
- for (const { group, action, index } of submenuInfo) {
- const target = isPrimaryAction(group) ? primaryBucket : secondaryBucket;
- // inlining submenus with length 0 or 1 is easy,
- // larger submenus need to be checked with the overall limit
- const submenuActions = action.actions;
- if ((submenuActions.length <= 1 || target.length + submenuActions.length - 2 <= primaryMaxCount) && shouldInlineSubmenu(action, group, target.length)) {
- target.splice(index, 1, ...submenuActions);
- }
- }
- // overflow items from the primary group into the secondary bucket
- if (primaryBucket !== secondaryBucket && primaryBucket.length > primaryMaxCount) {
- const overflow = primaryBucket.splice(primaryMaxCount, primaryBucket.length - primaryMaxCount);
- secondaryBucket.unshift(...overflow, new Separator());
- }
- }
- let MenuEntryActionViewItem = class MenuEntryActionViewItem extends ActionViewItem {
- constructor(_action, options, _keybindingService, _notificationService, _contextKeyService) {
- super(undefined, _action, { icon: !!(_action.class || _action.item.icon), label: !_action.class && !_action.item.icon, draggable: options === null || options === void 0 ? void 0 : options.draggable });
- this._keybindingService = _keybindingService;
- this._notificationService = _notificationService;
- this._contextKeyService = _contextKeyService;
- this._wantsAltCommand = false;
- this._itemClassDispose = this._register(new MutableDisposable());
- this._altKey = ModifierKeyEmitter.getInstance();
- }
- get _menuItemAction() {
- return this._action;
- }
- get _commandAction() {
- return this._wantsAltCommand && this._menuItemAction.alt || this._menuItemAction;
- }
- onClick(event) {
- return __awaiter(this, void 0, void 0, function* () {
- event.preventDefault();
- event.stopPropagation();
- try {
- yield this.actionRunner.run(this._commandAction, this._context);
- }
- catch (err) {
- this._notificationService.error(err);
- }
- });
- }
- render(container) {
- super.render(container);
- container.classList.add('menu-entry');
- this._updateItemClass(this._menuItemAction.item);
- let mouseOver = false;
- let alternativeKeyDown = this._altKey.keyStatus.altKey || ((isWindows || isLinux) && this._altKey.keyStatus.shiftKey);
- const updateAltState = () => {
- const wantsAltCommand = mouseOver && alternativeKeyDown;
- if (wantsAltCommand !== this._wantsAltCommand) {
- this._wantsAltCommand = wantsAltCommand;
- this.updateLabel();
- this.updateTooltip();
- this.updateClass();
- }
- };
- if (this._menuItemAction.alt) {
- this._register(this._altKey.event(value => {
- alternativeKeyDown = value.altKey || ((isWindows || isLinux) && value.shiftKey);
- updateAltState();
- }));
- }
- this._register(addDisposableListener(container, 'mouseleave', _ => {
- mouseOver = false;
- updateAltState();
- }));
- this._register(addDisposableListener(container, 'mouseenter', _ => {
- mouseOver = true;
- updateAltState();
- }));
- }
- updateLabel() {
- if (this.options.label && this.label) {
- this.label.textContent = this._commandAction.label;
- }
- }
- updateTooltip() {
- if (this.label) {
- const keybinding = this._keybindingService.lookupKeybinding(this._commandAction.id, this._contextKeyService);
- const keybindingLabel = keybinding && keybinding.getLabel();
- const tooltip = this._commandAction.tooltip || this._commandAction.label;
- let title = keybindingLabel
- ? localize('titleAndKb', "{0} ({1})", tooltip, keybindingLabel)
- : tooltip;
- if (!this._wantsAltCommand && this._menuItemAction.alt) {
- const altTooltip = this._menuItemAction.alt.tooltip || this._menuItemAction.alt.label;
- const altKeybinding = this._keybindingService.lookupKeybinding(this._menuItemAction.alt.id, this._contextKeyService);
- const altKeybindingLabel = altKeybinding && altKeybinding.getLabel();
- const altTitleSection = altKeybindingLabel
- ? localize('titleAndKb', "{0} ({1})", altTooltip, altKeybindingLabel)
- : altTooltip;
- title += `\n[${UILabelProvider.modifierLabels[OS].altKey}] ${altTitleSection}`;
- }
- this.label.title = title;
- }
- }
- updateClass() {
- if (this.options.icon) {
- if (this._commandAction !== this._menuItemAction) {
- if (this._menuItemAction.alt) {
- this._updateItemClass(this._menuItemAction.alt.item);
- }
- }
- else if (this._menuItemAction.alt) {
- this._updateItemClass(this._menuItemAction.item);
- }
- }
- }
- _updateItemClass(item) {
- var _a;
- this._itemClassDispose.value = undefined;
- const { element, label } = this;
- if (!element || !label) {
- return;
- }
- const icon = this._commandAction.checked && ((_a = item.toggled) === null || _a === void 0 ? void 0 : _a.icon) ? item.toggled.icon : item.icon;
- if (!icon) {
- return;
- }
- if (ThemeIcon.isThemeIcon(icon)) {
- // theme icons
- const iconClasses = ThemeIcon.asClassNameArray(icon);
- label.classList.add(...iconClasses);
- this._itemClassDispose.value = toDisposable(() => {
- label.classList.remove(...iconClasses);
- });
- }
- else {
- // icon path/url
- if (icon.light) {
- label.style.setProperty('--menu-entry-icon-light', asCSSUrl(icon.light));
- }
- if (icon.dark) {
- label.style.setProperty('--menu-entry-icon-dark', asCSSUrl(icon.dark));
- }
- label.classList.add('icon');
- this._itemClassDispose.value = toDisposable(() => {
- label.classList.remove('icon');
- label.style.removeProperty('--menu-entry-icon-light');
- label.style.removeProperty('--menu-entry-icon-dark');
- });
- }
- }
- };
- MenuEntryActionViewItem = __decorate([
- __param(2, IKeybindingService),
- __param(3, INotificationService),
- __param(4, IContextKeyService)
- ], MenuEntryActionViewItem);
- export { MenuEntryActionViewItem };
- let SubmenuEntryActionViewItem = class SubmenuEntryActionViewItem extends DropdownMenuActionViewItem {
- constructor(action, options, contextMenuService) {
- var _a, _b;
- const dropdownOptions = Object.assign({}, options !== null && options !== void 0 ? options : Object.create(null), {
- menuAsChild: (_a = options === null || options === void 0 ? void 0 : options.menuAsChild) !== null && _a !== void 0 ? _a : false,
- classNames: (_b = options === null || options === void 0 ? void 0 : options.classNames) !== null && _b !== void 0 ? _b : (ThemeIcon.isThemeIcon(action.item.icon) ? ThemeIcon.asClassName(action.item.icon) : undefined),
- });
- super(action, { getActions: () => action.actions }, contextMenuService, dropdownOptions);
- }
- render(container) {
- super.render(container);
- if (this.element) {
- container.classList.add('menu-entry');
- const { icon } = this._action.item;
- if (icon && !ThemeIcon.isThemeIcon(icon)) {
- this.element.classList.add('icon');
- if (icon.light) {
- this.element.style.setProperty('--menu-entry-icon-light', asCSSUrl(icon.light));
- }
- if (icon.dark) {
- this.element.style.setProperty('--menu-entry-icon-dark', asCSSUrl(icon.dark));
- }
- }
- }
- }
- };
- SubmenuEntryActionViewItem = __decorate([
- __param(2, IContextMenuService)
- ], SubmenuEntryActionViewItem);
- export { SubmenuEntryActionViewItem };
- let DropdownWithDefaultActionViewItem = class DropdownWithDefaultActionViewItem extends BaseActionViewItem {
- constructor(submenuAction, options, _keybindingService, _notificationService, _contextMenuService, _menuService, _instaService, _storageService) {
- var _a, _b, _c;
- super(null, submenuAction);
- this._keybindingService = _keybindingService;
- this._notificationService = _notificationService;
- this._contextMenuService = _contextMenuService;
- this._menuService = _menuService;
- this._instaService = _instaService;
- this._storageService = _storageService;
- this._container = null;
- this._storageKey = `${submenuAction.item.submenu._debugName}_lastActionId`;
- // determine default action
- let defaultAction;
- let defaultActionId = _storageService.get(this._storageKey, 1 /* WORKSPACE */);
- if (defaultActionId) {
- defaultAction = submenuAction.actions.find(a => defaultActionId === a.id);
- }
- if (!defaultAction) {
- defaultAction = submenuAction.actions[0];
- }
- this._defaultAction = this._instaService.createInstance(MenuEntryActionViewItem, defaultAction, undefined);
- const dropdownOptions = Object.assign({}, options !== null && options !== void 0 ? options : Object.create(null), {
- menuAsChild: (_a = options === null || options === void 0 ? void 0 : options.menuAsChild) !== null && _a !== void 0 ? _a : true,
- classNames: (_b = options === null || options === void 0 ? void 0 : options.classNames) !== null && _b !== void 0 ? _b : ['codicon', 'codicon-chevron-down'],
- actionRunner: (_c = options === null || options === void 0 ? void 0 : options.actionRunner) !== null && _c !== void 0 ? _c : new ActionRunner()
- });
- this._dropdown = new DropdownMenuActionViewItem(submenuAction, submenuAction.actions, this._contextMenuService, dropdownOptions);
- this._dropdown.actionRunner.onDidRun((e) => {
- if (e.action instanceof MenuItemAction) {
- this.update(e.action);
- }
- });
- }
- update(lastAction) {
- this._storageService.store(this._storageKey, lastAction.id, 1 /* WORKSPACE */, 0 /* USER */);
- this._defaultAction.dispose();
- this._defaultAction = this._instaService.createInstance(MenuEntryActionViewItem, lastAction, undefined);
- this._defaultAction.actionRunner = new class extends ActionRunner {
- runAction(action, context) {
- return __awaiter(this, void 0, void 0, function* () {
- yield action.run(undefined);
- });
- }
- }();
- if (this._container) {
- this._defaultAction.render(prepend(this._container, $('.action-container')));
- }
- }
- setActionContext(newContext) {
- super.setActionContext(newContext);
- this._defaultAction.setActionContext(newContext);
- this._dropdown.setActionContext(newContext);
- }
- render(container) {
- this._container = container;
- super.render(this._container);
- this._container.classList.add('monaco-dropdown-with-default');
- const primaryContainer = $('.action-container');
- this._defaultAction.render(append(this._container, primaryContainer));
- this._register(addDisposableListener(primaryContainer, EventType.KEY_DOWN, (e) => {
- const event = new StandardKeyboardEvent(e);
- if (event.equals(17 /* RightArrow */)) {
- this._defaultAction.element.tabIndex = -1;
- this._dropdown.focus();
- event.stopPropagation();
- }
- }));
- const dropdownContainer = $('.dropdown-action-container');
- this._dropdown.render(append(this._container, dropdownContainer));
- this._register(addDisposableListener(dropdownContainer, EventType.KEY_DOWN, (e) => {
- var _a;
- const event = new StandardKeyboardEvent(e);
- if (event.equals(15 /* LeftArrow */)) {
- this._defaultAction.element.tabIndex = 0;
- this._dropdown.setFocusable(false);
- (_a = this._defaultAction.element) === null || _a === void 0 ? void 0 : _a.focus();
- event.stopPropagation();
- }
- }));
- }
- focus(fromRight) {
- if (fromRight) {
- this._dropdown.focus();
- }
- else {
- this._defaultAction.element.tabIndex = 0;
- this._defaultAction.element.focus();
- }
- }
- blur() {
- this._defaultAction.element.tabIndex = -1;
- this._dropdown.blur();
- this._container.blur();
- }
- setFocusable(focusable) {
- if (focusable) {
- this._defaultAction.element.tabIndex = 0;
- }
- else {
- this._defaultAction.element.tabIndex = -1;
- this._dropdown.setFocusable(false);
- }
- }
- dispose() {
- this._defaultAction.dispose();
- this._dropdown.dispose();
- super.dispose();
- }
- };
- DropdownWithDefaultActionViewItem = __decorate([
- __param(2, IKeybindingService),
- __param(3, INotificationService),
- __param(4, IContextMenuService),
- __param(5, IMenuService),
- __param(6, IInstantiationService),
- __param(7, IStorageService)
- ], DropdownWithDefaultActionViewItem);
- /**
- * Creates action view items for menu actions or submenu actions.
- */
- export function createActionViewItem(instaService, action, options) {
- if (action instanceof MenuItemAction) {
- return instaService.createInstance(MenuEntryActionViewItem, action, undefined);
- }
- else if (action instanceof SubmenuItemAction) {
- if (action.item.rememberDefaultAction) {
- return instaService.createInstance(DropdownWithDefaultActionViewItem, action, options);
- }
- else {
- return instaService.createInstance(SubmenuEntryActionViewItem, action, options);
- }
- }
- else {
- return undefined;
- }
- }
|