From 063285778a89c69b0491c96e542cc9145f4659b3 Mon Sep 17 00:00:00 2001 From: vince-fugnitto Date: Tue, 14 Feb 2023 12:39:09 -0500 Subject: [PATCH 1/4] webview: add support for `activeWebviewPanelId` The commit adds support for the `activeWebviewPanelId` when-context-clause. Signed-off-by: vince-fugnitto --- .../src/main/browser/webview/webview.ts | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/packages/plugin-ext/src/main/browser/webview/webview.ts b/packages/plugin-ext/src/main/browser/webview/webview.ts index 1cd19c66177a5..728bc8196f38a 100644 --- a/packages/plugin-ext/src/main/browser/webview/webview.ts +++ b/packages/plugin-ext/src/main/browser/webview/webview.ts @@ -50,6 +50,7 @@ import { FileOperationError, FileOperationResult } from '@theia/filesystem/lib/c import { BinaryBufferReadableStream } from '@theia/core/lib/common/buffer'; import { ViewColumn } from '../../../plugin/types-impl'; import { ExtractableWidget } from '@theia/core/lib/browser/widgets/extractable-widget'; +import { ContextKey, ContextKeyService } from '@theia/core/lib/browser/context-key-service'; // Style from core const TRANSPARENT_OVERLAY_STYLE = 'theia-transparent-overlay'; @@ -105,6 +106,11 @@ export class WebviewWidget extends BaseWidget implements StatefulWidget, Extract // On `mousedown` we put a transparent div over the `iframe` to avoid losing the mouse tacking. protected transparentOverlay: HTMLElement; + protected _activeWebviewPanelId: ContextKey; + get activeWebviewPanelId(): ContextKey { + return this._activeWebviewPanelId; + } + @inject(WebviewWidgetIdentifier) readonly identifier: WebviewWidgetIdentifier; @@ -144,6 +150,9 @@ export class WebviewWidget extends BaseWidget implements StatefulWidget, Extract @inject(WebviewResourceCache) protected readonly resourceCache: WebviewResourceCache; + @inject(ContextKeyService) + protected readonly contextKeyService: ContextKeyService; + viewState: WebviewPanelViewState = { visible: false, active: false, @@ -202,6 +211,15 @@ export class WebviewWidget extends BaseWidget implements StatefulWidget, Extract this.transparentOverlay.style.display = 'none'; } })); + + this._activeWebviewPanelId = this.contextKeyService.createKey('activeWebviewPanelId', undefined); + this.toDispose.push(this.onDidChangeVisibility(() => { + if (this.isVisible) { + this.activeWebviewPanelId.set(this.viewType); + } else { + this.activeWebviewPanelId.set(undefined); + } + })); } protected override onBeforeAttach(msg: Message): void { From bf02f2291506eb67ccac2a71c7c2ea50884c364d Mon Sep 17 00:00:00 2001 From: Paul Marechal Date: Tue, 14 Feb 2023 17:30:26 -0500 Subject: [PATCH 2/4] move logic out from `WebviewWidget` Instead of having every `WebviewWidget` try to update the same context key, this commit adds a new component that listens for widget updates once and updates the context key accordingly. --- .../browser/plugin-ext-frontend-module.ts | 3 ++ .../browser/webview/webview-context-keys.ts | 49 +++++++++++++++++++ .../src/main/browser/webview/webview.ts | 18 ------- 3 files changed, 52 insertions(+), 18 deletions(-) create mode 100644 packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts diff --git a/packages/plugin-ext/src/main/browser/plugin-ext-frontend-module.ts b/packages/plugin-ext/src/main/browser/plugin-ext-frontend-module.ts index 2d72b2c6bd39c..411fdd04e8f91 100644 --- a/packages/plugin-ext/src/main/browser/plugin-ext-frontend-module.ts +++ b/packages/plugin-ext/src/main/browser/plugin-ext-frontend-module.ts @@ -82,6 +82,7 @@ import { PluginMenuCommandAdapter } from './menus/plugin-menu-command-adapter'; import './theme-icon-override'; import { PluginTerminalRegistry } from './plugin-terminal-registry'; import { DnDFileContentStore } from './view/dnd-file-content-store'; +import { WebviewContextKeys } from './webview/webview-context-keys'; export default new ContainerModule((bind, unbind, isBound, rebind) => { @@ -177,6 +178,8 @@ export default new ContainerModule((bind, unbind, isBound, rebind) => { bind(WebviewWidget).toSelf(); bind(WebviewWidgetFactory).toDynamicValue(ctx => new WebviewWidgetFactory(ctx.container)).inSingletonScope(); bind(WidgetFactory).toService(WebviewWidgetFactory); + bind(WebviewContextKeys).toSelf().inSingletonScope(); + bind(FrontendApplicationContribution).toService(WebviewContextKeys); bind(CustomEditorContribution).toSelf().inSingletonScope(); bind(CommandContribution).toService(CustomEditorContribution); diff --git a/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts b/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts new file mode 100644 index 0000000000000..e9f78186647d7 --- /dev/null +++ b/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts @@ -0,0 +1,49 @@ +// ***************************************************************************** +// Copyright (C) 2023 Ericsson 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 { inject, injectable, postConstruct } from '@theia/core/shared/inversify'; +import { ContextKey, ContextKeyService } from '@theia/core/lib/browser/context-key-service'; +import { ApplicationShell, FocusTracker, Widget } from '@theia/core/lib/browser'; +import { WebviewWidget } from './webview'; + +@injectable() +export class WebviewContextKeys { + + /** + * Context key representing the `viewType` of the active `WebviewWidget`, if any. + */ + activeWebviewPanelId: ContextKey; + + @inject(ApplicationShell) + protected applicationShell: ApplicationShell; + + @inject(ContextKeyService) + protected contextKeyService: ContextKeyService; + + @postConstruct() + protected postContruct(): void { + this.activeWebviewPanelId = this.contextKeyService.createKey('activeWebviewPanelId', undefined); + this.applicationShell.onDidChangeActiveWidget(this.handleDidChangeActiveWidget, this); + } + + protected handleDidChangeActiveWidget(change: FocusTracker.IChangedArgs): void { + if (change.newValue instanceof WebviewWidget) { + this.activeWebviewPanelId.set(change.newValue.viewType); + } else { + this.activeWebviewPanelId.set(undefined); + } + } +} diff --git a/packages/plugin-ext/src/main/browser/webview/webview.ts b/packages/plugin-ext/src/main/browser/webview/webview.ts index 728bc8196f38a..1cd19c66177a5 100644 --- a/packages/plugin-ext/src/main/browser/webview/webview.ts +++ b/packages/plugin-ext/src/main/browser/webview/webview.ts @@ -50,7 +50,6 @@ import { FileOperationError, FileOperationResult } from '@theia/filesystem/lib/c import { BinaryBufferReadableStream } from '@theia/core/lib/common/buffer'; import { ViewColumn } from '../../../plugin/types-impl'; import { ExtractableWidget } from '@theia/core/lib/browser/widgets/extractable-widget'; -import { ContextKey, ContextKeyService } from '@theia/core/lib/browser/context-key-service'; // Style from core const TRANSPARENT_OVERLAY_STYLE = 'theia-transparent-overlay'; @@ -106,11 +105,6 @@ export class WebviewWidget extends BaseWidget implements StatefulWidget, Extract // On `mousedown` we put a transparent div over the `iframe` to avoid losing the mouse tacking. protected transparentOverlay: HTMLElement; - protected _activeWebviewPanelId: ContextKey; - get activeWebviewPanelId(): ContextKey { - return this._activeWebviewPanelId; - } - @inject(WebviewWidgetIdentifier) readonly identifier: WebviewWidgetIdentifier; @@ -150,9 +144,6 @@ export class WebviewWidget extends BaseWidget implements StatefulWidget, Extract @inject(WebviewResourceCache) protected readonly resourceCache: WebviewResourceCache; - @inject(ContextKeyService) - protected readonly contextKeyService: ContextKeyService; - viewState: WebviewPanelViewState = { visible: false, active: false, @@ -211,15 +202,6 @@ export class WebviewWidget extends BaseWidget implements StatefulWidget, Extract this.transparentOverlay.style.display = 'none'; } })); - - this._activeWebviewPanelId = this.contextKeyService.createKey('activeWebviewPanelId', undefined); - this.toDispose.push(this.onDidChangeVisibility(() => { - if (this.isVisible) { - this.activeWebviewPanelId.set(this.viewType); - } else { - this.activeWebviewPanelId.set(undefined); - } - })); } protected override onBeforeAttach(msg: Message): void { From e1820b53a0e1cc95a2336263f191bfe1640280fe Mon Sep 17 00:00:00 2001 From: vince-fugnitto Date: Wed, 15 Feb 2023 14:13:54 -0500 Subject: [PATCH 3/4] fix postConstruct and event Signed-off-by: vince-fugnitto --- .../src/main/browser/webview/webview-context-keys.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts b/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts index e9f78186647d7..808cb0d67484d 100644 --- a/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts +++ b/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts @@ -34,9 +34,9 @@ export class WebviewContextKeys { protected contextKeyService: ContextKeyService; @postConstruct() - protected postContruct(): void { + protected postConstruct(): void { this.activeWebviewPanelId = this.contextKeyService.createKey('activeWebviewPanelId', undefined); - this.applicationShell.onDidChangeActiveWidget(this.handleDidChangeActiveWidget, this); + this.applicationShell.onDidChangeCurrentWidget(this.handleDidChangeActiveWidget, this); } protected handleDidChangeActiveWidget(change: FocusTracker.IChangedArgs): void { From 9b47318dc66e2fced218225128bafc3324e3fb47 Mon Sep 17 00:00:00 2001 From: vince-fugnitto Date: Wed, 15 Feb 2023 14:19:24 -0500 Subject: [PATCH 4/4] update listener name Signed-off-by: vince-fugnitto --- .../src/main/browser/webview/webview-context-keys.ts | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts b/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts index 808cb0d67484d..36c2ebbd1d239 100644 --- a/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts +++ b/packages/plugin-ext/src/main/browser/webview/webview-context-keys.ts @@ -25,7 +25,7 @@ export class WebviewContextKeys { /** * Context key representing the `viewType` of the active `WebviewWidget`, if any. */ - activeWebviewPanelId: ContextKey; + activeWebviewPanelId: ContextKey; @inject(ApplicationShell) protected applicationShell: ApplicationShell; @@ -35,15 +35,15 @@ export class WebviewContextKeys { @postConstruct() protected postConstruct(): void { - this.activeWebviewPanelId = this.contextKeyService.createKey('activeWebviewPanelId', undefined); - this.applicationShell.onDidChangeCurrentWidget(this.handleDidChangeActiveWidget, this); + this.activeWebviewPanelId = this.contextKeyService.createKey('activeWebviewPanelId', ''); + this.applicationShell.onDidChangeCurrentWidget(this.handleDidChangeCurrentWidget, this); } - protected handleDidChangeActiveWidget(change: FocusTracker.IChangedArgs): void { + protected handleDidChangeCurrentWidget(change: FocusTracker.IChangedArgs): void { if (change.newValue instanceof WebviewWidget) { this.activeWebviewPanelId.set(change.newValue.viewType); } else { - this.activeWebviewPanelId.set(undefined); + this.activeWebviewPanelId.set(''); } } }