mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-28 09:35:38 -05:00
298 lines
9.6 KiB
TypeScript
298 lines
9.6 KiB
TypeScript
/*---------------------------------------------------------------------------------------------
|
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
|
* Licensed under the Source EULA. See License.txt in the project root for license information.
|
|
*--------------------------------------------------------------------------------------------*/
|
|
|
|
import { IQuickPick, IQuickPickItem } from 'vs/platform/quickinput/common/quickInput';
|
|
import { CancellationToken, CancellationTokenSource } from 'vs/base/common/cancellation';
|
|
import { IQuickPickSeparator, IKeyMods, IQuickPickAcceptEvent } from 'vs/base/parts/quickinput/common/quickInput';
|
|
import { IQuickAccessProvider } from 'vs/platform/quickinput/common/quickAccess';
|
|
import { IDisposable, DisposableStore, Disposable, MutableDisposable } from 'vs/base/common/lifecycle';
|
|
import { timeout, isThenable } from 'vs/base/common/async';
|
|
|
|
export enum TriggerAction {
|
|
|
|
/**
|
|
* Do nothing after the button was clicked.
|
|
*/
|
|
NO_ACTION,
|
|
|
|
/**
|
|
* Close the picker.
|
|
*/
|
|
CLOSE_PICKER,
|
|
|
|
/**
|
|
* Update the results of the picker.
|
|
*/
|
|
REFRESH_PICKER,
|
|
|
|
/**
|
|
* Remove the item from the picker.
|
|
*/
|
|
REMOVE_ITEM
|
|
}
|
|
|
|
export interface IPickerQuickAccessItem extends IQuickPickItem {
|
|
|
|
/**
|
|
* A method that will be executed when the pick item is accepted from
|
|
* the picker. The picker will close automatically before running this.
|
|
*
|
|
* @param keyMods the state of modifier keys when the item was accepted.
|
|
* @param event the underlying event that caused the accept to trigger.
|
|
*/
|
|
accept?(keyMods: IKeyMods, event: IQuickPickAcceptEvent): void;
|
|
|
|
/**
|
|
* A method that will be executed when a button of the pick item was
|
|
* clicked on.
|
|
*
|
|
* @param buttonIndex index of the button of the item that
|
|
* was clicked.
|
|
*
|
|
* @param the state of modifier keys when the button was triggered.
|
|
*
|
|
* @returns a value that indicates what should happen after the trigger
|
|
* which can be a `Promise` for long running operations.
|
|
*/
|
|
trigger?(buttonIndex: number, keyMods: IKeyMods): TriggerAction | Promise<TriggerAction>;
|
|
}
|
|
|
|
export interface IPickerQuickAccessProviderOptions {
|
|
canAcceptInBackground?: boolean;
|
|
}
|
|
|
|
export type Pick<T> = T | IQuickPickSeparator;
|
|
export type PicksWithActive<T> = { items: ReadonlyArray<Pick<T>>, active?: T };
|
|
export type Picks<T> = ReadonlyArray<Pick<T>> | PicksWithActive<T>;
|
|
export type FastAndSlowPicks<T> = { picks: Picks<T>, additionalPicks: Promise<Picks<T>> };
|
|
export type FastAndSlowPicksWithActive<T> = { picks: PicksWithActive<T>, additionalPicks: PicksWithActive<Picks<T>> };
|
|
|
|
function isPicksWithActive<T>(obj: unknown): obj is PicksWithActive<T> {
|
|
const candidate = obj as PicksWithActive<T>;
|
|
|
|
return Array.isArray(candidate.items);
|
|
}
|
|
|
|
function isFastAndSlowPicks<T>(obj: unknown): obj is FastAndSlowPicks<T> {
|
|
const candidate = obj as FastAndSlowPicks<T>;
|
|
|
|
return !!candidate.picks && isThenable(candidate.additionalPicks); // {{SQL CARBON EDIT}} workaround since we use zone promise
|
|
}
|
|
|
|
export abstract class PickerQuickAccessProvider<T extends IPickerQuickAccessItem> extends Disposable implements IQuickAccessProvider {
|
|
|
|
private static FAST_PICKS_RACE_DELAY = 200; // timeout before we accept fast results before slow results are present
|
|
|
|
constructor(private prefix: string, protected options?: IPickerQuickAccessProviderOptions) {
|
|
super();
|
|
}
|
|
|
|
provide(picker: IQuickPick<T>, token: CancellationToken): IDisposable {
|
|
const disposables = new DisposableStore();
|
|
|
|
// Apply options if any
|
|
picker.canAcceptInBackground = !!this.options?.canAcceptInBackground;
|
|
|
|
// Disable filtering & sorting, we control the results
|
|
picker.matchOnLabel = picker.matchOnDescription = picker.matchOnDetail = picker.sortByLabel = false;
|
|
|
|
// Set initial picks and update on type
|
|
let picksCts: CancellationTokenSource | undefined = undefined;
|
|
const picksDisposable = disposables.add(new MutableDisposable());
|
|
const updatePickerItems = async () => {
|
|
const picksDisposables = picksDisposable.value = new DisposableStore();
|
|
|
|
// Cancel any previous ask for picks and busy
|
|
picksCts?.dispose(true);
|
|
picker.busy = false;
|
|
|
|
// Create new cancellation source for this run
|
|
picksCts = new CancellationTokenSource(token);
|
|
|
|
// Collect picks and support both long running and short or combined
|
|
const picksToken = picksCts.token;
|
|
const providedPicks = this.getPicks(picker.value.substr(this.prefix.length).trim(), picksDisposables, picksToken);
|
|
|
|
function applyPicks(picks: Picks<T>): void {
|
|
if (isPicksWithActive(picks)) {
|
|
picker.items = picks.items;
|
|
if (picks.active) {
|
|
picker.activeItems = [picks.active];
|
|
}
|
|
} else {
|
|
picker.items = picks;
|
|
}
|
|
}
|
|
|
|
// No Picks
|
|
if (providedPicks === null) {
|
|
// Ignore
|
|
}
|
|
|
|
// Fast and Slow Picks
|
|
else if (isFastAndSlowPicks(providedPicks)) {
|
|
let fastPicksHandlerDone = false;
|
|
let slowPicksHandlerDone = false;
|
|
|
|
await Promise.all([
|
|
|
|
// Fast Picks: to reduce amount of flicker, we race against
|
|
// the slow picks over 500ms and then set the fast picks.
|
|
// If the slow picks are faster, we reduce the flicker by
|
|
// only setting the items once.
|
|
(async () => {
|
|
try {
|
|
await timeout(PickerQuickAccessProvider.FAST_PICKS_RACE_DELAY);
|
|
if (picksToken.isCancellationRequested) {
|
|
return;
|
|
}
|
|
|
|
if (!slowPicksHandlerDone) {
|
|
applyPicks(providedPicks.picks);
|
|
}
|
|
} finally {
|
|
fastPicksHandlerDone = true;
|
|
}
|
|
})(),
|
|
|
|
// Slow Picks: we await the slow picks and then set them at
|
|
// once together with the fast picks, but only if we actually
|
|
// have additional results.
|
|
(async () => {
|
|
picker.busy = true;
|
|
try {
|
|
const awaitedAdditionalPicks = await providedPicks.additionalPicks;
|
|
if (picksToken.isCancellationRequested) {
|
|
return;
|
|
}
|
|
|
|
let picks: ReadonlyArray<Pick<T>>;
|
|
let activePick: Pick<T> | undefined = undefined;
|
|
if (isPicksWithActive(providedPicks.picks)) {
|
|
picks = providedPicks.picks.items;
|
|
activePick = providedPicks.picks.active;
|
|
} else {
|
|
picks = providedPicks.picks;
|
|
}
|
|
|
|
let additionalPicks: ReadonlyArray<Pick<T>>;
|
|
let additionalActivePick: Pick<T> | undefined = undefined;
|
|
if (isPicksWithActive(awaitedAdditionalPicks)) {
|
|
additionalPicks = awaitedAdditionalPicks.items;
|
|
additionalActivePick = awaitedAdditionalPicks.active;
|
|
} else {
|
|
additionalPicks = awaitedAdditionalPicks;
|
|
}
|
|
|
|
if (additionalPicks.length > 0 || !fastPicksHandlerDone) {
|
|
applyPicks({
|
|
items: [...picks, ...additionalPicks],
|
|
active: activePick as T || additionalActivePick as T // {{SQL CARBON EDIT}} strict-null-checks
|
|
});
|
|
}
|
|
} finally {
|
|
if (!picksToken.isCancellationRequested) {
|
|
picker.busy = false;
|
|
}
|
|
|
|
slowPicksHandlerDone = true;
|
|
}
|
|
})()
|
|
]);
|
|
}
|
|
|
|
// Fast Picks
|
|
else if (!(isThenable(providedPicks))) { // {{SQL CARBON EDIT}} workaround since we use zone promise
|
|
applyPicks(providedPicks);
|
|
}
|
|
|
|
// Slow Picks
|
|
else {
|
|
picker.busy = true;
|
|
try {
|
|
const awaitedPicks = await providedPicks;
|
|
if (picksToken.isCancellationRequested) {
|
|
return;
|
|
}
|
|
|
|
applyPicks(awaitedPicks);
|
|
} finally {
|
|
if (!picksToken.isCancellationRequested) {
|
|
picker.busy = false;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
disposables.add(picker.onDidChangeValue(() => updatePickerItems()));
|
|
updatePickerItems();
|
|
|
|
// Accept the pick on accept and hide picker
|
|
disposables.add(picker.onDidAccept(event => {
|
|
const [item] = picker.selectedItems;
|
|
if (typeof item?.accept === 'function') {
|
|
if (!event.inBackground) {
|
|
picker.hide(); // hide picker unless we accept in background
|
|
}
|
|
|
|
item.accept(picker.keyMods, event);
|
|
}
|
|
}));
|
|
|
|
// Trigger the pick with button index if button triggered
|
|
disposables.add(picker.onDidTriggerItemButton(async ({ button, item }) => {
|
|
if (typeof item.trigger === 'function') {
|
|
const buttonIndex = item.buttons?.indexOf(button) ?? -1;
|
|
if (buttonIndex >= 0) {
|
|
const result = item.trigger(buttonIndex, picker.keyMods);
|
|
const action = (typeof result === 'number') ? result : await result;
|
|
|
|
if (token.isCancellationRequested) {
|
|
return;
|
|
}
|
|
|
|
switch (action) {
|
|
case TriggerAction.NO_ACTION:
|
|
break;
|
|
case TriggerAction.CLOSE_PICKER:
|
|
picker.hide();
|
|
break;
|
|
case TriggerAction.REFRESH_PICKER:
|
|
updatePickerItems();
|
|
break;
|
|
case TriggerAction.REMOVE_ITEM:
|
|
const index = picker.items.indexOf(item);
|
|
if (index !== -1) {
|
|
const items = picker.items.slice();
|
|
items.splice(index, 1);
|
|
picker.items = items;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}));
|
|
|
|
return disposables;
|
|
}
|
|
|
|
/**
|
|
* Returns an array of picks and separators as needed. If the picks are resolved
|
|
* long running, the provided cancellation token should be used to cancel the
|
|
* operation when the token signals this.
|
|
*
|
|
* The implementor is responsible for filtering and sorting the picks given the
|
|
* provided `filter`.
|
|
*
|
|
* @param filter a filter to apply to the picks.
|
|
* @param disposables can be used to register disposables that should be cleaned
|
|
* up when the picker closes.
|
|
* @param token for long running tasks, implementors need to check on cancellation
|
|
* through this token.
|
|
* @returns the picks either directly, as promise or combined fast and slow results.
|
|
* Pickers can return `null` to signal that no change in picks is needed.
|
|
*/
|
|
protected abstract getPicks(filter: string, disposables: DisposableStore, token: CancellationToken): Picks<T> | Promise<Picks<T>> | FastAndSlowPicks<T> | null;
|
|
}
|