From 5025b6a7b4f4dda93342559e5904589681e68b2b Mon Sep 17 00:00:00 2001 From: Igor Vinokur Date: Mon, 5 Aug 2019 12:35:39 +0300 Subject: [PATCH] QuickInput/InputBox API`s enhancements Signed-off-by: Igor Vinokur --- packages/core/src/browser/keybinding.ts | 21 +-- .../browser/quick-open/quick-input-service.ts | 15 +- .../quick-open/quick-open-action-provider.ts | 38 +++--- .../browser/quick-open/quick-open-model.ts | 120 ++++------------ .../quick-open/quick-pick-service-impl.ts | 32 +++-- .../src/browser/quick-open/quick-title-bar.ts | 82 +++++------ packages/core/src/browser/style/index.css | 1 + .../src/browser/style/quick-title-bar.css | 53 +++++++ packages/core/src/common/keybinding.ts | 32 +++++ packages/core/src/common/quick-open-model.ts | 129 ++++++++++++++++++ .../core/src/common/quick-pick-service.ts | 7 +- .../src/browser/monaco-quick-open-service.ts | 9 +- packages/plugin-ext/src/api/plugin-api.ts | 16 +-- .../src/main/browser/quick-open-main.ts | 28 ++-- packages/plugin-ext/src/plugin/quick-open.ts | 27 ++-- .../plugin-ext/src/plugin/type-converters.ts | 2 +- 16 files changed, 383 insertions(+), 229 deletions(-) create mode 100644 packages/core/src/browser/style/quick-title-bar.css create mode 100644 packages/core/src/common/keybinding.ts create mode 100644 packages/core/src/common/quick-open-model.ts diff --git a/packages/core/src/browser/keybinding.ts b/packages/core/src/browser/keybinding.ts index 1ea2ef860200a..2f9ec091f6b3b 100644 --- a/packages/core/src/browser/keybinding.ts +++ b/packages/core/src/browser/keybinding.ts @@ -24,6 +24,7 @@ import { ContributionProvider } from '../common/contribution-provider'; import { ILogger } from '../common/logger'; import { StatusBarAlignment, StatusBar } from './status-bar/status-bar'; import { ContextKeyService } from './context-key-service'; +import * as common from '../common/keybinding.js'; export enum KeybindingScope { DEFAULT, @@ -60,22 +61,10 @@ export namespace Keybinding { } } -export interface Keybinding { - /** Command identifier, this needs to be a unique string. */ - command: string; - /** Keybinding string as defined in packages/keymaps/README.md. */ - keybinding: string; - /** - * The optional keybinding context where this binding belongs to. - * If not specified, then this keybinding context belongs to the NOOP - * keybinding context. - */ - context?: string; - /** - * https://code.visualstudio.com/docs/getstarted/keybindings#_when-clause-contexts - */ - when?: string; -} +/** + * @deprecated import from `@theia/core/lib/common/keybinding` instead + */ +export type Keybinding = common.Keybinding; export interface ResolvedKeybinding extends Keybinding { /** diff --git a/packages/core/src/browser/quick-open/quick-input-service.ts b/packages/core/src/browser/quick-open/quick-input-service.ts index a3cbc41efa7b2..da37c063a54e0 100644 --- a/packages/core/src/browser/quick-open/quick-input-service.ts +++ b/packages/core/src/browser/quick-open/quick-input-service.ts @@ -21,7 +21,7 @@ import { Deferred } from '../../common/promise-util'; import { MaybePromise } from '../../common/types'; import { MessageType } from '../../common/message-service-protocol'; import { Emitter, Event } from '../../common/event'; -import { QuickTitleBar, QuickInputTitleButton } from './quick-title-bar'; +import { QuickTitleBar, QuickTitleButton } from './quick-title-bar'; export interface QuickInputOptions { @@ -53,7 +53,7 @@ export interface QuickInputOptions { /** * Buttons that are displayed on the title panel */ - buttons?: ReadonlyArray + buttons?: ReadonlyArray /** * Text for when there is a problem with the current input value @@ -119,10 +119,6 @@ export class QuickInputService { let currentText = ''; const validateInput = options && options.validateInput; - if (options && this.quickTitleBar.shouldShowTitleBar(options.title, options.step)) { - this.quickTitleBar.attachTitleBar(this.quickOpenService.widgetNode, options.title, options.step, options.totalSteps, options.buttons); - } - this.quickOpenService.open({ onType: async (lookFor, acceptor) => { this.onDidChangeValueEmitter.fire(lookFor); @@ -159,10 +155,15 @@ export class QuickInputService { this.quickTitleBar.hide(); } }); + + if (options && this.quickTitleBar.shouldShowTitleBar(options.title, options.step)) { + this.quickTitleBar.attachTitleBar(this.quickOpenService.widgetNode, options.title, options.step, options.totalSteps, options.buttons); + } + return result.promise; } - refresh() { + refresh(): void { this.quickOpenService.refresh(); } diff --git a/packages/core/src/browser/quick-open/quick-open-action-provider.ts b/packages/core/src/browser/quick-open/quick-open-action-provider.ts index d5deda316dccf..f666bc5da9481 100644 --- a/packages/core/src/browser/quick-open/quick-open-action-provider.ts +++ b/packages/core/src/browser/quick-open/quick-open-action-provider.ts @@ -14,28 +14,24 @@ * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 ********************************************************************************/ -import { Disposable } from '../../common/disposable'; import { injectable } from 'inversify'; -import { QuickOpenItem } from './quick-open-model'; - -export interface QuickOpenActionProvider { - hasActions(item: QuickOpenItem): boolean; - getActions(item: QuickOpenItem): Promise; -} - -export interface QuickOpenActionOptions { - id: string; - label?: string; - tooltip?: string; - class?: string | undefined; - enabled?: boolean; - checked?: boolean; - radio?: boolean; -} - -export interface QuickOpenAction extends QuickOpenActionOptions, Disposable { - run(item?: QuickOpenItem): PromiseLike; -} +import { QuickOpenItem } from '../../common/quick-open-model'; +import * as common from '../../common/quick-open-model'; + +/** + * @deprecated import from `@theia/core/lib/common/quick-open-model` instead + */ +export type QuickOpenActionProvider = common.QuickOpenActionProvider; + +/** + * @deprecated import from `@theia/core/lib/common/quick-open-model` instead + */ +export type QuickOpenActionOptions = common.QuickOpenActionOptions; + +/** + * @deprecated import from `@theia/core/lib/common/quick-open-model` instead + */ +export type QuickOpenAction = common.QuickOpenAction; @injectable() export abstract class QuickOpenBaseAction implements QuickOpenAction { diff --git a/packages/core/src/browser/quick-open/quick-open-model.ts b/packages/core/src/browser/quick-open/quick-open-model.ts index 7b66e1046d456..0f04e4e61622b 100644 --- a/packages/core/src/browser/quick-open/quick-open-model.ts +++ b/packages/core/src/browser/quick-open/quick-open-model.ts @@ -14,97 +14,29 @@ * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 ********************************************************************************/ -import URI from '../../common/uri'; -import { Keybinding } from '../keybinding'; -import { QuickOpenActionProvider } from './quick-open-action-provider'; - -export interface Highlight { - start: number - end: number -} - -export enum QuickOpenMode { - PREVIEW, - OPEN, - OPEN_IN_BACKGROUND -} - -export interface QuickOpenItemOptions { - tooltip?: string; - label?: string; - labelHighlights?: Highlight[]; - description?: string; - descriptionHighlights?: Highlight[]; - detail?: string; - detailHighlights?: Highlight[]; - hidden?: boolean; - uri?: URI; - iconClass?: string; - keybinding?: Keybinding; - run?(mode: QuickOpenMode): boolean; -} -export interface QuickOpenGroupItemOptions extends QuickOpenItemOptions { - groupLabel?: string; - showBorder?: boolean; -} - -export class QuickOpenItem { - - constructor( - protected options: T = {} as T - ) { } - - getTooltip(): string | undefined { - return this.options.tooltip || this.getLabel(); - } - getLabel(): string | undefined { - return this.options.label; - } - getLabelHighlights(): Highlight[] { - return this.options.labelHighlights || []; - } - getDescription(): string | undefined { - return this.options.description; - } - getDescriptionHighlights(): Highlight[] | undefined { - return this.options.descriptionHighlights; - } - getDetail(): string | undefined { - return this.options.detail; - } - getDetailHighlights(): Highlight[] | undefined { - return this.options.detailHighlights; - } - isHidden(): boolean { - return this.options.hidden || false; - } - getUri(): URI | undefined { - return this.options.uri; - } - getIconClass(): string | undefined { - return this.options.iconClass; - } - getKeybinding(): Keybinding | undefined { - return this.options.keybinding; - } - run(mode: QuickOpenMode): boolean { - if (!this.options.run) { - return false; - } - return this.options.run(mode); - } -} - -export class QuickOpenGroupItem extends QuickOpenItem { - - getGroupLabel(): string | undefined { - return this.options.groupLabel; - } - showBorder(): boolean { - return this.options.showBorder || false; - } -} - -export interface QuickOpenModel { - onType(lookFor: string, acceptor: (items: QuickOpenItem[], actionProvider?: QuickOpenActionProvider) => void): void; -} +import * as common from '../../common/quick-open-model'; + +/** + * @deprecated import from `@theia/core/lib/common/quick-open-model` instead + */ +export type Highlight = common.Highlight; + +/** + * @deprecated import from `@theia/core/lib/common/quick-open-model` instead + */ +export type QuickOpenItemOptions = common.QuickOpenItemOptions; + +/** + * @deprecated import from `@theia/core/lib/common/quick-open-model` instead + */ +export type QuickOpenGroupItemOptions = common.QuickOpenGroupItemOptions; + +/** + * @deprecated import from `@theia/core/lib/common/quick-open-model` instead + */ +export { QuickOpenItem, QuickOpenGroupItem, QuickOpenMode } from '../../common/quick-open-model'; + +/** + * @deprecated import from `@theia/core/lib/common/quick-open-model` instead + */ +export type QuickOpenModel = common.QuickOpenModel; diff --git a/packages/core/src/browser/quick-open/quick-pick-service-impl.ts b/packages/core/src/browser/quick-open/quick-pick-service-impl.ts index 0fcfd3665f860..38adbe068299e 100644 --- a/packages/core/src/browser/quick-open/quick-pick-service-impl.ts +++ b/packages/core/src/browser/quick-open/quick-pick-service-impl.ts @@ -31,23 +31,27 @@ export class QuickPickServiceImpl implements QuickPickService { @inject(QuickOpenService) protected readonly quickOpenService: QuickOpenService; + private items: QuickOpenItem[] = []; + show(elements: string[], options?: QuickPickOptions): Promise; show(elements: QuickPickItem[], options?: QuickPickOptions): Promise; async show(elements: (string | QuickPickItem)[], options?: QuickPickOptions): Promise { return new Promise(resolve => { - const items = this.toItems(elements, resolve); - if (items.length === 1) { - items[0].run(QuickOpenMode.OPEN); + this.items = this.toItems(elements, resolve); + if (this.items.length === 1) { + this.items[0].run(QuickOpenMode.OPEN); return; } - if (options && this.quickTitleBar.shouldShowTitleBar(options.title, options.step)) { - this.quickTitleBar.attachTitleBar(this.quickOpenService.widgetNode, options.title, options.step, options.totalSteps, options.buttons); - } const prefix = options && options.value ? options.value : ''; + let savedValue: string; this.quickOpenService.open({ - onType: (_, acceptor) => { - acceptor(items); - this.onDidChangeActiveItemsEmitter.fire(items); + onType: (value, acceptor) => { + acceptor(this.items); + if (savedValue !== value) { + this.onDidChangeValueEmitter.fire(value); + this.onDidChangeActiveItemsEmitter.fire(this.items); + savedValue = value; + } } }, Object.assign({ onClose: () => { @@ -58,6 +62,9 @@ export class QuickPickServiceImpl implements QuickPickService { fuzzyMatchDescription: true, prefix }, options)); + if (options && this.quickTitleBar.shouldShowTitleBar(options.title, options.step)) { + this.quickTitleBar.attachTitleBar(this.quickOpenService.widgetNode, options.title, options.step, options.totalSteps, options.buttons); + } }); } protected toItems(elements: (string | QuickPickItem)[], resolve: (element: Object) => void): QuickOpenItem[] { @@ -110,4 +117,11 @@ export class QuickPickServiceImpl implements QuickPickService { private readonly onDidChangeActiveItemsEmitter: Emitter[]> = new Emitter[]>(); readonly onDidChangeActiveItems: Event[]> = this.onDidChangeActiveItemsEmitter.event; + private readonly onDidChangeValueEmitter: Emitter = new Emitter(); + readonly onDidChangeValue: Event = this.onDidChangeValueEmitter.event; + + setItems(items: QuickOpenItem[]): void { + this.items = items; + this.quickOpenService.refresh(); + } } diff --git a/packages/core/src/browser/quick-open/quick-title-bar.ts b/packages/core/src/browser/quick-open/quick-title-bar.ts index cc86a9993f2dd..cd1194a608d01 100644 --- a/packages/core/src/browser/quick-open/quick-title-bar.ts +++ b/packages/core/src/browser/quick-open/quick-title-bar.ts @@ -14,26 +14,25 @@ * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 ********************************************************************************/ -import { Emitter } from '../../common/event'; -import { DisposableCollection } from '../../common/disposable'; +import { Emitter, Event } from '../../common/event'; import { injectable } from 'inversify'; -export enum QuickInputTitleButtonSide { +export enum QuickInputButtonSide { LEFT = 0, RIGHT = 1 } -export interface QuickInputTitleButton { +export interface QuickTitleButton { icon: string; // a background image coming from a url iconClass?: string; // a class such as one coming from font awesome tooltip?: string | undefined; - side: QuickInputTitleButtonSide + side: QuickInputButtonSide } @injectable() export class QuickTitleBar { - private readonly onDidTriggerButtonEmitter: Emitter; + private readonly onDidTriggerButtonEmitter: Emitter; private _isAttached: boolean; private titleElement: HTMLElement; @@ -43,21 +42,18 @@ export class QuickTitleBar { private _title: string | undefined; private _step: number | undefined; private _totalSteps: number | undefined; - private _buttons: ReadonlyArray; + private _buttons: ReadonlyArray; private tabIndex = 2; // Keep track of the tabIndex for the buttons - private disposableCollection: DisposableCollection; constructor() { this.titleElement = document.createElement('h3'); - this.titleElement.style.textAlign = 'center'; - this.titleElement.style.margin = '0'; + this.titleElement.className = QuickTitleBar.Styles.QUICK_TITLE_HEADER; - this.disposableCollection = new DisposableCollection(); - this.disposableCollection.push(this.onDidTriggerButtonEmitter = new Emitter()); + this.onDidTriggerButtonEmitter = new Emitter(); } - get onDidTriggerButton() { + get onDidTriggerButton(): Event { return this.onDidTriggerButtonEmitter.event; } @@ -96,7 +92,7 @@ export class QuickTitleBar { return this._totalSteps; } - set buttons(buttons: ReadonlyArray | undefined) { + set buttons(buttons: ReadonlyArray | undefined) { if (buttons === undefined) { this._buttons = []; return; @@ -105,7 +101,7 @@ export class QuickTitleBar { this._buttons = buttons; } - get buttons() { + get buttons(): ReadonlyArray | undefined { return this._buttons; } @@ -126,27 +122,26 @@ export class QuickTitleBar { } // Left buttons are for the buttons dervied from QuickInputButtons - private getLeftButtons() { + private getLeftButtons(): ReadonlyArray { if (this._buttons === undefined || this._buttons.length === 0) { return []; } - return this._buttons.filter(btn => btn.side === QuickInputTitleButtonSide.LEFT); + return this._buttons.filter(btn => btn.side === QuickInputButtonSide.LEFT); } - private getRightButtons() { + private getRightButtons(): ReadonlyArray { if (this._buttons === undefined || this._buttons.length === 0) { return []; } - return this._buttons.filter(btn => btn.side === QuickInputTitleButtonSide.RIGHT); + return this._buttons.filter(btn => btn.side === QuickInputButtonSide.RIGHT); } - private createButtonElement(buttons: ReadonlyArray) { + private createButtonElement(buttons: ReadonlyArray): HTMLDivElement { const buttonDiv = document.createElement('div'); - buttonDiv.style.display = 'inline-flex'; + buttonDiv.className = QuickTitleBar.Styles.QUICK_TITLE_BUTTON_DIV; for (const btn of buttons) { const aElement = document.createElement('a'); - aElement.style.width = '16px'; - aElement.style.height = '16px'; + aElement.className = QuickTitleBar.Styles.QUICK_TITLE_BUTTON; aElement.tabIndex = 0; if (btn.iconClass) { aElement.classList.add(...btn.iconClass.split(' ')); @@ -157,10 +152,6 @@ export class QuickTitleBar { } aElement.classList.add('icon'); - aElement.style.display = 'flex'; - aElement.style.justifyContent = 'center'; - aElement.style.alignItems = 'center'; - aElement.style.cursor = 'pointer'; aElement.tabIndex = this.tabIndex; aElement.title = btn.tooltip ? btn.tooltip : ''; aElement.onclick = () => { @@ -177,15 +168,9 @@ export class QuickTitleBar { return buttonDiv; } - private createTitleBarDiv() { + private createTitleBarDiv(): HTMLDivElement { const div = document.createElement('div'); - div.style.display = 'flex'; - div.style.flexDirection = 'row'; - div.style.fontSize = '13px'; - div.style.padding = '0px 1px'; - div.style.justifyContent = 'flex-start'; - div.style.alignItems = 'center'; - div.style.background = 'var(--theia-layout-color4)'; + div.className = QuickTitleBar.Styles.QUICK_TITLE_CONTAINER; div.onclick = event => { event.stopPropagation(); event.preventDefault(); @@ -193,26 +178,24 @@ export class QuickTitleBar { return div; } - private createLeftButtonDiv() { + private createLeftButtonDiv(): HTMLDivElement { const leftButtonDiv = document.createElement('div'); // Holds all the buttons that get added to the left - leftButtonDiv.style.flex = '1'; - leftButtonDiv.style.textAlign = 'left'; + leftButtonDiv.className = QuickTitleBar.Styles.QUICK_TITLE_LEFT_BAR; leftButtonDiv.appendChild(this.createButtonElement(this.getLeftButtons())); return leftButtonDiv; } - private createRightButtonDiv() { + private createRightButtonDiv(): HTMLDivElement { const rightButtonDiv = document.createElement('div'); - rightButtonDiv.style.flex = '1'; - rightButtonDiv.style.textAlign = 'right'; + rightButtonDiv.className = QuickTitleBar.Styles.QUICK_TITLE_RIGHT_BAR; rightButtonDiv.appendChild(this.createButtonElement(this.getRightButtons())); return rightButtonDiv; } // tslint:disable-next-line:max-line-length - public attachTitleBar(widgetNode: HTMLElement, title: string | undefined, step: number | undefined, totalSteps: number | undefined, buttons: ReadonlyArray | undefined) { + public attachTitleBar(widgetNode: HTMLElement, title: string | undefined, step: number | undefined, totalSteps: number | undefined, buttons: ReadonlyArray | undefined): void { const div = this.createTitleBarDiv(); this.updateInnerTitleText(); @@ -236,7 +219,7 @@ export class QuickTitleBar { this.isAttached = true; } - hide() { + hide(): void { this.title = undefined; this.buttons = undefined; this.step = undefined; @@ -252,8 +235,15 @@ export class QuickTitleBar { return ((title !== undefined) || (step !== undefined)); } - dispose() { - this.disposableCollection.dispose(); - } +} +export namespace QuickTitleBar { + export namespace Styles { + export const QUICK_TITLE_CONTAINER = 'theia-quick-title-container'; + export const QUICK_TITLE_LEFT_BAR = 'theia-quick-title-left-bar'; + export const QUICK_TITLE_RIGHT_BAR = 'theia-quick-title-right-bar'; + export const QUICK_TITLE_HEADER = 'theia-quick-title-header'; + export const QUICK_TITLE_BUTTON_DIV = 'theia-quick-title-button-div'; + export const QUICK_TITLE_BUTTON = 'theia-quick-title-button'; + } } diff --git a/packages/core/src/browser/style/index.css b/packages/core/src/browser/style/index.css index 68f19284eb438..e41a39c1ea078 100644 --- a/packages/core/src/browser/style/index.css +++ b/packages/core/src/browser/style/index.css @@ -200,3 +200,4 @@ textarea { @import './alert-messages.css'; @import './icons.css'; @import './widget.css'; +@import './quick-title-bar.css'; diff --git a/packages/core/src/browser/style/quick-title-bar.css b/packages/core/src/browser/style/quick-title-bar.css new file mode 100644 index 0000000000000..a3d9dfab69fb1 --- /dev/null +++ b/packages/core/src/browser/style/quick-title-bar.css @@ -0,0 +1,53 @@ +/******************************************************************************** + * Copyright (C) 2019 Red Hat, Inc. and others. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License v. 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0. + * + * This Source Code may also be made available under the following Secondary + * Licenses when the conditions for such availability set forth in the Eclipse + * Public License v. 2.0 are satisfied: GNU General Public License, version 2 + * with the GNU Classpath Exception which is available at + * https://www.gnu.org/software/classpath/license.html. + * + * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 + ********************************************************************************/ + +.theia-quick-title-container { + display: flex; + flex-direction: row; + font-size: 11px; + padding: 3px 3px; + justify-content: flex-start; + align-items: center; + background: var(--theia-layout-color4); +} + +.theia-quick-title-left-bar { + flex: 1px; + text-align: left; +} + +.theia-quick-title-right-bar { + flex: 1px; + text-align: right; +} + +.theia-quick-title-header { + text-align: center; + margin: 0; +} + +.theia-quick-title-button-div { + display: inline-flex; +} + +.theia-quick-title-button { + width: 16px; + height: 16px; + display: flex; + justify-content: center; + align-items: center; + cursor: pointer; +} diff --git a/packages/core/src/common/keybinding.ts b/packages/core/src/common/keybinding.ts new file mode 100644 index 0000000000000..98791f0fff496 --- /dev/null +++ b/packages/core/src/common/keybinding.ts @@ -0,0 +1,32 @@ +/******************************************************************************** + * Copyright (C) 2017 TypeFox and others. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License v. 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0. + * + * This Source Code may also be made available under the following Secondary + * Licenses when the conditions for such availability set forth in the Eclipse + * Public License v. 2.0 are satisfied: GNU General Public License, version 2 + * with the GNU Classpath Exception which is available at + * https://www.gnu.org/software/classpath/license.html. + * + * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 + ********************************************************************************/ + +export interface Keybinding { + /** Command identifier, this needs to be a unique string. */ + command: string; + /** Keybinding string as defined in packages/keymaps/README.md. */ + keybinding: string; + /** + * The optional keybinding context where this binding belongs to. + * If not specified, then this keybinding context belongs to the NOOP + * keybinding context. + */ + context?: string; + /** + * https://code.visualstudio.com/docs/getstarted/keybindings#_when-clause-contexts + */ + when?: string; +} diff --git a/packages/core/src/common/quick-open-model.ts b/packages/core/src/common/quick-open-model.ts new file mode 100644 index 0000000000000..fb38a1ab0bf1a --- /dev/null +++ b/packages/core/src/common/quick-open-model.ts @@ -0,0 +1,129 @@ +/******************************************************************************** + * Copyright (C) 2017 TypeFox and others. + * + * This program and the accompanying materials are made available under the + * terms of the Eclipse Public License v. 2.0 which is available at + * http://www.eclipse.org/legal/epl-2.0. + * + * This Source Code may also be made available under the following Secondary + * Licenses when the conditions for such availability set forth in the Eclipse + * Public License v. 2.0 are satisfied: GNU General Public License, version 2 + * with the GNU Classpath Exception which is available at + * https://www.gnu.org/software/classpath/license.html. + * + * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 + ********************************************************************************/ + +import URI from './uri'; +import { Keybinding } from './keybinding'; +import { Disposable } from './disposable'; + +export interface Highlight { + start: number + end: number +} + +export enum QuickOpenMode { + PREVIEW, + OPEN, + OPEN_IN_BACKGROUND +} + +export interface QuickOpenItemOptions { + tooltip?: string; + label?: string; + labelHighlights?: Highlight[]; + description?: string; + descriptionHighlights?: Highlight[]; + detail?: string; + detailHighlights?: Highlight[]; + hidden?: boolean; + uri?: URI; + iconClass?: string; + keybinding?: Keybinding; + run?(mode: QuickOpenMode): boolean; +} +export interface QuickOpenGroupItemOptions extends QuickOpenItemOptions { + groupLabel?: string; + showBorder?: boolean; +} + +export class QuickOpenItem { + + constructor( + protected options: T = {} as T + ) { } + + getTooltip(): string | undefined { + return this.options.tooltip || this.getLabel(); + } + getLabel(): string | undefined { + return this.options.label; + } + getLabelHighlights(): Highlight[] { + return this.options.labelHighlights || []; + } + getDescription(): string | undefined { + return this.options.description; + } + getDescriptionHighlights(): Highlight[] | undefined { + return this.options.descriptionHighlights; + } + getDetail(): string | undefined { + return this.options.detail; + } + getDetailHighlights(): Highlight[] | undefined { + return this.options.detailHighlights; + } + isHidden(): boolean { + return this.options.hidden || false; + } + getUri(): URI | undefined { + return this.options.uri; + } + getIconClass(): string | undefined { + return this.options.iconClass; + } + getKeybinding(): Keybinding | undefined { + return this.options.keybinding; + } + run(mode: QuickOpenMode): boolean { + if (!this.options.run) { + return false; + } + return this.options.run(mode); + } +} + +export class QuickOpenGroupItem extends QuickOpenItem { + + getGroupLabel(): string | undefined { + return this.options.groupLabel; + } + showBorder(): boolean { + return this.options.showBorder || false; + } +} + +export interface QuickOpenModel { + onType(lookFor: string, acceptor: (items: QuickOpenItem[], actionProvider?: QuickOpenActionProvider) => void): void; +} + +export interface QuickOpenActionProvider { + hasActions(item: QuickOpenItem): boolean; + getActions(item: QuickOpenItem): Promise; +} + +export interface QuickOpenActionOptions { + id: string; + label?: string; + tooltip?: string; + class?: string | undefined; + enabled?: boolean; + checked?: boolean; + radio?: boolean; +} + +export interface QuickOpenAction extends QuickOpenActionOptions, Disposable { + run(item?: QuickOpenItem): PromiseLike; +} diff --git a/packages/core/src/common/quick-pick-service.ts b/packages/core/src/common/quick-pick-service.ts index 7a9a3aa028c79..7256381075058 100644 --- a/packages/core/src/common/quick-pick-service.ts +++ b/packages/core/src/common/quick-pick-service.ts @@ -15,7 +15,7 @@ ********************************************************************************/ import { QuickOpenHideReason } from './quick-open-service'; -import { QuickInputTitleButton } from '../browser/quick-open/quick-title-bar'; +import { QuickTitleButton } from '../browser/quick-open/quick-title-bar'; import { Event } from '../common/event'; import { QuickOpenItem, QuickOpenItemOptions } from '../browser/quick-open/quick-open-model'; @@ -68,7 +68,7 @@ export interface QuickPickOptions { /** * Buttons that are displayed on the title panel */ - buttons?: ReadonlyArray + buttons?: ReadonlyArray /** * Set to `true` to keep the input box open when focus moves to another part of the editor or to another window. @@ -89,8 +89,11 @@ export interface QuickPickService { show(elements: QuickPickItem[], options?: QuickPickOptions): Promise; + setItems(items: QuickOpenItem[]): void; + hide(reason?: QuickOpenHideReason): void readonly onDidAccept: Event; + readonly onDidChangeValue: Event; readonly onDidChangeActiveItems: Event[]>; } diff --git a/packages/monaco/src/browser/monaco-quick-open-service.ts b/packages/monaco/src/browser/monaco-quick-open-service.ts index 8ac1a81de234b..c61576b6d2e71 100644 --- a/packages/monaco/src/browser/monaco-quick-open-service.ts +++ b/packages/monaco/src/browser/monaco-quick-open-service.ts @@ -113,6 +113,9 @@ export class MonacoQuickOpenService extends QuickOpenService { } internalOpen(opts: MonacoQuickOpenControllerOpts): void { + if (this.widgetNode && this.widgetNode.offsetParent !== null) { + this.hide(); + } this.opts = opts; const activeContext = window.document.activeElement || undefined; if (!activeContext || !this.container.contains(activeContext)) { @@ -134,7 +137,7 @@ export class MonacoQuickOpenService extends QuickOpenService { } } - setValueSelected(value: string | undefined, selectLocation: Readonly<[number, number]> | undefined) { + setValueSelected(value: string | undefined, selectLocation: Readonly<[number, number]> | undefined): void { if (!value) { return; } @@ -156,14 +159,14 @@ export class MonacoQuickOpenService extends QuickOpenService { } } - setEnabled(isEnabled: boolean | undefined) { + setEnabled(isEnabled: boolean | undefined): void { const widget = this.widget; if (widget.inputBox) { widget.inputBox.inputElement.readOnly = (isEnabled !== undefined) ? !isEnabled : false; } } - setValue(value: string | undefined) { + setValue(value: string | undefined): void { if (this.widget && this.widget.inputBox) { this.widget.inputBox.inputElement.value = (value !== undefined) ? value : ''; } diff --git a/packages/plugin-ext/src/api/plugin-api.ts b/packages/plugin-ext/src/api/plugin-api.ts index 020a63e42bae9..7c72dd07dfaba 100644 --- a/packages/plugin-ext/src/api/plugin-api.ts +++ b/packages/plugin-ext/src/api/plugin-api.ts @@ -74,7 +74,7 @@ import { SymbolInformation } from 'vscode-languageserver-types'; import { ScmCommand } from '@theia/scm/lib/browser/scm-provider'; import { ArgumentProcessor } from '../plugin/command-registry'; import { MaybePromise } from '@theia/core/lib/common/types'; -import { QuickInputTitleButton } from '@theia/core/lib/browser/quick-open/quick-title-bar'; +import { QuickTitleButton } from '@theia/core/lib/browser/quick-open/quick-title-bar'; import { QuickOpenItem, QuickOpenItemOptions } from '@theia/core/lib/browser/quick-open/quick-open-model'; export interface PluginInitData { @@ -305,7 +305,7 @@ export interface QuickOpenExt { $acceptOnDidAccept(quickInputNumber: number): Promise; $acceptDidChangeValue(quickInputNumber: number, changedValue: string): Promise; $acceptOnDidHide(quickInputNumber: number): Promise; - $acceptOnDidTriggerButton(quickInputNumber: number, btn: QuickInputTitleButton): Promise; + $acceptOnDidTriggerButton(quickInputNumber: number, btn: QuickTitleButton): Promise; $acceptDidChangeActive(quickInputNumber: number, changedItems: QuickOpenItem[]): Promise; $acceptDidChangeSelection(quickInputNumber: number, selection: string): Promise; } @@ -394,11 +394,11 @@ export interface WorkspaceFolderPickOptionsMain { ignoreFocusOut?: boolean; } -export interface QuickInputTitleButtonHandle extends QuickInputTitleButton { +export interface QuickInputTitleButtonHandle extends QuickTitleButton { index: number; // index of where they are in buttons array if QuickInputButton or -1 if QuickInputButtons.Back } -export interface ITransferQuickInput { +export interface TransferQuickInput { quickInputIndex: number; title: string | undefined; step: number | undefined; @@ -408,7 +408,7 @@ export interface ITransferQuickInput { ignoreFocusOut: boolean; } -export interface ITransferInputBox extends ITransferQuickInput { +export interface TransferInputBox extends TransferQuickInput { value: string; placeholder: string | undefined; password: boolean; @@ -418,7 +418,7 @@ export interface ITransferInputBox extends ITransferQuickInput { validateInput(value: string): MaybePromise; } -export interface ITransferQuickPick extends ITransferQuickInput { +export interface TransferQuickPick extends TransferQuickInput { value: string; placeholder: string | undefined; buttons: ReadonlyArray; @@ -436,8 +436,8 @@ export interface QuickOpenMain { $setError(error: Error): Promise; $input(options: theia.InputBoxOptions, validateInput: boolean): Promise; $hide(): void; - $showInputBox(inputBox: ITransferInputBox, validateInput: boolean): void; - $showCustomQuickPick(inputBox: ITransferQuickPick): void; + $showInputBox(inputBox: TransferInputBox, validateInput: boolean): void; + $showCustomQuickPick(inputBox: TransferQuickPick): void; $setQuickInputChanged(changed: object): void; $refreshQuickInput(): void; } diff --git a/packages/plugin-ext/src/main/browser/quick-open-main.ts b/packages/plugin-ext/src/main/browser/quick-open-main.ts index 445d851b01ced..76d11ede89493 100644 --- a/packages/plugin-ext/src/main/browser/quick-open-main.ts +++ b/packages/plugin-ext/src/main/browser/quick-open-main.ts @@ -16,16 +16,21 @@ import { InputBoxOptions, QuickPickItem as QuickPickItemExt } from '@theia/plugin'; import { interfaces } from 'inversify'; -import { QuickOpenModel, QuickOpenItem, QuickOpenMode } from '@theia/core/lib/browser/quick-open/quick-open-model'; +import { + QuickOpenModel, + QuickOpenItem, + QuickOpenMode, + QuickOpenItemOptions +} from '@theia/core/lib/browser/quick-open/quick-open-model'; import { RPCProtocol } from '../../api/rpc-protocol'; -import { QuickOpenExt, QuickOpenMain, MAIN_RPC_CONTEXT, PickOptions, PickOpenItem, ITransferInputBox, QuickInputTitleButtonHandle, ITransferQuickPick } from '../../api/plugin-api'; +import { QuickOpenExt, QuickOpenMain, MAIN_RPC_CONTEXT, PickOptions, PickOpenItem, TransferInputBox, QuickInputTitleButtonHandle, TransferQuickPick } from '../../api/plugin-api'; import { MonacoQuickOpenService } from '@theia/monaco/lib/browser/monaco-quick-open-service'; import { QuickInputService, FOLDER_ICON, FILE_ICON } from '@theia/core/lib/browser'; import { PluginSharedStyle } from './plugin-shared-style'; import URI from 'vscode-uri'; import { ThemeIcon, QuickInputButton } from '../../plugin/types-impl'; import { QuickPickService, QuickPickItem, QuickPickValue } from '@theia/core/lib/common/quick-pick-service'; -import { QuickTitleBar, QuickInputTitleButtonSide } from '@theia/core/lib/browser/quick-open/quick-title-bar'; +import { QuickTitleBar, QuickInputButtonSide } from '@theia/core/lib/browser/quick-open/quick-title-bar'; import { DisposableCollection } from '@theia/core/lib/common/disposable'; export class QuickOpenMainImpl implements QuickOpenMain, QuickOpenModel { @@ -152,7 +157,7 @@ export class QuickOpenMainImpl implements QuickOpenMain, QuickOpenModel { icon: newIcon, iconClass: newIconClass, tooltip: quickInputButton.tooltip, - side: isDefaultQuickInputButton ? QuickInputTitleButtonSide.LEFT : QuickInputTitleButtonSide.RIGHT, + side: isDefaultQuickInputButton ? QuickInputButtonSide.LEFT : QuickInputButtonSide.RIGHT, index: isDefaultQuickInputButton ? -1 : index }; } @@ -174,7 +179,7 @@ export class QuickOpenMainImpl implements QuickOpenMain, QuickOpenModel { } } - $showInputBox(inputBox: ITransferInputBox, validateInput: boolean): void { + $showInputBox(inputBox: TransferInputBox, validateInput: boolean): void { if (validateInput) { inputBox.validateInput = val => this.proxy.$validateInput(val); } @@ -211,7 +216,7 @@ export class QuickOpenMainImpl implements QuickOpenMain, QuickOpenModel { } // tslint:disable-next-line:no-any - private findChangedKey(key: string, value: any) { + private findChangedKey(key: string, value: any): void { switch (key) { case 'title': { this.quickTitleBar.title = value; @@ -245,11 +250,15 @@ export class QuickOpenMainImpl implements QuickOpenMain, QuickOpenModel { this.delegate.setPlaceHolder(value); break; } + case 'items': { + this.quickPick.setItems(value.map((options: QuickOpenItemOptions) => new QuickOpenItem(options))); + break; + } } } // tslint:disable-next-line:no-any - $setQuickInputChanged(changed: any) { + $setQuickInputChanged(changed: any): void { for (const key in changed) { if (changed.hasOwnProperty(key)) { const value = changed[key]; @@ -257,11 +266,11 @@ export class QuickOpenMainImpl implements QuickOpenMain, QuickOpenModel { } } } - $refreshQuickInput() { + $refreshQuickInput(): void { this.quickInput.refresh(); } - $showCustomQuickPick(options: ITransferQuickPick): void { + $showCustomQuickPick(options: TransferQuickPick): void { const items = this.convertPickOpenItemToQuickOpenItem(options.items); const quickPick = this.quickPick.show(items, { buttons: options.buttons.map((btn, i) => this.convertQuickInputButton(btn, i)), @@ -278,6 +287,7 @@ export class QuickOpenMainImpl implements QuickOpenMain, QuickOpenModel { const disposableListeners = new DisposableCollection(); disposableListeners.push(this.quickPick.onDidAccept(() => this.proxy.$acceptOnDidAccept(options.quickInputIndex))); disposableListeners.push(this.quickPick.onDidChangeActiveItems(changedItems => this.proxy.$acceptDidChangeActive(options.quickInputIndex, changedItems))); + disposableListeners.push(this.quickPick.onDidChangeValue(value => this.proxy.$acceptDidChangeValue(options.quickInputIndex, value))); disposableListeners.push(this.quickTitleBar.onDidTriggerButton(button => { this.proxy.$acceptOnDidTriggerButton(options.quickInputIndex, button); })); diff --git a/packages/plugin-ext/src/plugin/quick-open.ts b/packages/plugin-ext/src/plugin/quick-open.ts index dca47be90b92d..62f11fd609524 100644 --- a/packages/plugin-ext/src/plugin/quick-open.ts +++ b/packages/plugin-ext/src/plugin/quick-open.ts @@ -13,7 +13,7 @@ * * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0 ********************************************************************************/ -import { QuickOpenExt, PLUGIN_RPC_CONTEXT as Ext, QuickOpenMain, ITransferInputBox, Plugin } from '../api/plugin-api'; +import { QuickOpenExt, PLUGIN_RPC_CONTEXT as Ext, QuickOpenMain, TransferInputBox, Plugin } from '../api/plugin-api'; import { QuickPickOptions, QuickPickItem, InputBoxOptions, InputBox, QuickPick, QuickInput } from '@theia/plugin'; import { CancellationToken } from '@theia/core/lib/common/cancellation'; import { RPCProtocol } from '../api/rpc-protocol'; @@ -22,7 +22,7 @@ import { hookCancellationToken } from '../api/async-util'; import { Emitter, Event } from '@theia/core/lib/common/event'; import { DisposableCollection } from '@theia/core/lib/common/disposable'; import { QuickInputButtons, QuickInputButton, ThemeIcon } from './types-impl'; -import { QuickInputTitleButtonHandle, ITransferQuickPick, PluginPackage } from '../common'; +import { QuickInputTitleButtonHandle, TransferQuickPick, PluginPackage } from '../common'; import URI from 'vscode-uri'; import * as path from 'path'; import { quickPickItemToPickOpenItem } from './type-converters'; @@ -102,7 +102,7 @@ export class QuickOpenExtImpl implements QuickOpenExt { return hookCancellationToken(token, promise); } - showCustomQuickPick(options: ITransferQuickPick): void { + showCustomQuickPick(options: TransferQuickPick): void { this.proxy.$showCustomQuickPick(options); } @@ -129,7 +129,7 @@ export class QuickOpenExtImpl implements QuickOpenExt { hide(): void { this.proxy.$hide(); } - showInputBox(options: ITransferInputBox): void { + showInputBox(options: TransferInputBox): void { this.validateInputHandler = options && options.validateInput; this.proxy.$showInputBox(options, typeof this.validateInputHandler === 'function'); } @@ -291,15 +291,15 @@ export class QuickInputExt implements QuickInput { this.update({ value }); } - show() { + show(): void { throw new Error('Method implementation must be provided by extenders'); } - dispose() { + dispose(): void { this.disposableCollection.dispose(); } - protected update(changed: object) { + protected update(changed: object): void { /** * The args are just going to be set when we call show for the first time. * We return early when its invisible to avoid race condition @@ -341,19 +341,19 @@ export class QuickInputExt implements QuickInput { } } - _fireAccept() { + _fireAccept(): void { this.onDidAcceptEmitter.fire(undefined); } - _fireChangedValue(changedValue: string) { + _fireChangedValue(changedValue: string): void { this.onDidChangeValueEmitter.fire(changedValue); } - _fireHide() { + _fireHide(): void { this.onDidHideEmitter.fire(undefined); } - _fireButtonTrigger(btn: QuickInputButton) { + _fireButtonTrigger(btn: QuickInputButton): void { this.onDidTriggerButtonEmitter.fire(btn); } @@ -550,11 +550,11 @@ export class QuickPickExt extends QuickInputExt impleme this.update({ placeholder }); } - _fireChangedSelection(selections: T[]) { + _fireChangedSelection(selections: T[]): void { this.onDidChangeSelectionEmitter.fire(selections); } - _fireChangedActiveItem(changedItems: QuickOpenItem[]) { + _fireChangedActiveItem(changedItems: QuickOpenItem[]): void { this.onDidChangeActiveEmitter.fire(changedItems as unknown as T[]); } @@ -567,6 +567,7 @@ export class QuickPickExt extends QuickInputExt impleme } show(): void { + this.visible = true; this.quickOpen.showCustomQuickPick({ quickInputIndex: this.quickInputIndex, title: this.title, diff --git a/packages/plugin-ext/src/plugin/type-converters.ts b/packages/plugin-ext/src/plugin/type-converters.ts index 33c88d0aba11d..c5ba91f9abdc6 100644 --- a/packages/plugin-ext/src/plugin/type-converters.ts +++ b/packages/plugin-ext/src/plugin/type-converters.ts @@ -871,7 +871,7 @@ export function fromColorPresentation(colorPresentation: theia.ColorPresentation }; } -export function quickPickItemToPickOpenItem(items: Item[]) { +export function quickPickItemToPickOpenItem(items: Item[]): PickOpenItem[] { const pickItems: PickOpenItem[] = []; for (let handle = 0; handle < items.length; handle++) { const item = items[handle];