quickInputList.ts 19.1 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 'vs/css!./quickInput';
J
Joao Moreno 已提交
7
import { IListVirtualDelegate, IListRenderer } from 'vs/base/browser/ui/list/list';
8 9 10 11
import * as dom from 'vs/base/browser/dom';
import { dispose, IDisposable } from 'vs/base/common/lifecycle';
import { WorkbenchList } from 'vs/platform/list/browser/listService';
import { IInstantiationService } from 'vs/platform/instantiation/common/instantiation';
C
Christof Marti 已提交
12
import { IQuickPickItem, IQuickPickItemButtonEvent, IQuickPickSeparator } from 'vs/platform/quickinput/common/quickInput';
13 14 15
import { IMatch } from 'vs/base/common/filters';
import { matchesFuzzyOcticonAware, parseOcticons } from 'vs/base/common/octicon';
import { compareAnything } from 'vs/base/common/comparers';
C
Christof Marti 已提交
16
import { Emitter, Event, mapEvent } from 'vs/base/common/event';
C
Christof Marti 已提交
17 18 19
import { assign } from 'vs/base/common/objects';
import { KeyCode } from 'vs/base/common/keyCodes';
import { StandardKeyboardEvent } from 'vs/base/browser/keyboardEvent';
20 21
import { IconLabel, IIconLabelValueOptions } from 'vs/base/browser/ui/iconLabel/iconLabel';
import { HighlightedLabel } from 'vs/base/browser/ui/highlightedlabel/highlightedLabel';
22
import { memoize } from 'vs/base/common/decorators';
23 24
import { range } from 'vs/base/common/arrays';
import * as platform from 'vs/base/common/platform';
C
Christof Marti 已提交
25
import { listFocusBackground, pickerGroupBorder, pickerGroupForeground } from 'vs/platform/theme/common/colorRegistry';
26
import { registerThemingParticipant } from 'vs/platform/theme/common/themeService';
C
Christof Marti 已提交
27 28 29
import { ActionBar } from 'vs/base/browser/ui/actionbar/actionbar';
import { Action } from 'vs/base/common/actions';
import { getIconClass } from 'vs/workbench/browser/parts/quickinput/quickInputUtils';
30 31 32

const $ = dom.$;

33
interface IListElement {
34
	index: number;
35
	item: IQuickPickItem;
C
Christof Marti 已提交
36 37 38
	saneLabel: string;
	saneDescription?: string;
	saneDetail?: string;
39
	checked: boolean;
C
Christof Marti 已提交
40
	separator: IQuickPickSeparator;
C
Christof Marti 已提交
41
	fireButtonTriggered: (event: IQuickPickItemButtonEvent<IQuickPickItem>) => void;
C
Christof Marti 已提交
42 43
}

44
class ListElement implements IListElement {
C
Christof Marti 已提交
45
	index: number;
46
	item: IQuickPickItem;
C
Christof Marti 已提交
47 48 49
	saneLabel: string;
	saneDescription?: string;
	saneDetail?: string;
C
Christof Marti 已提交
50
	hidden = false;
51 52
	private _onChecked = new Emitter<boolean>();
	onChecked = this._onChecked.event;
53
	_checked?: boolean;
54 55
	get checked() {
		return this._checked;
C
Christof Marti 已提交
56
	}
57 58 59 60
	set checked(value: boolean) {
		if (value !== this._checked) {
			this._checked = value;
			this._onChecked.fire(value);
C
Christof Marti 已提交
61 62
		}
	}
C
Christof Marti 已提交
63
	separator: IQuickPickSeparator;
64 65 66
	labelHighlights?: IMatch[];
	descriptionHighlights?: IMatch[];
	detailHighlights?: IMatch[];
C
Christof Marti 已提交
67
	fireButtonTriggered: (event: IQuickPickItemButtonEvent<IQuickPickItem>) => void;
C
Christof Marti 已提交
68

69
	constructor(init: IListElement) {
C
Christof Marti 已提交
70 71
		assign(this, init);
	}
72 73
}

74
interface IListElementTemplateData {
C
Christof Marti 已提交
75
	entry: HTMLDivElement;
76
	checkbox: HTMLInputElement;
77 78
	label: IconLabel;
	detail: HighlightedLabel;
C
Christof Marti 已提交
79
	separator: HTMLDivElement;
C
Christof Marti 已提交
80
	actionBar: ActionBar;
81
	element: ListElement;
C
Christof Marti 已提交
82 83
	toDisposeElement: IDisposable[];
	toDisposeTemplate: IDisposable[];
84 85
}

J
Joao Moreno 已提交
86
class ListElementRenderer implements IListRenderer<ListElement, IListElementTemplateData> {
87

88
	static readonly ID = 'listelement';
89 90

	get templateId() {
91
		return ListElementRenderer.ID;
92 93
	}

94 95
	renderTemplate(container: HTMLElement): IListElementTemplateData {
		const data: IListElementTemplateData = Object.create(null);
96 97
		data.toDisposeElement = [];
		data.toDisposeTemplate = [];
98

C
Christof Marti 已提交
99
		data.entry = dom.append(container, $('.quick-input-list-entry'));
100

101
		// Checkbox
C
Christof Marti 已提交
102
		const label = dom.append(data.entry, $('label.quick-input-list-label'));
103
		data.checkbox = <HTMLInputElement>dom.append(label, $('input.quick-input-list-checkbox'));
104
		data.checkbox.type = 'checkbox';
C
Christof Marti 已提交
105
		data.toDisposeTemplate.push(dom.addStandardDisposableListener(data.checkbox, dom.EventType.CHANGE, e => {
106
			data.element.checked = data.checkbox.checked;
C
Christof Marti 已提交
107
		}));
108

109
		// Rows
110 111 112
		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'));
113 114 115

		// Label
		data.label = new IconLabel(row1, { supportHighlights: true, supportDescriptionHighlights: true });
116

117
		// Detail
118
		const detailContainer = dom.append(row2, $('.quick-input-list-label-meta'));
S
Sandeep Somavarapu 已提交
119
		data.detail = new HighlightedLabel(detailContainer, true);
120

C
Christof Marti 已提交
121 122 123
		// Separator
		data.separator = dom.append(data.entry, $('.quick-input-list-separator'));

C
Christof Marti 已提交
124
		// Actions
C
Christof Marti 已提交
125
		data.actionBar = new ActionBar(data.entry);
C
Christof Marti 已提交
126 127 128
		data.actionBar.domNode.classList.add('quick-input-list-entry-action-bar');
		data.toDisposeTemplate.push(data.actionBar);

129 130 131
		return data;
	}

132
	renderElement(element: ListElement, index: number, data: IListElementTemplateData): void {
C
Christof Marti 已提交
133
		data.toDisposeElement = dispose(data.toDisposeElement);
134
		data.element = element;
135 136
		data.checkbox.checked = element.checked;
		data.toDisposeElement.push(element.onChecked(checked => data.checkbox.checked = checked));
137 138 139 140 141 142

		const { labelHighlights, descriptionHighlights, detailHighlights } = element;

		// Label
		const options: IIconLabelValueOptions = Object.create(null);
		options.matches = labelHighlights || [];
C
Christof Marti 已提交
143
		options.descriptionTitle = element.saneDescription;
144
		options.descriptionMatches = descriptionHighlights || [];
C
Christof Marti 已提交
145
		options.extraClasses = element.item.iconClasses;
C
Christof Marti 已提交
146
		data.label.setValue(element.saneLabel, element.saneDescription, options);
147 148

		// Meta
C
Christof Marti 已提交
149
		data.detail.set(element.saneDetail, detailHighlights);
C
Christof Marti 已提交
150

151
		// ARIA label
C
Christof Marti 已提交
152
		data.entry.setAttribute('aria-label', [element.saneLabel, element.saneDescription, element.saneDetail]
153
			.map(s => s && parseOcticons(s).text)
154 155 156
			.filter(s => !!s)
			.join(', '));

C
Christof Marti 已提交
157 158 159 160 161 162 163
		// Separator
		if (element.separator && element.separator.label) {
			data.separator.textContent = element.separator.label;
			data.separator.style.display = null;
		} else {
			data.separator.style.display = 'none';
		}
C
Christof Marti 已提交
164
		if (element.separator) {
C
Christof Marti 已提交
165 166 167 168 169
			dom.addClass(data.entry, 'quick-input-list-separator-border');
		} else {
			dom.removeClass(data.entry, 'quick-input-list-separator-border');
		}

C
Christof Marti 已提交
170 171
		// Actions
		data.actionBar.clear();
C
Christof Marti 已提交
172 173 174
		const buttons = element.item.buttons;
		if (buttons && buttons.length) {
			data.actionBar.push(buttons.map((button, index) => {
C
Christof Marti 已提交
175 176 177 178 179 180 181 182 183 184
				const action = new Action(`id-${index}`, '', button.iconClass || getIconClass(button.iconPath), true, () => {
					element.fireButtonTriggered({
						button,
						item: element.item
					});
					return null;
				});
				action.tooltip = button.tooltip;
				return action;
			}), { icon: true, label: false });
C
Christof Marti 已提交
185 186 187
			dom.addClass(data.entry, 'has-actions');
		} else {
			dom.removeClass(data.entry, 'has-actions');
C
Christof Marti 已提交
188
		}
189 190
	}

C
Christof Marti 已提交
191 192
	disposeElement(element: ListElement, index: number, data: IListElementTemplateData): void {
		data.toDisposeElement = dispose(data.toDisposeElement);
J
Joao Moreno 已提交
193 194
	}

195
	disposeTemplate(data: IListElementTemplateData): void {
C
Christof Marti 已提交
196 197
		data.toDisposeElement = dispose(data.toDisposeElement);
		data.toDisposeTemplate = dispose(data.toDisposeTemplate);
198 199 200
	}
}

J
Joao Moreno 已提交
201
class ListElementDelegate implements IListVirtualDelegate<ListElement> {
202

203
	getHeight(element: ListElement): number {
C
Christof Marti 已提交
204
		return element.saneDetail ? 44 : 22;
205 206
	}

207 208
	getTemplateId(element: ListElement): string {
		return ListElementRenderer.ID;
209 210 211
	}
}

212
export class QuickInputList {
213

214
	readonly id: string;
C
Christof Marti 已提交
215
	private container: HTMLElement;
216
	private list: WorkbenchList<ListElement>;
C
Christof Marti 已提交
217
	private inputElements: (IQuickPickItem | IQuickPickSeparator)[];
218
	private elements: ListElement[] = [];
219
	private elementsToIndexes = new Map<IQuickPickItem, number>();
220 221
	matchOnDescription = false;
	matchOnDetail = false;
222 223 224 225
	private _onChangedAllVisibleChecked = new Emitter<boolean>();
	onChangedAllVisibleChecked: Event<boolean> = this._onChangedAllVisibleChecked.event;
	private _onChangedCheckedCount = new Emitter<number>();
	onChangedCheckedCount: Event<number> = this._onChangedCheckedCount.event;
226 227
	private _onChangedVisibleCount = new Emitter<number>();
	onChangedVisibleCount: Event<number> = this._onChangedVisibleCount.event;
228 229
	private _onChangedCheckedElements = new Emitter<IQuickPickItem[]>();
	onChangedCheckedElements: Event<IQuickPickItem[]> = this._onChangedCheckedElements.event;
C
Christof Marti 已提交
230 231
	private _onButtonTriggered = new Emitter<IQuickPickItemButtonEvent<IQuickPickItem>>();
	onButtonTriggered = this._onButtonTriggered.event;
232 233
	private _onLeave = new Emitter<void>();
	onLeave: Event<void> = this._onLeave.event;
234
	private _fireCheckedEvents = true;
C
Christof Marti 已提交
235
	private elementDisposables: IDisposable[] = [];
C
Christof Marti 已提交
236
	private disposables: IDisposable[] = [];
237 238 239

	constructor(
		private parent: HTMLElement,
240
		id: string,
241 242
		@IInstantiationService private instantiationService: IInstantiationService
	) {
243
		this.id = id;
244 245 246
		this.container = dom.append(this.parent, $('.quick-input-list'));
		const delegate = new ListElementDelegate();
		this.list = this.instantiationService.createInstance(WorkbenchList, this.container, delegate, [new ListElementRenderer()], {
J
Joao Moreno 已提交
247
			identityProvider: { getId: element => element.label },
C
Christof Marti 已提交
248
			openController: { shouldOpen: () => false }, // Workaround #58124
249
			setRowLineHeight: false,
250
			multipleSelectionSupport: false
251
		}) as WorkbenchList<ListElement>;
252
		this.list.getHTMLElement().id = id;
C
Christof Marti 已提交
253 254 255
		this.disposables.push(this.list);
		this.disposables.push(this.list.onKeyDown(e => {
			const event = new StandardKeyboardEvent(e);
256 257 258 259
			switch (event.keyCode) {
				case KeyCode.Space:
					this.toggleCheckbox();
					break;
260 261 262 263 264
				case KeyCode.KEY_A:
					if (platform.isMacintosh ? e.metaKey : e.ctrlKey) {
						this.list.setFocus(range(this.list.length));
					}
					break;
265
				case KeyCode.UpArrow:
C
Christof Marti 已提交
266
				case KeyCode.PageUp:
267 268 269 270 271 272
					const focus1 = this.list.getFocus();
					if (focus1.length === 1 && focus1[0] === 0) {
						this._onLeave.fire();
					}
					break;
				case KeyCode.DownArrow:
C
Christof Marti 已提交
273
				case KeyCode.PageDown:
274 275
					const focus2 = this.list.getFocus();
					if (focus2.length === 1 && focus2[0] === this.list.length - 1) {
276 277 278
						this._onLeave.fire();
					}
					break;
C
Christof Marti 已提交
279 280
			}
		}));
281 282 283 284 285
		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();
			}
		}));
286 287
	}

288
	@memoize
289
	get onDidChangeFocus() {
C
Christof Marti 已提交
290 291 292
		return mapEvent(this.list.onFocusChange, e => e.elements.map(e => e.item));
	}

293
	@memoize
294
	get onDidChangeSelection() {
295 296 297
		return mapEvent(this.list.onSelectionChange, e => e.elements.map(e => e.item));
	}

298
	getAllVisibleChecked() {
299
		return this.allVisibleChecked(this.elements, false);
300 301
	}

302
	private allVisibleChecked(elements: ListElement[], whenNoneVisible = true) {
303 304
		for (let i = 0, n = elements.length; i < n; i++) {
			const element = elements[i];
305 306 307 308 309 310
			if (!element.hidden) {
				if (!element.checked) {
					return false;
				} else {
					whenNoneVisible = true;
				}
311 312
			}
		}
313
		return whenNoneVisible;
C
Christof Marti 已提交
314 315
	}

316
	getCheckedCount() {
317 318 319 320 321 322 323 324
		let count = 0;
		const elements = this.elements;
		for (let i = 0, n = elements.length; i < n; i++) {
			if (elements[i].checked) {
				count++;
			}
		}
		return count;
C
Christof Marti 已提交
325 326
	}

327 328 329 330 331 332 333 334 335 336 337
	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;
	}

338
	setAllVisibleChecked(checked: boolean) {
339 340 341 342 343 344 345 346 347 348 349
		try {
			this._fireCheckedEvents = false;
			this.elements.forEach(element => {
				if (!element.hidden) {
					element.checked = checked;
				}
			});
		} finally {
			this._fireCheckedEvents = true;
			this.fireCheckedEvents();
		}
C
Christof Marti 已提交
350 351
	}

C
Christof Marti 已提交
352
	setElements(inputElements: (IQuickPickItem | IQuickPickSeparator)[]): void {
C
Christof Marti 已提交
353
		this.elementDisposables = dispose(this.elementDisposables);
C
Christof Marti 已提交
354
		const fireButtonTriggered = (event: IQuickPickItemButtonEvent<IQuickPickItem>) => this.fireButtonTriggered(event);
C
Christof Marti 已提交
355 356 357 358 359 360 361
		this.inputElements = inputElements;
		this.elements = inputElements.reduce((result, item, index) => {
			if (item.type !== 'separator') {
				const previous = index && inputElements[index - 1];
				result.push(new ListElement({
					index,
					item,
C
Christof Marti 已提交
362 363 364
					saneLabel: item.label && item.label.replace(/\r?\n/g, ' '),
					saneDescription: item.description && item.description.replace(/\r?\n/g, ' '),
					saneDetail: item.detail && item.detail.replace(/\r?\n/g, ' '),
C
Christof Marti 已提交
365 366 367 368 369 370 371
					checked: false,
					separator: previous && previous.type === 'separator' ? previous : undefined,
					fireButtonTriggered
				}));
			}
			return result;
		}, [] as ListElement[]);
372
		this.elementDisposables.push(...this.elements.map(element => element.onChecked(() => this.fireCheckedEvents())));
373 374 375 376 377

		this.elementsToIndexes = this.elements.reduce((map, element, index) => {
			map.set(element.item, index);
			return map;
		}, new Map<IQuickPickItem, number>());
378
		this.list.splice(0, this.list.length); // Clear focus and selection first, sending the events when the list is empty.
379
		this.list.splice(0, this.list.length, this.elements);
380
		this._onChangedVisibleCount.fire(this.elements.length);
381 382
	}

383 384
	getFocusedElements() {
		return this.list.getFocusedElements()
385 386 387
			.map(e => e.item);
	}

388 389 390 391 392 393
	setFocusedElements(items: IQuickPickItem[]) {
		this.list.setFocus(items
			.filter(item => this.elementsToIndexes.has(item))
			.map(item => this.elementsToIndexes.get(item)));
	}

394 395 396 397
	getActiveDescendant() {
		return this.list.getHTMLElement().getAttribute('aria-activedescendant');
	}

398 399 400 401 402
	getSelectedElements() {
		return this.list.getSelectedElements()
			.map(e => e.item);
	}

403 404 405 406 407 408
	setSelectedElements(items: IQuickPickItem[]) {
		this.list.setSelection(items
			.filter(item => this.elementsToIndexes.has(item))
			.map(item => this.elementsToIndexes.get(item)));
	}

409 410
	getCheckedElements() {
		return this.elements.filter(e => e.checked)
411 412 413
			.map(e => e.item);
	}

414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429
	setCheckedElements(items: IQuickPickItem[]) {
		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();
		}
	}

C
Christof Marti 已提交
430 431 432 433
	set enabled(value: boolean) {
		this.list.getHTMLElement().style.pointerEvents = value ? null : 'none';
	}

434
	focus(what: 'First' | 'Last' | 'Next' | 'Previous' | 'NextPage' | 'PreviousPage'): void {
435 436 437 438
		if (!this.list.length) {
			return;
		}

C
Christof Marti 已提交
439
		if ((what === 'Next' || what === 'NextPage') && this.list.getFocus()[0] === this.list.length - 1) {
440 441
			what = 'First';
		}
C
Christof Marti 已提交
442
		if ((what === 'Previous' || what === 'PreviousPage') && this.list.getFocus()[0] === 0) {
443 444 445
			what = 'Last';
		}

C
Christof Marti 已提交
446
		this.list['focus' + what]();
447
		this.list.reveal(this.list.getFocus()[0]);
448 449
	}

450 451 452 453
	clearFocus() {
		this.list.setFocus([]);
	}

454 455 456 457
	domFocus() {
		this.list.domFocus();
	}

458 459 460 461 462 463 464 465 466 467 468 469 470 471
	layout(): void {
		this.list.layout();
	}

	filter(query: string) {
		query = query.trim();

		// Reset filtering
		if (!query) {
			this.elements.forEach(element => {
				element.labelHighlights = undefined;
				element.descriptionHighlights = undefined;
				element.detailHighlights = undefined;
				element.hidden = false;
C
Christof Marti 已提交
472 473
				const previous = element.index && this.inputElements[element.index - 1];
				element.separator = previous && previous.type === 'separator' ? previous : undefined;
474 475 476 477 478 479
			});
		}

		// Filter by value (since we support octicons, use octicon aware fuzzy matching)
		else {
			this.elements.forEach(element => {
C
Christof Marti 已提交
480 481 482
				const labelHighlights = matchesFuzzyOcticonAware(query, parseOcticons(element.saneLabel));
				const descriptionHighlights = this.matchOnDescription ? matchesFuzzyOcticonAware(query, parseOcticons(element.saneDescription || '')) : undefined;
				const detailHighlights = this.matchOnDetail ? matchesFuzzyOcticonAware(query, parseOcticons(element.saneDetail || '')) : undefined;
483

C
Christof Marti 已提交
484
				if (labelHighlights || descriptionHighlights || detailHighlights) {
485 486 487 488 489 490 491 492
					element.labelHighlights = labelHighlights;
					element.descriptionHighlights = descriptionHighlights;
					element.detailHighlights = detailHighlights;
					element.hidden = false;
				} else {
					element.labelHighlights = undefined;
					element.descriptionHighlights = undefined;
					element.detailHighlights = undefined;
C
Christof Marti 已提交
493
					element.hidden = !element.item.alwaysShow;
494
				}
C
Christof Marti 已提交
495
				element.separator = undefined;
496 497 498
			});
		}

499 500
		const shownElements = this.elements.filter(element => !element.hidden);

501
		// Sort by value
C
Christof Marti 已提交
502 503 504 505 506 507
		if (query) {
			const normalizedSearchValue = query.toLowerCase();
			shownElements.sort((a, b) => {
				return compareEntries(a, b, normalizedSearchValue);
			});
		}
508

509 510 511 512
		this.elementsToIndexes = shownElements.reduce((map, element, index) => {
			map.set(element.item, index);
			return map;
		}, new Map<IQuickPickItem, number>());
513
		this.list.splice(0, this.list.length, shownElements);
514
		this.list.setFocus([]);
C
Christof Marti 已提交
515
		this.list.layout();
516

517
		this._onChangedAllVisibleChecked.fire(this.getAllVisibleChecked());
518
		this._onChangedVisibleCount.fire(shownElements.length);
519
	}
C
Christof Marti 已提交
520 521

	toggleCheckbox() {
522 523 524 525 526 527 528 529 530 531
		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();
C
Christof Marti 已提交
532 533 534
		}
	}

535
	display(display: boolean) {
536
		this.container.style.display = display ? '' : 'none';
537 538
	}

539 540 541 542
	isDisplayed() {
		return this.container.style.display !== 'none';
	}

C
Christof Marti 已提交
543
	dispose() {
C
Christof Marti 已提交
544
		this.elementDisposables = dispose(this.elementDisposables);
C
Christof Marti 已提交
545 546
		this.disposables = dispose(this.disposables);
	}
547 548 549

	private fireCheckedEvents() {
		if (this._fireCheckedEvents) {
550 551 552
			this._onChangedAllVisibleChecked.fire(this.getAllVisibleChecked());
			this._onChangedCheckedCount.fire(this.getCheckedCount());
			this._onChangedCheckedElements.fire(this.getCheckedElements());
553 554
		}
	}
C
Christof Marti 已提交
555 556 557 558

	private fireButtonTriggered(event: IQuickPickItemButtonEvent<IQuickPickItem>) {
		this._onButtonTriggered.fire(event);
	}
559 560
}

561
function compareEntries(elementA: ListElement, elementB: ListElement, lookFor: string): number {
562 563 564 565 566 567 568 569 570 571 572

	const labelHighlightsA = elementA.labelHighlights || [];
	const labelHighlightsB = elementB.labelHighlights || [];
	if (labelHighlightsA.length && !labelHighlightsB.length) {
		return -1;
	}

	if (!labelHighlightsA.length && labelHighlightsB.length) {
		return 1;
	}

C
Christof Marti 已提交
573
	return compareAnything(elementA.saneLabel, elementB.saneLabel, lookFor);
574 575 576 577 578 579
}

registerThemingParticipant((theme, collector) => {
	// Override inactive focus background with active focus background for single-pick case.
	const listInactiveFocusBackground = theme.getColor(listFocusBackground);
	if (listInactiveFocusBackground) {
580 581
		collector.addRule(`.quick-input-list .monaco-list .monaco-list-row.focused { background-color:  ${listInactiveFocusBackground}; }`);
		collector.addRule(`.quick-input-list .monaco-list .monaco-list-row.focused:hover { background-color:  ${listInactiveFocusBackground}; }`);
582
	}
C
Christof Marti 已提交
583 584 585 586 587 588 589 590
	const pickerGroupBorderColor = theme.getColor(pickerGroupBorder);
	if (pickerGroupBorderColor) {
		collector.addRule(`.quick-input-list .quick-input-list-entry { border-top-color:  ${pickerGroupBorderColor}; }`);
	}
	const pickerGroupForegroundColor = theme.getColor(pickerGroupForeground);
	if (pickerGroupForegroundColor) {
		collector.addRule(`.quick-input-list .quick-input-list-separator { color:  ${pickerGroupForegroundColor}; }`);
	}
591
});