mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-24 17:23:05 -05:00
150 lines
5.2 KiB
TypeScript
150 lines
5.2 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 { UriComponents, URI } from 'vs/base/common/uri';
|
|
import * as modes from 'vs/editor/common/modes';
|
|
import { MainContext, MainThreadEditorInsetsShape, IExtHostContext, ExtHostEditorInsetsShape, ExtHostContext } from 'vs/workbench/api/common/extHost.protocol';
|
|
import { extHostNamedCustomer } from '../common/extHostCustomers';
|
|
import { ICodeEditorService } from 'vs/editor/browser/services/codeEditorService';
|
|
import { IWebviewService, WebviewElement } from 'vs/workbench/contrib/webview/browser/webview';
|
|
import { DisposableStore } from 'vs/base/common/lifecycle';
|
|
import { IActiveCodeEditor, IViewZone } from 'vs/editor/browser/editorBrowser';
|
|
import { ExtensionIdentifier } from 'vs/platform/extensions/common/extensions';
|
|
import { isEqual } from 'vs/base/common/resources';
|
|
|
|
// todo@joh move these things back into something like contrib/insets
|
|
class EditorWebviewZone implements IViewZone {
|
|
|
|
readonly domNode: HTMLElement;
|
|
readonly afterLineNumber: number;
|
|
readonly afterColumn: number;
|
|
readonly heightInLines: number;
|
|
|
|
private _id?: string;
|
|
// suppressMouseDown?: boolean | undefined;
|
|
// heightInPx?: number | undefined;
|
|
// minWidthInPx?: number | undefined;
|
|
// marginDomNode?: HTMLElement | null | undefined;
|
|
// onDomNodeTop?: ((top: number) => void) | undefined;
|
|
// onComputedHeight?: ((height: number) => void) | undefined;
|
|
|
|
constructor(
|
|
readonly editor: IActiveCodeEditor,
|
|
readonly line: number,
|
|
readonly height: number,
|
|
readonly webview: WebviewElement,
|
|
) {
|
|
this.domNode = document.createElement('div');
|
|
this.domNode.style.zIndex = '10'; // without this, the webview is not interactive
|
|
this.afterLineNumber = line;
|
|
this.afterColumn = 1;
|
|
this.heightInLines = height;
|
|
|
|
editor.changeViewZones(accessor => this._id = accessor.addZone(this));
|
|
webview.mountTo(this.domNode);
|
|
}
|
|
|
|
dispose(): void {
|
|
this.editor.changeViewZones(accessor => this._id && accessor.removeZone(this._id));
|
|
}
|
|
}
|
|
|
|
@extHostNamedCustomer(MainContext.MainThreadEditorInsets)
|
|
export class MainThreadEditorInsets implements MainThreadEditorInsetsShape {
|
|
|
|
private readonly _proxy: ExtHostEditorInsetsShape;
|
|
private readonly _disposables = new DisposableStore();
|
|
private readonly _insets = new Map<number, EditorWebviewZone>();
|
|
|
|
constructor(
|
|
context: IExtHostContext,
|
|
@ICodeEditorService private readonly _editorService: ICodeEditorService,
|
|
@IWebviewService private readonly _webviewService: IWebviewService,
|
|
) {
|
|
this._proxy = context.getProxy(ExtHostContext.ExtHostEditorInsets);
|
|
}
|
|
|
|
dispose(): void {
|
|
this._disposables.dispose();
|
|
}
|
|
|
|
async $createEditorInset(handle: number, id: string, uri: UriComponents, line: number, height: number, options: modes.IWebviewOptions, extensionId: ExtensionIdentifier, extensionLocation: UriComponents): Promise<void> {
|
|
|
|
let editor: IActiveCodeEditor | undefined;
|
|
id = id.substr(0, id.indexOf(',')); //todo@joh HACK
|
|
|
|
for (const candidate of this._editorService.listCodeEditors()) {
|
|
if (candidate.getId() === id && candidate.hasModel() && isEqual(candidate.getModel().uri, URI.revive(uri))) {
|
|
editor = candidate;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (!editor) {
|
|
setTimeout(() => this._proxy.$onDidDispose(handle));
|
|
return;
|
|
}
|
|
|
|
const disposables = new DisposableStore();
|
|
|
|
const webview = this._webviewService.createWebviewElement('' + handle, {
|
|
enableFindWidget: false,
|
|
}, {
|
|
allowScripts: options.enableScripts,
|
|
localResourceRoots: options.localResourceRoots ? options.localResourceRoots.map(uri => URI.revive(uri)) : undefined
|
|
}, { id: extensionId, location: URI.revive(extensionLocation) });
|
|
|
|
const webviewZone = new EditorWebviewZone(editor, line, height, webview);
|
|
|
|
const remove = () => {
|
|
disposables.dispose();
|
|
this._proxy.$onDidDispose(handle);
|
|
this._insets.delete(handle);
|
|
};
|
|
|
|
disposables.add(editor.onDidChangeModel(remove));
|
|
disposables.add(editor.onDidDispose(remove));
|
|
disposables.add(webviewZone);
|
|
disposables.add(webview);
|
|
disposables.add(webview.onMessage(msg => this._proxy.$onDidReceiveMessage(handle, msg)));
|
|
|
|
this._insets.set(handle, webviewZone);
|
|
}
|
|
|
|
$disposeEditorInset(handle: number): void {
|
|
const inset = this.getInset(handle);
|
|
this._insets.delete(handle);
|
|
inset.dispose();
|
|
|
|
}
|
|
|
|
$setHtml(handle: number, value: string): void {
|
|
const inset = this.getInset(handle);
|
|
inset.webview.html = value;
|
|
}
|
|
|
|
$setOptions(handle: number, options: modes.IWebviewOptions): void {
|
|
const inset = this.getInset(handle);
|
|
inset.webview.contentOptions = {
|
|
...options,
|
|
localResourceRoots: options.localResourceRoots?.map(components => URI.from(components)),
|
|
};
|
|
}
|
|
|
|
async $postMessage(handle: number, value: any): Promise<boolean> {
|
|
const inset = this.getInset(handle);
|
|
inset.webview.postMessage(value);
|
|
return true;
|
|
}
|
|
|
|
private getInset(handle: number): EditorWebviewZone {
|
|
const inset = this._insets.get(handle);
|
|
if (!inset) {
|
|
throw new Error('Unknown inset');
|
|
}
|
|
return inset;
|
|
}
|
|
}
|