UNPKG

monaco-editor

Version:
899 lines (898 loc) • 39.2 kB
/*--------------------------------------------------------------------------------------------- * 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; }; import * as dom from '../../../base/browser/dom.js'; import { StandardKeyboardEvent } from '../../../base/browser/keyboardEvent.js'; import { ActionBar } from '../../../base/browser/ui/actionbar/actionbar.js'; import { IconLabel } from '../../../base/browser/ui/iconLabel/iconLabel.js'; import { KeybindingLabel } from '../../../base/browser/ui/keybindingLabel/keybindingLabel.js'; import { range } from '../../../base/common/arrays.js'; import { ThrottledDelayer } from '../../../base/common/async.js'; import { compareAnything } from '../../../base/common/comparers.js'; import { memoize } from '../../../base/common/decorators.js'; import { isCancellationError } from '../../../base/common/errors.js'; import { Emitter, Event } from '../../../base/common/event.js'; import { getCodiconAriaLabel, matchesFuzzyIconAware, parseLabelWithIcons } from '../../../base/common/iconLabels.js'; import { DisposableStore, dispose } from '../../../base/common/lifecycle.js'; import * as platform from '../../../base/common/platform.js'; import { ltrim } from '../../../base/common/strings.js'; import './media/quickInput.css'; import { localize } from '../../../nls.js'; import { quickInputButtonToAction } from './quickInputUtils.js'; import { Lazy } from '../../../base/common/lazy.js'; import { URI } from '../../../base/common/uri.js'; import { isDark } from '../../theme/common/theme.js'; const $ = dom.$; class ListElement { constructor(mainItem, previous, index, hasCheckbox, fireButtonTriggered, fireSeparatorButtonTriggered, onCheckedEmitter) { var _a, _b, _c; // state will get updated later this._checked = false; this._hidden = false; this.hasCheckbox = hasCheckbox; this.index = index; this.fireButtonTriggered = fireButtonTriggered; this.fireSeparatorButtonTriggered = fireSeparatorButtonTriggered; this._onChecked = onCheckedEmitter; this.onChecked = hasCheckbox ? Event.map(Event.filter(this._onChecked.event, e => e.listElement === this), e => e.checked) : Event.None; if (mainItem.type === 'separator') { this._separator = mainItem; } else { this.item = mainItem; if (previous && previous.type === 'separator' && !previous.buttons) { this._separator = previous; } this.saneDescription = this.item.description; this.saneDetail = this.item.detail; this._labelHighlights = (_a = this.item.highlights) === null || _a === void 0 ? void 0 : _a.label; this._descriptionHighlights = (_b = this.item.highlights) === null || _b === void 0 ? void 0 : _b.description; this._detailHighlights = (_c = this.item.highlights) === null || _c === void 0 ? void 0 : _c.detail; this.saneTooltip = this.item.tooltip; } this._init = new Lazy(() => { var _a; const saneLabel = (_a = mainItem.label) !== null && _a !== void 0 ? _a : ''; const saneSortLabel = parseLabelWithIcons(saneLabel).text.trim(); const saneAriaLabel = mainItem.ariaLabel || [saneLabel, this.saneDescription, this.saneDetail] .map(s => getCodiconAriaLabel(s)) .filter(s => !!s) .join(', '); return { saneLabel, saneSortLabel, saneAriaLabel }; }); } // #region Lazy Getters get saneLabel() { return this._init.value.saneLabel; } get saneSortLabel() { return this._init.value.saneSortLabel; } get saneAriaLabel() { return this._init.value.saneAriaLabel; } // #endregion // #region Getters and Setters get element() { return this._element; } set element(value) { this._element = value; } get hidden() { return this._hidden; } set hidden(value) { this._hidden = value; } get checked() { return this._checked; } set checked(value) { if (value !== this._checked) { this._checked = value; this._onChecked.fire({ listElement: this, checked: value }); } } get separator() { return this._separator; } set separator(value) { this._separator = value; } get labelHighlights() { return this._labelHighlights; } set labelHighlights(value) { this._labelHighlights = value; } get descriptionHighlights() { return this._descriptionHighlights; } set descriptionHighlights(value) { this._descriptionHighlights = value; } get detailHighlights() { return this._detailHighlights; } set detailHighlights(value) { this._detailHighlights = value; } } class ListElementRenderer { constructor(themeService, hoverDelegate) { this.themeService = themeService; this.hoverDelegate = hoverDelegate; } get templateId() { return ListElementRenderer.ID; } renderTemplate(container) { const data = Object.create(null); data.toDisposeElement = []; data.toDisposeTemplate = []; data.entry = dom.append(container, $('.quick-input-list-entry')); // Checkbox const label = dom.append(data.entry, $('label.quick-input-list-label')); data.toDisposeTemplate.push(dom.addStandardDisposableListener(label, dom.EventType.CLICK, e => { if (!data.checkbox.offsetParent) { // If checkbox not visible: e.preventDefault(); // Prevent toggle of checkbox when it is immediately shown afterwards. #91740 } })); data.checkbox = dom.append(label, $('input.quick-input-list-checkbox')); data.checkbox.type = 'checkbox'; data.toDisposeTemplate.push(dom.addStandardDisposableListener(data.checkbox, dom.EventType.CHANGE, e => { data.element.checked = data.checkbox.checked; })); // Rows const rows = dom.append(label, $('.quick-input-list-rows')); const row1 = dom.append(rows, $('.quick-input-list-row')); const row2 = dom.append(rows, $('.quick-input-list-row')); // Label data.label = new IconLabel(row1, { supportHighlights: true, supportDescriptionHighlights: true, supportIcons: true, hoverDelegate: this.hoverDelegate }); data.toDisposeTemplate.push(data.label); data.icon = dom.prepend(data.label.element, $('.quick-input-list-icon')); // Keybinding const keybindingContainer = dom.append(row1, $('.quick-input-list-entry-keybinding')); data.keybinding = new KeybindingLabel(keybindingContainer, platform.OS); // Detail const detailContainer = dom.append(row2, $('.quick-input-list-label-meta')); data.detail = new IconLabel(detailContainer, { supportHighlights: true, supportIcons: true, hoverDelegate: this.hoverDelegate }); data.toDisposeTemplate.push(data.detail); // Separator data.separator = dom.append(data.entry, $('.quick-input-list-separator')); // Actions data.actionBar = new ActionBar(data.entry, this.hoverDelegate ? { hoverDelegate: this.hoverDelegate } : undefined); data.actionBar.domNode.classList.add('quick-input-list-entry-action-bar'); data.toDisposeTemplate.push(data.actionBar); return data; } renderElement(element, index, data) { var _a, _b, _c, _d; data.element = element; element.element = (_a = data.entry) !== null && _a !== void 0 ? _a : undefined; const mainItem = element.item ? element.item : element.separator; data.checkbox.checked = element.checked; data.toDisposeElement.push(element.onChecked(checked => data.checkbox.checked = checked)); const { labelHighlights, descriptionHighlights, detailHighlights } = element; if ((_b = element.item) === null || _b === void 0 ? void 0 : _b.iconPath) { const icon = isDark(this.themeService.getColorTheme().type) ? element.item.iconPath.dark : ((_c = element.item.iconPath.light) !== null && _c !== void 0 ? _c : element.item.iconPath.dark); const iconUrl = URI.revive(icon); data.icon.className = 'quick-input-list-icon'; data.icon.style.backgroundImage = dom.asCSSUrl(iconUrl); } else { data.icon.style.backgroundImage = ''; data.icon.className = ((_d = element.item) === null || _d === void 0 ? void 0 : _d.iconClass) ? `quick-input-list-icon ${element.item.iconClass}` : ''; } // Label const options = { matches: labelHighlights || [], // If we have a tooltip, we want that to be shown and not any other hover descriptionTitle: element.saneTooltip ? undefined : element.saneDescription, descriptionMatches: descriptionHighlights || [], labelEscapeNewLines: true }; if (mainItem.type !== 'separator') { options.extraClasses = mainItem.iconClasses; options.italic = mainItem.italic; options.strikethrough = mainItem.strikethrough; data.entry.classList.remove('quick-input-list-separator-as-item'); } else { data.entry.classList.add('quick-input-list-separator-as-item'); } data.label.setLabel(element.saneLabel, element.saneDescription, options); // Keybinding data.keybinding.set(mainItem.type === 'separator' ? undefined : mainItem.keybinding); // Detail if (element.saneDetail) { data.detail.element.style.display = ''; data.detail.setLabel(element.saneDetail, undefined, { matches: detailHighlights, // If we have a tooltip, we want that to be shown and not any other hover title: element.saneTooltip ? undefined : element.saneDetail, labelEscapeNewLines: true }); } else { data.detail.element.style.display = 'none'; } // Separator if (element.item && element.separator && element.separator.label) { data.separator.textContent = element.separator.label; data.separator.style.display = ''; } else { data.separator.style.display = 'none'; } data.entry.classList.toggle('quick-input-list-separator-border', !!element.separator); // Actions const buttons = mainItem.buttons; if (buttons && buttons.length) { data.actionBar.push(buttons.map((button, index) => quickInputButtonToAction(button, `id-${index}`, () => mainItem.type !== 'separator' ? element.fireButtonTriggered({ button, item: mainItem }) : element.fireSeparatorButtonTriggered({ button, separator: mainItem }))), { icon: true, label: false }); data.entry.classList.add('has-actions'); } else { data.entry.classList.remove('has-actions'); } } disposeElement(element, index, data) { data.toDisposeElement = dispose(data.toDisposeElement); data.actionBar.clear(); } disposeTemplate(data) { data.toDisposeElement = dispose(data.toDisposeElement); data.toDisposeTemplate = dispose(data.toDisposeTemplate); } } ListElementRenderer.ID = 'listelement'; class ListElementDelegate { getHeight(element) { if (!element.item) { // must be a separator return 24; } return element.saneDetail ? 44 : 22; } getTemplateId(element) { return ListElementRenderer.ID; } } export var QuickInputListFocus; (function (QuickInputListFocus) { QuickInputListFocus[QuickInputListFocus["First"] = 1] = "First"; QuickInputListFocus[QuickInputListFocus["Second"] = 2] = "Second"; QuickInputListFocus[QuickInputListFocus["Last"] = 3] = "Last"; QuickInputListFocus[QuickInputListFocus["Next"] = 4] = "Next"; QuickInputListFocus[QuickInputListFocus["Previous"] = 5] = "Previous"; QuickInputListFocus[QuickInputListFocus["NextPage"] = 6] = "NextPage"; QuickInputListFocus[QuickInputListFocus["PreviousPage"] = 7] = "PreviousPage"; })(QuickInputListFocus || (QuickInputListFocus = {})); export class QuickInputList { constructor(parent, id, options, themeService) { this.parent = parent; this.options = options; this.inputElements = []; this.elements = []; this.elementsToIndexes = new Map(); this.matchOnDescription = false; this.matchOnDetail = false; this.matchOnLabel = true; this.matchOnLabelMode = 'fuzzy'; this.sortByLabel = true; this._onChangedAllVisibleChecked = new Emitter(); this.onChangedAllVisibleChecked = this._onChangedAllVisibleChecked.event; this._onChangedCheckedCount = new Emitter(); this.onChangedCheckedCount = this._onChangedCheckedCount.event; this._onChangedVisibleCount = new Emitter(); this.onChangedVisibleCount = this._onChangedVisibleCount.event; this._onChangedCheckedElements = new Emitter(); this.onChangedCheckedElements = this._onChangedCheckedElements.event; this._onButtonTriggered = new Emitter(); this.onButtonTriggered = this._onButtonTriggered.event; this._onSeparatorButtonTriggered = new Emitter(); this.onSeparatorButtonTriggered = this._onSeparatorButtonTriggered.event; this._onKeyDown = new Emitter(); this.onKeyDown = this._onKeyDown.event; this._onLeave = new Emitter(); this.onLeave = this._onLeave.event; this._listElementChecked = new Emitter(); this._fireCheckedEvents = true; this.elementDisposables = []; this.disposables = []; this.id = id; this.container = dom.append(this.parent, $('.quick-input-list')); const delegate = new ListElementDelegate(); const accessibilityProvider = new QuickInputAccessibilityProvider(); this.list = options.createList('QuickInput', this.container, delegate, [new ListElementRenderer(themeService, options.hoverDelegate)], { identityProvider: { getId: element => { var _a, _b, _c, _d, _e, _f, _g, _h; // always prefer item over separator because if item is defined, it must be the main item type // always prefer a defined id if one was specified and use label as a fallback return (_h = (_f = (_d = (_b = (_a = element.item) === null || _a === void 0 ? void 0 : _a.id) !== null && _b !== void 0 ? _b : (_c = element.item) === null || _c === void 0 ? void 0 : _c.label) !== null && _d !== void 0 ? _d : (_e = element.separator) === null || _e === void 0 ? void 0 : _e.id) !== null && _f !== void 0 ? _f : (_g = element.separator) === null || _g === void 0 ? void 0 : _g.label) !== null && _h !== void 0 ? _h : ''; } }, setRowLineHeight: false, multipleSelectionSupport: false, horizontalScrolling: false, accessibilityProvider }); this.list.getHTMLElement().id = id; this.disposables.push(this.list); this.disposables.push(this.list.onKeyDown(e => { const event = new StandardKeyboardEvent(e); switch (event.keyCode) { case 10 /* KeyCode.Space */: this.toggleCheckbox(); break; case 31 /* KeyCode.KeyA */: if (platform.isMacintosh ? e.metaKey : e.ctrlKey) { this.list.setFocus(range(this.list.length)); } break; case 16 /* KeyCode.UpArrow */: { const focus1 = this.list.getFocus(); if (focus1.length === 1 && focus1[0] === 0) { this._onLeave.fire(); } break; } case 18 /* KeyCode.DownArrow */: { const focus2 = this.list.getFocus(); if (focus2.length === 1 && focus2[0] === this.list.length - 1) { this._onLeave.fire(); } break; } } this._onKeyDown.fire(event); })); this.disposables.push(this.list.onMouseDown(e => { if (e.browserEvent.button !== 2) { // Works around / fixes #64350. e.browserEvent.preventDefault(); } })); this.disposables.push(dom.addDisposableListener(this.container, dom.EventType.CLICK, e => { if (e.x || e.y) { // Avoid 'click' triggered by 'space' on checkbox. this._onLeave.fire(); } })); this.disposables.push(this.list.onMouseMiddleClick(e => { this._onLeave.fire(); })); this.disposables.push(this.list.onContextMenu(e => { if (typeof e.index === 'number') { e.browserEvent.preventDefault(); // we want to treat a context menu event as // a gesture to open the item at the index // since we do not have any context menu // this enables for example macOS to Ctrl- // click on an item to open it. this.list.setSelection([e.index]); } })); const delayer = new ThrottledDelayer(options.hoverDelegate.delay); // onMouseOver triggers every time a new element has been moused over // even if it's on the same list item. this.disposables.push(this.list.onMouseOver(async (e) => { var _a; // If we hover over an anchor element, we don't want to show the hover because // the anchor may have a tooltip that we want to show instead. if (e.browserEvent.target instanceof HTMLAnchorElement) { delayer.cancel(); return; } if ( // anchors are an exception as called out above so we skip them here !(e.browserEvent.relatedTarget instanceof HTMLAnchorElement) && // check if the mouse is still over the same element dom.isAncestor(e.browserEvent.relatedTarget, (_a = e.element) === null || _a === void 0 ? void 0 : _a.element)) { return; } try { await delayer.trigger(async () => { if (e.element) { this.showHover(e.element); } }); } catch (e) { // Ignore cancellation errors due to mouse out if (!isCancellationError(e)) { throw e; } } })); this.disposables.push(this.list.onMouseOut(e => { var _a; // onMouseOut triggers every time a new element has been moused over // even if it's on the same list item. We only want one event, so we // check if the mouse is still over the same element. if (dom.isAncestor(e.browserEvent.relatedTarget, (_a = e.element) === null || _a === void 0 ? void 0 : _a.element)) { return; } delayer.cancel(); })); this.disposables.push(delayer); this.disposables.push(this._listElementChecked.event(_ => this.fireCheckedEvents())); this.disposables.push(this._onChangedAllVisibleChecked, this._onChangedCheckedCount, this._onChangedVisibleCount, this._onChangedCheckedElements, this._onButtonTriggered, this._onSeparatorButtonTriggered, this._onLeave, this._onKeyDown); } get onDidChangeFocus() { return Event.map(this.list.onDidChangeFocus, e => e.elements.map(e => e.item)); } get onDidChangeSelection() { return Event.map(this.list.onDidChangeSelection, e => ({ items: e.elements.map(e => e.item), event: e.browserEvent })); } get scrollTop() { return this.list.scrollTop; } set scrollTop(scrollTop) { this.list.scrollTop = scrollTop; } get ariaLabel() { return this.list.getHTMLElement().ariaLabel; } set ariaLabel(label) { this.list.getHTMLElement().ariaLabel = label; } getAllVisibleChecked() { return this.allVisibleChecked(this.elements, false); } allVisibleChecked(elements, whenNoneVisible = true) { for (let i = 0, n = elements.length; i < n; i++) { const element = elements[i]; if (!element.hidden) { if (!element.checked) { return false; } else { whenNoneVisible = true; } } } return whenNoneVisible; } getCheckedCount() { let count = 0; const elements = this.elements; for (let i = 0, n = elements.length; i < n; i++) { if (elements[i].checked) { count++; } } return count; } getVisibleCount() { let count = 0; const elements = this.elements; for (let i = 0, n = elements.length; i < n; i++) { if (!elements[i].hidden) { count++; } } return count; } setAllVisibleChecked(checked) { try { this._fireCheckedEvents = false; this.elements.forEach(element => { if (!element.hidden) { element.checked = checked; } }); } finally { this._fireCheckedEvents = true; this.fireCheckedEvents(); } } setElements(inputElements) { this.elementDisposables = dispose(this.elementDisposables); const fireButtonTriggered = (event) => this.fireButtonTriggered(event); const fireSeparatorButtonTriggered = (event) => this.fireSeparatorButtonTriggered(event); this.inputElements = inputElements; const elementsToIndexes = new Map(); const hasCheckbox = this.parent.classList.contains('show-checkboxes'); this.elements = inputElements.reduce((result, item, index) => { var _a; const previous = index > 0 ? inputElements[index - 1] : undefined; if (item.type === 'separator') { if (!item.buttons) { // This separator will be rendered as a part of the list item return result; } } const element = new ListElement(item, previous, index, hasCheckbox, fireButtonTriggered, fireSeparatorButtonTriggered, this._listElementChecked); const resultIndex = result.length; result.push(element); elementsToIndexes.set((_a = element.item) !== null && _a !== void 0 ? _a : element.separator, resultIndex); return result; }, []); this.elementsToIndexes = elementsToIndexes; this.list.splice(0, this.list.length); // Clear focus and selection first, sending the events when the list is empty. this.list.splice(0, this.list.length, this.elements); this._onChangedVisibleCount.fire(this.elements.length); } getFocusedElements() { return this.list.getFocusedElements() .map(e => e.item); } setFocusedElements(items) { this.list.setFocus(items .filter(item => this.elementsToIndexes.has(item)) .map(item => this.elementsToIndexes.get(item))); if (items.length > 0) { const focused = this.list.getFocus()[0]; if (typeof focused === 'number') { this.list.reveal(focused); } } } getActiveDescendant() { return this.list.getHTMLElement().getAttribute('aria-activedescendant'); } setSelectedElements(items) { this.list.setSelection(items .filter(item => this.elementsToIndexes.has(item)) .map(item => this.elementsToIndexes.get(item))); } getCheckedElements() { return this.elements.filter(e => e.checked) .map(e => e.item) .filter(e => !!e); } setCheckedElements(items) { try { this._fireCheckedEvents = false; const checked = new Set(); for (const item of items) { checked.add(item); } for (const element of this.elements) { element.checked = checked.has(element.item); } } finally { this._fireCheckedEvents = true; this.fireCheckedEvents(); } } set enabled(value) { this.list.getHTMLElement().style.pointerEvents = value ? '' : 'none'; } focus(what) { if (!this.list.length) { return; } if (what === QuickInputListFocus.Second && this.list.length < 2) { what = QuickInputListFocus.First; } switch (what) { case QuickInputListFocus.First: this.list.scrollTop = 0; this.list.focusFirst(undefined, (e) => !!e.item); break; case QuickInputListFocus.Second: this.list.scrollTop = 0; this.list.focusNth(1, undefined, (e) => !!e.item); break; case QuickInputListFocus.Last: this.list.scrollTop = this.list.scrollHeight; this.list.focusLast(undefined, (e) => !!e.item); break; case QuickInputListFocus.Next: { this.list.focusNext(undefined, true, undefined, (e) => !!e.item); const index = this.list.getFocus()[0]; if (index !== 0 && !this.elements[index - 1].item && this.list.firstVisibleIndex > index - 1) { this.list.reveal(index - 1); } break; } case QuickInputListFocus.Previous: { this.list.focusPrevious(undefined, true, undefined, (e) => !!e.item); const index = this.list.getFocus()[0]; if (index !== 0 && !this.elements[index - 1].item && this.list.firstVisibleIndex > index - 1) { this.list.reveal(index - 1); } break; } case QuickInputListFocus.NextPage: this.list.focusNextPage(undefined, (e) => !!e.item); break; case QuickInputListFocus.PreviousPage: this.list.focusPreviousPage(undefined, (e) => !!e.item); break; } const focused = this.list.getFocus()[0]; if (typeof focused === 'number') { this.list.reveal(focused); } } clearFocus() { this.list.setFocus([]); } domFocus() { this.list.domFocus(); } /** * Disposes of the hover and shows a new one for the given index if it has a tooltip. * @param element The element to show the hover for */ showHover(element) { var _a, _b, _c; if (this._lastHover && !this._lastHover.isDisposed) { (_b = (_a = this.options.hoverDelegate).onDidHideHover) === null || _b === void 0 ? void 0 : _b.call(_a); (_c = this._lastHover) === null || _c === void 0 ? void 0 : _c.dispose(); } if (!element.element || !element.saneTooltip) { return; } this._lastHover = this.options.hoverDelegate.showHover({ content: element.saneTooltip, target: element.element, linkHandler: (url) => { this.options.linkOpenerDelegate(url); }, appearance: { showPointer: true, }, container: this.container, position: { hoverPosition: 1 /* HoverPosition.RIGHT */ } }, false); } layout(maxHeight) { this.list.getHTMLElement().style.maxHeight = maxHeight ? `${ // Make sure height aligns with list item heights Math.floor(maxHeight / 44) * 44 // Add some extra height so that it's clear there's more to scroll + 6}px` : ''; this.list.layout(); } filter(query) { if (!(this.sortByLabel || this.matchOnLabel || this.matchOnDescription || this.matchOnDetail)) { this.list.layout(); return false; } const queryWithWhitespace = query; query = query.trim(); // Reset filtering if (!query || !(this.matchOnLabel || this.matchOnDescription || this.matchOnDetail)) { this.elements.forEach(element => { element.labelHighlights = undefined; element.descriptionHighlights = undefined; element.detailHighlights = undefined; element.hidden = false; const previous = element.index && this.inputElements[element.index - 1]; if (element.item) { element.separator = previous && previous.type === 'separator' && !previous.buttons ? previous : undefined; } }); } // Filter by value (since we support icons in labels, use $(..) aware fuzzy matching) else { let currentSeparator; this.elements.forEach(element => { var _a, _b, _c, _d; let labelHighlights; if (this.matchOnLabelMode === 'fuzzy') { labelHighlights = this.matchOnLabel ? (_a = matchesFuzzyIconAware(query, parseLabelWithIcons(element.saneLabel))) !== null && _a !== void 0 ? _a : undefined : undefined; } else { labelHighlights = this.matchOnLabel ? (_b = matchesContiguousIconAware(queryWithWhitespace, parseLabelWithIcons(element.saneLabel))) !== null && _b !== void 0 ? _b : undefined : undefined; } const descriptionHighlights = this.matchOnDescription ? (_c = matchesFuzzyIconAware(query, parseLabelWithIcons(element.saneDescription || ''))) !== null && _c !== void 0 ? _c : undefined : undefined; const detailHighlights = this.matchOnDetail ? (_d = matchesFuzzyIconAware(query, parseLabelWithIcons(element.saneDetail || ''))) !== null && _d !== void 0 ? _d : undefined : undefined; if (labelHighlights || descriptionHighlights || detailHighlights) { element.labelHighlights = labelHighlights; element.descriptionHighlights = descriptionHighlights; element.detailHighlights = detailHighlights; element.hidden = false; } else { element.labelHighlights = undefined; element.descriptionHighlights = undefined; element.detailHighlights = undefined; element.hidden = element.item ? !element.item.alwaysShow : true; } // Ensure separators are filtered out first before deciding if we need to bring them back if (element.item) { element.separator = undefined; } else if (element.separator) { element.hidden = true; } // we can show the separator unless the list gets sorted by match if (!this.sortByLabel) { const previous = element.index && this.inputElements[element.index - 1]; currentSeparator = previous && previous.type === 'separator' ? previous : currentSeparator; if (currentSeparator && !element.hidden) { element.separator = currentSeparator; currentSeparator = undefined; } } }); } const shownElements = this.elements.filter(element => !element.hidden); // Sort by value if (this.sortByLabel && query) { const normalizedSearchValue = query.toLowerCase(); shownElements.sort((a, b) => { return compareEntries(a, b, normalizedSearchValue); }); } this.elementsToIndexes = shownElements.reduce((map, element, index) => { var _a; map.set((_a = element.item) !== null && _a !== void 0 ? _a : element.separator, index); return map; }, new Map()); this.list.splice(0, this.list.length, shownElements); this.list.setFocus([]); this.list.layout(); this._onChangedAllVisibleChecked.fire(this.getAllVisibleChecked()); this._onChangedVisibleCount.fire(shownElements.length); return true; } toggleCheckbox() { try { this._fireCheckedEvents = false; const elements = this.list.getFocusedElements(); const allChecked = this.allVisibleChecked(elements); for (const element of elements) { element.checked = !allChecked; } } finally { this._fireCheckedEvents = true; this.fireCheckedEvents(); } } display(display) { this.container.style.display = display ? '' : 'none'; } isDisplayed() { return this.container.style.display !== 'none'; } dispose() { this.elementDisposables = dispose(this.elementDisposables); this.disposables = dispose(this.disposables); } fireCheckedEvents() { if (this._fireCheckedEvents) { this._onChangedAllVisibleChecked.fire(this.getAllVisibleChecked()); this._onChangedCheckedCount.fire(this.getCheckedCount()); this._onChangedCheckedElements.fire(this.getCheckedElements()); } } fireButtonTriggered(event) { this._onButtonTriggered.fire(event); } fireSeparatorButtonTriggered(event) { this._onSeparatorButtonTriggered.fire(event); } style(styles) { this.list.style(styles); } toggleHover() { const element = this.list.getFocusedElements()[0]; if (!(element === null || element === void 0 ? void 0 : element.saneTooltip)) { return; } // if there's a hover already, hide it (toggle off) if (this._lastHover && !this._lastHover.isDisposed) { this._lastHover.dispose(); return; } // If there is no hover, show it (toggle on) const focused = this.list.getFocusedElements()[0]; if (!focused) { return; } this.showHover(focused); const store = new DisposableStore(); store.add(this.list.onDidChangeFocus(e => { if (e.indexes.length) { this.showHover(e.elements[0]); } })); if (this._lastHover) { store.add(this._lastHover); } this._toggleHover = store; this.elementDisposables.push(this._toggleHover); } } __decorate([ memoize ], QuickInputList.prototype, "onDidChangeFocus", null); __decorate([ memoize ], QuickInputList.prototype, "onDidChangeSelection", null); function matchesContiguousIconAware(query, target) { const { text, iconOffsets } = target; // Return early if there are no icon markers in the word to match against if (!iconOffsets || iconOffsets.length === 0) { return matchesContiguous(query, text); } // Trim the word to match against because it could have leading // whitespace now if the word started with an icon const wordToMatchAgainstWithoutIconsTrimmed = ltrim(text, ' '); const leadingWhitespaceOffset = text.length - wordToMatchAgainstWithoutIconsTrimmed.length; // match on value without icon const matches = matchesContiguous(query, wordToMatchAgainstWithoutIconsTrimmed); // Map matches back to offsets with icon and trimming if (matches) { for (const match of matches) { const iconOffset = iconOffsets[match.start + leadingWhitespaceOffset] /* icon offsets at index */ + leadingWhitespaceOffset /* overall leading whitespace offset */; match.start += iconOffset; match.end += iconOffset; } } return matches; } function matchesContiguous(word, wordToMatchAgainst) { const matchIndex = wordToMatchAgainst.toLowerCase().indexOf(word.toLowerCase()); if (matchIndex !== -1) { return [{ start: matchIndex, end: matchIndex + word.length }]; } return null; } function compareEntries(elementA, elementB, lookFor) { const labelHighlightsA = elementA.labelHighlights || []; const labelHighlightsB = elementB.labelHighlights || []; if (labelHighlightsA.length && !labelHighlightsB.length) { return -1; } if (!labelHighlightsA.length && labelHighlightsB.length) { return 1; } if (labelHighlightsA.length === 0 && labelHighlightsB.length === 0) { return 0; } return compareAnything(elementA.saneSortLabel, elementB.saneSortLabel, lookFor); } class QuickInputAccessibilityProvider { getWidgetAriaLabel() { return localize('quickInput', "Quick Input"); } getAriaLabel(element) { var _a; return ((_a = element.separator) === null || _a === void 0 ? void 0 : _a.label) ? `${element.saneAriaLabel}, ${element.separator.label}` : element.saneAriaLabel; } getWidgetRole() { return 'listbox'; } getRole(element) { return element.hasCheckbox ? 'checkbox' : 'option'; } isChecked(element) { if (!element.hasCheckbox) { return undefined; } return { value: element.checked, onDidChange: element.onChecked }; } }