/*--------------------------------------------------------------------------------------------- * Copyright (c) Microsoft Corporation. All rights reserved. * Licensed under the Source EULA. See License.txt in the project root for license information. *--------------------------------------------------------------------------------------------*/ import { extHostNamedCustomer } from 'vs/workbench/api/common/extHostCustomers'; import { MainContext, MainThreadNotebookShape, NotebookExtensionDescription, IExtHostContext, ExtHostNotebookShape, ExtHostContext } from '../common/extHost.protocol'; import { Disposable } from 'vs/base/common/lifecycle'; import { URI, UriComponents } from 'vs/base/common/uri'; import { INotebookService, IMainNotebookController } from 'vs/workbench/contrib/notebook/browser/notebookService'; import { INotebookTextModel, INotebookMimeTypeSelector, NOTEBOOK_DISPLAY_ORDER, NotebookCellsSplice, NotebookCellOutputsSplice, CellKind, NotebookDocumentMetadata } from 'vs/workbench/contrib/notebook/common/notebookCommon'; import { IConfigurationService } from 'vs/platform/configuration/common/configuration'; import { NotebookCellTextModel } from 'vs/workbench/contrib/notebook/common/model/notebookCellTextModel'; import { NotebookTextModel } from 'vs/workbench/contrib/notebook/common/model/notebookTextModel'; import { IEditorService } from 'vs/workbench/services/editor/common/editorService'; import { INotebookEditor } from 'vs/workbench/contrib/notebook/browser/notebookBrowser'; export class MainThreadNotebookDocument extends Disposable { private _textModel: NotebookTextModel; get textModel() { return this._textModel; } constructor( private readonly _proxy: ExtHostNotebookShape, public handle: number, public viewType: string, public uri: URI ) { super(); this._textModel = new NotebookTextModel(handle, viewType, uri); } async deleteCell(uri: URI, index: number): Promise { let deleteExtHostCell = await this._proxy.$deleteCell(this.viewType, uri, index); if (deleteExtHostCell) { this._textModel.removeCell(index); return true; } return false; } dispose() { this._textModel.dispose(); super.dispose(); } } @extHostNamedCustomer(MainContext.MainThreadNotebook) export class MainThreadNotebooks extends Disposable implements MainThreadNotebookShape { private readonly _notebookProviders = new Map(); private readonly _proxy: ExtHostNotebookShape; constructor( extHostContext: IExtHostContext, @INotebookService private _notebookService: INotebookService, @IConfigurationService private readonly configurationService: IConfigurationService, @IEditorService private readonly editorService: IEditorService, ) { super(); this._proxy = extHostContext.getProxy(ExtHostContext.ExtHostNotebook); this.registerListeners(); } registerListeners() { this._register(this._notebookService.onDidChangeActiveEditor(e => { this._proxy.$updateActiveEditor(e.viewType, e.uri); })); let userOrder = this.configurationService.getValue('notebook.displayOrder'); this._proxy.$acceptDisplayOrder({ defaultOrder: NOTEBOOK_DISPLAY_ORDER, userOrder: userOrder }); this.configurationService.onDidChangeConfiguration(e => { if (e.affectedKeys.indexOf('notebook.displayOrder') >= 0) { let userOrder = this.configurationService.getValue('notebook.displayOrder'); this._proxy.$acceptDisplayOrder({ defaultOrder: NOTEBOOK_DISPLAY_ORDER, userOrder: userOrder }); } }); } async $registerNotebookRenderer(extension: NotebookExtensionDescription, type: string, selectors: INotebookMimeTypeSelector, handle: number, preloads: UriComponents[]): Promise { this._notebookService.registerNotebookRenderer(handle, extension, type, selectors, preloads.map(uri => URI.revive(uri))); } async $unregisterNotebookRenderer(handle: number): Promise { this._notebookService.unregisterNotebookRenderer(handle); } async $registerNotebookProvider(extension: NotebookExtensionDescription, viewType: string): Promise { let controller = new MainThreadNotebookController(this._proxy, this, viewType); this._notebookProviders.set(viewType, controller); this._notebookService.registerNotebookController(viewType, extension, controller); return; } async $unregisterNotebookProvider(viewType: string): Promise { this._notebookProviders.delete(viewType); this._notebookService.unregisterNotebookProvider(viewType); return; } async $createNotebookDocument(handle: number, viewType: string, resource: UriComponents): Promise { let controller = this._notebookProviders.get(viewType); if (controller) { controller.createNotebookDocument(handle, viewType, resource); } return; } async $updateNotebookLanguages(viewType: string, resource: UriComponents, languages: string[]): Promise { let controller = this._notebookProviders.get(viewType); if (controller) { controller.updateLanguages(resource, languages); } } async $updateNotebookMetadata(viewType: string, resource: UriComponents, metadata: NotebookDocumentMetadata | undefined): Promise { let controller = this._notebookProviders.get(viewType); if (controller) { controller.updateNotebookMetadata(resource, metadata); } } async resolveNotebook(viewType: string, uri: URI): Promise { let handle = await this._proxy.$resolveNotebook(viewType, uri); return handle; } async $spliceNotebookCells(viewType: string, resource: UriComponents, splices: NotebookCellsSplice[], renderers: number[]): Promise { let controller = this._notebookProviders.get(viewType); controller?.spliceNotebookCells(resource, splices, renderers); } async $spliceNotebookCellOutputs(viewType: string, resource: UriComponents, cellHandle: number, splices: NotebookCellOutputsSplice[], renderers: number[]): Promise { let controller = this._notebookProviders.get(viewType); controller?.spliceNotebookCellOutputs(resource, cellHandle, splices, renderers); } async executeNotebook(viewType: string, uri: URI): Promise { return this._proxy.$executeNotebook(viewType, uri, undefined); } async $postMessage(handle: number, value: any): Promise { const activeEditorPane = this.editorService.activeEditorPane as any | undefined; if (activeEditorPane?.isNotebookEditor) { const notebookEditor = (activeEditorPane as INotebookEditor); if (notebookEditor.viewModel?.handle === handle) { notebookEditor.postMessage(value); return true; } } return false; } } export class MainThreadNotebookController implements IMainNotebookController { private _mapping: Map = new Map(); constructor( private readonly _proxy: ExtHostNotebookShape, private _mainThreadNotebook: MainThreadNotebooks, private _viewType: string ) { } async resolveNotebook(viewType: string, uri: URI): Promise { // TODO: resolve notebook should wait for all notebook document destory operations to finish. let mainthreadNotebook = this._mapping.get(URI.from(uri).toString()); if (mainthreadNotebook) { return mainthreadNotebook.textModel; } let notebookHandle = await this._mainThreadNotebook.resolveNotebook(viewType, uri); if (notebookHandle !== undefined) { mainthreadNotebook = this._mapping.get(URI.from(uri).toString()); return mainthreadNotebook?.textModel; } return undefined; } spliceNotebookCells(resource: UriComponents, splices: NotebookCellsSplice[], renderers: number[]): void { let mainthreadNotebook = this._mapping.get(URI.from(resource).toString()); mainthreadNotebook?.textModel.updateRenderers(renderers); mainthreadNotebook?.textModel.$spliceNotebookCells(splices); } spliceNotebookCellOutputs(resource: UriComponents, cellHandle: number, splices: NotebookCellOutputsSplice[], renderers: number[]): void { let mainthreadNotebook = this._mapping.get(URI.from(resource).toString()); mainthreadNotebook?.textModel.updateRenderers(renderers); mainthreadNotebook?.textModel.$spliceNotebookCellOutputs(cellHandle, splices); } async executeNotebook(viewType: string, uri: URI): Promise { this._mainThreadNotebook.executeNotebook(viewType, uri); } onDidReceiveMessage(uri: UriComponents, message: any): void { this._proxy.$onDidReceiveMessage(uri, message); } // Methods for ExtHost async createNotebookDocument(handle: number, viewType: string, resource: UriComponents): Promise { let document = new MainThreadNotebookDocument(this._proxy, handle, viewType, URI.revive(resource)); this._mapping.set(URI.revive(resource).toString(), document); } updateLanguages(resource: UriComponents, languages: string[]) { let document = this._mapping.get(URI.from(resource).toString()); document?.textModel.updateLanguages(languages); } updateNotebookMetadata(resource: UriComponents, metadata: NotebookDocumentMetadata | undefined) { let document = this._mapping.get(URI.from(resource).toString()); document?.textModel.updateNotebookMetadata(metadata); } updateNotebookRenderers(resource: UriComponents, renderers: number[]): void { let document = this._mapping.get(URI.from(resource).toString()); document?.textModel.updateRenderers(renderers); } updateNotebookActiveCell(uri: URI, cellHandle: number): void { let mainthreadNotebook = this._mapping.get(URI.from(uri).toString()); mainthreadNotebook?.textModel.updateActiveCell(cellHandle); } async createRawCell(uri: URI, index: number, language: string, type: CellKind): Promise { let cell = await this._proxy.$createEmptyCell(this._viewType, uri, index, language, type); if (cell) { let mainCell = new NotebookCellTextModel(URI.revive(cell.uri), cell.handle, cell.source, cell.language, cell.cellKind, cell.outputs); return mainCell; } return undefined; } async deleteCell(uri: URI, index: number): Promise { let mainthreadNotebook = this._mapping.get(URI.from(uri).toString()); if (mainthreadNotebook) { return mainthreadNotebook.deleteCell(uri, index); } return false; } async executeNotebookActiveCell(uri: URI): Promise { let mainthreadNotebook = this._mapping.get(URI.from(uri).toString()); if (mainthreadNotebook && mainthreadNotebook.textModel.activeCell) { return this._proxy.$executeNotebook(this._viewType, uri, mainthreadNotebook.textModel.activeCell.handle); } } async destoryNotebookDocument(notebook: INotebookTextModel): Promise { let document = this._mapping.get(URI.from(notebook.uri).toString()); if (!document) { return; } let removeFromExtHost = await this._proxy.$destoryNotebookDocument(this._viewType, notebook.uri); if (removeFromExtHost) { document.dispose(); this._mapping.delete(URI.from(notebook.uri).toString()); } } async save(uri: URI): Promise { return this._proxy.$saveNotebook(this._viewType, uri); } }