quickInputList.ts 19.2 KB
Newer Older
1 2 3 4 5 6 7 8
/*---------------------------------------------------------------------------------------------
 *  Copyright (c) Microsoft Corporation. All rights reserved.
 *  Licensed under the MIT License. See License.txt in the project root for license information.
 *--------------------------------------------------------------------------------------------*/

'use strict';

import 'vs/css!./quickInput';
J
Joao Moreno 已提交
9
import { IVirtualDelegate, IRenderer } from 'vs/base/browser/ui/list/list';
10 11 12 13
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 已提交
14
import { IQuickPickItem, IQuickPickItemButtonEvent, IQuickPickSeparator } from 'vs/platform/quickinput/common/quickInput';
15 16 17
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 已提交
18
import { Emitter, Event, mapEvent } from 'vs/base/common/event';
C
Christof Marti 已提交
19 20 21
import { assign } from 'vs/base/common/objects';
import { KeyCode } from 'vs/base/common/keyCodes';
import { StandardKeyboardEvent } from 'vs/base/browser/keyboardEvent';
22 23
import { IconLabel, IIconLabelValueOptions } from 'vs/base/browser/ui/iconLabel/iconLabel';
import { HighlightedLabel } from 'vs/base/browser/ui/highlightedlabel/highlightedLabel';
24
import { memoize } from 'vs/base/common/decorators';
25 26
import { range } from 'vs/base/common/arrays';
import * as platform from 'vs/base/common/platform';
C
Christof Marti 已提交
27
import { listFocusBackground, pickerGroupBorder, pickerGroupForeground } from 'vs/platform/theme/common/colorRegistry';
28
import { registerThemingParticipant } from 'vs/platform/theme/common/themeService';
C
Christof Marti 已提交
29 30 31
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';
32 33 34

const $ = dom.$;

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

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

72
	constructor(init: IListElement) {
C
Christof Marti 已提交
73 74
		assign(this, init);
	}
75 76
}

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

89
class ListElementRenderer implements IRenderer<ListElement, IListElementTemplateData> {
90

91
	static readonly ID = 'listelement';
92 93

	get templateId() {
94
		return ListElementRenderer.ID;
95 96
	}

97 98
	renderTemplate(container: HTMLElement): IListElementTemplateData {
		const data: IListElementTemplateData = Object.create(null);
99 100
		data.toDisposeElement = [];
		data.toDisposeTemplate = [];
101

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

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

112
		// Rows
113 114 115
		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'));
116 117 118

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

120
		// Detail
121
		const detailContainer = dom.append(row2, $('.quick-input-list-label-meta'));
122
		data.detail = new HighlightedLabel(detailContainer);
123

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

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

132 133 134
		return data;
	}

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

		const { labelHighlights, descriptionHighlights, detailHighlights } = element;

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

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

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

C
Christof Marti 已提交
160 161 162 163 164 165 166
		// 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 已提交
167
		if (element.separator) {
C
Christof Marti 已提交
168 169 170 171 172
			dom.addClass(data.entry, 'quick-input-list-separator-border');
		} else {
			dom.removeClass(data.entry, 'quick-input-list-separator-border');
		}

C
Christof Marti 已提交
173 174
		// Actions
		data.actionBar.clear();
C
Christof Marti 已提交
175 176 177
		const buttons = element.item.buttons;
		if (buttons && buttons.length) {
			data.actionBar.push(buttons.map((button, index) => {
C
Christof Marti 已提交
178 179 180 181 182 183 184 185 186 187
				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 已提交
188 189 190
			dom.addClass(data.entry, 'has-actions');
		} else {
			dom.removeClass(data.entry, 'has-actions');
C
Christof Marti 已提交
191
		}
192 193
	}

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

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

J
Joao Moreno 已提交
204
class ListElementDelegate implements IVirtualDelegate<ListElement> {
205

206
	getHeight(element: ListElement): number {
C
Christof Marti 已提交
207
		return element.saneDetail ? 44 : 22;
208 209
	}

210 211
	getTemplateId(element: ListElement): string {
		return ListElementRenderer.ID;
212 213 214
	}
}

215
export class QuickInputList {
216

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

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

295
	@memoize
296
	get onDidChangeFocus() {
C
Christof Marti 已提交
297 298 299
		return mapEvent(this.list.onFocusChange, e => e.elements.map(e => e.item));
	}

300
	@memoize
301
	get onDidChangeSelection() {
302 303 304
		return mapEvent(this.list.onSelectionChange, e => e.elements.map(e => e.item));
	}

305
	getAllVisibleChecked() {
306
		return this.allVisibleChecked(this.elements, false);
307 308
	}

309
	private allVisibleChecked(elements: ListElement[], whenNoneVisible = true) {
310 311
		for (let i = 0, n = elements.length; i < n; i++) {
			const element = elements[i];
312 313 314 315 316 317
			if (!element.hidden) {
				if (!element.checked) {
					return false;
				} else {
					whenNoneVisible = true;
				}
318 319
			}
		}
320
		return whenNoneVisible;
C
Christof Marti 已提交
321 322
	}

323
	getCheckedCount() {
324 325 326 327 328 329 330 331
		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 已提交
332 333
	}

334 335 336 337 338 339 340 341 342 343 344
	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;
	}

345
	setAllVisibleChecked(checked: boolean) {
346 347 348 349 350 351 352 353 354 355 356
		try {
			this._fireCheckedEvents = false;
			this.elements.forEach(element => {
				if (!element.hidden) {
					element.checked = checked;
				}
			});
		} finally {
			this._fireCheckedEvents = true;
			this.fireCheckedEvents();
		}
C
Christof Marti 已提交
357 358
	}

C
Christof Marti 已提交
359
	setElements(inputElements: (IQuickPickItem | IQuickPickSeparator)[]): void {
C
Christof Marti 已提交
360
		this.elementDisposables = dispose(this.elementDisposables);
C
Christof Marti 已提交
361
		const fireButtonTriggered = (event: IQuickPickItemButtonEvent<IQuickPickItem>) => this.fireButtonTriggered(event);
C
Christof Marti 已提交
362 363 364 365 366 367 368
		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 已提交
369 370 371
					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 已提交
372 373 374 375 376 377 378
					checked: false,
					separator: previous && previous.type === 'separator' ? previous : undefined,
					fireButtonTriggered
				}));
			}
			return result;
		}, [] as ListElement[]);
379
		this.elementDisposables.push(...this.elements.map(element => element.onChecked(() => this.fireCheckedEvents())));
380 381 382 383 384

		this.elementsToIndexes = this.elements.reduce((map, element, index) => {
			map.set(element.item, index);
			return map;
		}, new Map<IQuickPickItem, number>());
385
		this.list.splice(0, this.list.length, this.elements);
386
		this.list.setFocus([]);
387
		this._onChangedVisibleCount.fire(this.elements.length);
388 389
	}

390 391
	getFocusedElements() {
		return this.list.getFocusedElements()
392 393 394
			.map(e => e.item);
	}

395 396 397 398 399 400
	setFocusedElements(items: IQuickPickItem[]) {
		this.list.setFocus(items
			.filter(item => this.elementsToIndexes.has(item))
			.map(item => this.elementsToIndexes.get(item)));
	}

401 402 403 404
	getActiveDescendant() {
		return this.list.getHTMLElement().getAttribute('aria-activedescendant');
	}

405 406 407 408 409
	getSelectedElements() {
		return this.list.getSelectedElements()
			.map(e => e.item);
	}

410 411 412 413 414 415
	setSelectedElements(items: IQuickPickItem[]) {
		this.list.setSelection(items
			.filter(item => this.elementsToIndexes.has(item))
			.map(item => this.elementsToIndexes.get(item)));
	}

416 417
	getCheckedElements() {
		return this.elements.filter(e => e.checked)
418 419 420
			.map(e => e.item);
	}

421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436
	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 已提交
437 438 439 440
	set enabled(value: boolean) {
		this.list.getHTMLElement().style.pointerEvents = value ? null : 'none';
	}

441
	focus(what: 'First' | 'Last' | 'Next' | 'Previous' | 'NextPage' | 'PreviousPage'): void {
442 443 444 445
		if (!this.list.length) {
			return;
		}

C
Christof Marti 已提交
446
		if ((what === 'Next' || what === 'NextPage') && this.list.getFocus()[0] === this.list.length - 1) {
447 448
			what = 'First';
		}
C
Christof Marti 已提交
449
		if ((what === 'Previous' || what === 'PreviousPage') && this.list.getFocus()[0] === 0) {
450 451 452
			what = 'Last';
		}

C
Christof Marti 已提交
453
		this.list['focus' + what]();
454
		this.list.reveal(this.list.getFocus()[0]);
455 456
	}

457 458 459 460
	clearFocus() {
		this.list.setFocus([]);
	}

461 462 463 464
	domFocus() {
		this.list.domFocus();
	}

465 466 467 468 469 470 471 472 473 474 475 476 477 478
	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 已提交
479 480
				const previous = element.index && this.inputElements[element.index - 1];
				element.separator = previous && previous.type === 'separator' ? previous : undefined;
481 482 483 484 485 486
			});
		}

		// Filter by value (since we support octicons, use octicon aware fuzzy matching)
		else {
			this.elements.forEach(element => {
C
Christof Marti 已提交
487 488 489
				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;
490 491 492 493 494 495 496 497 498 499

				if (element.shouldAlwaysShow || 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;
C
Christof Marti 已提交
500
					element.hidden = !element.item.alwaysShow;
501
				}
C
Christof Marti 已提交
502
				element.separator = undefined;
503 504 505
			});
		}

506 507
		const shownElements = this.elements.filter(element => !element.hidden);

508
		// Sort by value
C
Christof Marti 已提交
509 510 511 512 513 514
		if (query) {
			const normalizedSearchValue = query.toLowerCase();
			shownElements.sort((a, b) => {
				return compareEntries(a, b, normalizedSearchValue);
			});
		}
515

516 517 518 519
		this.elementsToIndexes = shownElements.reduce((map, element, index) => {
			map.set(element.item, index);
			return map;
		}, new Map<IQuickPickItem, number>());
520
		this.list.splice(0, this.list.length, shownElements);
521
		this.list.setFocus([]);
C
Christof Marti 已提交
522
		this.list.layout();
523

524
		this._onChangedAllVisibleChecked.fire(this.getAllVisibleChecked());
525
		this._onChangedVisibleCount.fire(shownElements.length);
526
	}
C
Christof Marti 已提交
527 528

	toggleCheckbox() {
529 530 531 532 533 534 535 536 537 538
		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 已提交
539 540 541
		}
	}

542
	display(display: boolean) {
543
		this.container.style.display = display ? '' : 'none';
544 545
	}

546 547 548 549
	isDisplayed() {
		return this.container.style.display !== 'none';
	}

C
Christof Marti 已提交
550
	dispose() {
C
Christof Marti 已提交
551
		this.elementDisposables = dispose(this.elementDisposables);
C
Christof Marti 已提交
552 553
		this.disposables = dispose(this.disposables);
	}
554 555 556

	private fireCheckedEvents() {
		if (this._fireCheckedEvents) {
557 558 559
			this._onChangedAllVisibleChecked.fire(this.getAllVisibleChecked());
			this._onChangedCheckedCount.fire(this.getCheckedCount());
			this._onChangedCheckedElements.fire(this.getCheckedElements());
560 561
		}
	}
C
Christof Marti 已提交
562 563 564 565

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

568
function compareEntries(elementA: ListElement, elementB: ListElement, lookFor: string): number {
569 570 571 572 573 574 575 576 577 578 579

	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 已提交
580
	return compareAnything(elementA.saneLabel, elementB.saneLabel, lookFor);
581 582 583 584 585 586
}

registerThemingParticipant((theme, collector) => {
	// Override inactive focus background with active focus background for single-pick case.
	const listInactiveFocusBackground = theme.getColor(listFocusBackground);
	if (listInactiveFocusBackground) {
587 588
		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}; }`);
589
	}
C
Christof Marti 已提交
590 591 592 593 594 595 596 597
	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}; }`);
	}
598
});