mirror of
https://github.com/ckaczor/azuredatastudio.git
synced 2026-01-20 09:35:38 -05:00
548 lines
22 KiB
TypeScript
548 lines
22 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 { VSBuffer } from 'vs/base/common/buffer';
|
|
import { CancellationToken } from 'vs/base/common/cancellation';
|
|
import { Emitter, Event } from 'vs/base/common/event';
|
|
import { IRelativePattern } from 'vs/base/common/glob';
|
|
import { hash } from 'vs/base/common/hash';
|
|
import { DisposableStore, IDisposable, toDisposable } from 'vs/base/common/lifecycle';
|
|
import { ResourceMap } from 'vs/base/common/map';
|
|
import { isFalsyOrWhitespace } from 'vs/base/common/strings';
|
|
import { assertIsDefined } from 'vs/base/common/types';
|
|
import { URI, UriComponents } from 'vs/base/common/uri';
|
|
import { IExtensionDescription } from 'vs/platform/extensions/common/extensions';
|
|
import { Cache } from 'vs/workbench/api/common/cache';
|
|
import { ExtHostNotebookShape, IMainContext, IModelAddedData, INotebookCellStatusBarListDto, INotebookDocumentsAndEditorsDelta, INotebookDocumentShowOptions, INotebookEditorAddData, MainContext, MainThreadNotebookDocumentsShape, MainThreadNotebookEditorsShape, MainThreadNotebookShape } from 'vs/workbench/api/common/extHost.protocol';
|
|
import { CommandsConverter, ExtHostCommands } from 'vs/workbench/api/common/extHostCommands';
|
|
import { ExtHostDocuments } from 'vs/workbench/api/common/extHostDocuments';
|
|
import { ExtHostDocumentsAndEditors } from 'vs/workbench/api/common/extHostDocumentsAndEditors';
|
|
import { IExtensionStoragePaths } from 'vs/workbench/api/common/extHostStoragePaths';
|
|
import * as typeConverters from 'vs/workbench/api/common/extHostTypeConverters';
|
|
import * as extHostTypes from 'vs/workbench/api/common/extHostTypes';
|
|
import { INotebookExclusiveDocumentFilter, INotebookContributionData, NotebookCellsChangeType, NotebookDataDto } from 'vs/workbench/contrib/notebook/common/notebookCommon';
|
|
import type * as vscode from 'vscode';
|
|
import { ExtHostCell, ExtHostNotebookDocument } from './extHostNotebookDocument';
|
|
import { ExtHostNotebookEditor } from './extHostNotebookEditor';
|
|
|
|
|
|
type NotebookContentProviderData = {
|
|
readonly provider: vscode.NotebookContentProvider;
|
|
readonly extension: IExtensionDescription;
|
|
};
|
|
|
|
export class ExtHostNotebookController implements ExtHostNotebookShape {
|
|
private static _notebookStatusBarItemProviderHandlePool: number = 0;
|
|
|
|
private readonly _notebookProxy: MainThreadNotebookShape;
|
|
private readonly _notebookDocumentsProxy: MainThreadNotebookDocumentsShape;
|
|
private readonly _notebookEditorsProxy: MainThreadNotebookEditorsShape;
|
|
|
|
private readonly _notebookContentProviders = new Map<string, NotebookContentProviderData>();
|
|
private readonly _notebookStatusBarItemProviders = new Map<number, vscode.NotebookCellStatusBarItemProvider>();
|
|
private readonly _documents = new ResourceMap<ExtHostNotebookDocument>();
|
|
private readonly _editors = new Map<string, ExtHostNotebookEditor>();
|
|
private readonly _commandsConverter: CommandsConverter;
|
|
|
|
private readonly _onDidChangeNotebookCells = new Emitter<vscode.NotebookCellsChangeEvent>();
|
|
readonly onDidChangeNotebookCells = this._onDidChangeNotebookCells.event;
|
|
private readonly _onDidChangeCellOutputs = new Emitter<vscode.NotebookCellOutputsChangeEvent>();
|
|
readonly onDidChangeCellOutputs = this._onDidChangeCellOutputs.event;
|
|
private readonly _onDidChangeCellMetadata = new Emitter<vscode.NotebookCellMetadataChangeEvent>();
|
|
readonly onDidChangeCellMetadata = this._onDidChangeCellMetadata.event;
|
|
private readonly _onDidChangeActiveNotebookEditor = new Emitter<vscode.NotebookEditor | undefined>();
|
|
readonly onDidChangeActiveNotebookEditor = this._onDidChangeActiveNotebookEditor.event;
|
|
private readonly _onDidChangeCellExecutionState = new Emitter<vscode.NotebookCellExecutionStateChangeEvent>();
|
|
readonly onDidChangeNotebookCellExecutionState = this._onDidChangeCellExecutionState.event;
|
|
|
|
private _activeNotebookEditor: ExtHostNotebookEditor | undefined;
|
|
get activeNotebookEditor(): vscode.NotebookEditor | undefined {
|
|
return this._activeNotebookEditor?.apiEditor;
|
|
}
|
|
private _visibleNotebookEditors: ExtHostNotebookEditor[] = [];
|
|
get visibleNotebookEditors(): vscode.NotebookEditor[] {
|
|
return this._visibleNotebookEditors.map(editor => editor.apiEditor);
|
|
}
|
|
|
|
private _onDidOpenNotebookDocument = new Emitter<vscode.NotebookDocument>();
|
|
onDidOpenNotebookDocument: Event<vscode.NotebookDocument> = this._onDidOpenNotebookDocument.event;
|
|
private _onDidCloseNotebookDocument = new Emitter<vscode.NotebookDocument>();
|
|
onDidCloseNotebookDocument: Event<vscode.NotebookDocument> = this._onDidCloseNotebookDocument.event;
|
|
|
|
private _onDidChangeVisibleNotebookEditors = new Emitter<vscode.NotebookEditor[]>();
|
|
onDidChangeVisibleNotebookEditors = this._onDidChangeVisibleNotebookEditors.event;
|
|
|
|
private _statusBarCache = new Cache<IDisposable>('NotebookCellStatusBarCache');
|
|
|
|
constructor(
|
|
mainContext: IMainContext,
|
|
commands: ExtHostCommands,
|
|
private _textDocumentsAndEditors: ExtHostDocumentsAndEditors,
|
|
private _textDocuments: ExtHostDocuments,
|
|
private readonly _extensionStoragePaths: IExtensionStoragePaths,
|
|
) {
|
|
this._notebookProxy = mainContext.getProxy(MainContext.MainThreadNotebook);
|
|
this._notebookDocumentsProxy = mainContext.getProxy(MainContext.MainThreadNotebookDocuments);
|
|
this._notebookEditorsProxy = mainContext.getProxy(MainContext.MainThreadNotebookEditors);
|
|
this._commandsConverter = commands.converter;
|
|
|
|
commands.registerArgumentProcessor({
|
|
// Serialized INotebookCellActionContext
|
|
processArgument: (arg) => {
|
|
if (arg && arg.$mid === 12) {
|
|
const notebookUri = arg.notebookEditor?.notebookUri;
|
|
const cellHandle = arg.cell.handle;
|
|
|
|
const data = this._documents.get(notebookUri);
|
|
const cell = data?.getCell(cellHandle);
|
|
if (cell) {
|
|
return cell.apiCell;
|
|
}
|
|
}
|
|
return arg;
|
|
}
|
|
});
|
|
}
|
|
|
|
getEditorById(editorId: string): ExtHostNotebookEditor {
|
|
const editor = this._editors.get(editorId);
|
|
if (!editor) {
|
|
throw new Error(`unknown text editor: ${editorId}. known editors: ${[...this._editors.keys()]} `);
|
|
}
|
|
return editor;
|
|
}
|
|
|
|
getIdByEditor(editor: vscode.NotebookEditor): string | undefined {
|
|
for (const [id, candidate] of this._editors) {
|
|
if (candidate.apiEditor === editor) {
|
|
return id;
|
|
}
|
|
}
|
|
return undefined;
|
|
}
|
|
|
|
get notebookDocuments() {
|
|
return [...this._documents.values()];
|
|
}
|
|
|
|
getNotebookDocument(uri: URI, relaxed: true): ExtHostNotebookDocument | undefined;
|
|
getNotebookDocument(uri: URI): ExtHostNotebookDocument;
|
|
getNotebookDocument(uri: URI, relaxed?: true): ExtHostNotebookDocument | undefined {
|
|
const result = this._documents.get(uri);
|
|
if (!result && !relaxed) {
|
|
throw new Error(`NO notebook document for '${uri}'`);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
private _getProviderData(viewType: string): NotebookContentProviderData {
|
|
const result = this._notebookContentProviders.get(viewType);
|
|
if (!result) {
|
|
throw new Error(`NO provider for '${viewType}'`);
|
|
}
|
|
return result;
|
|
}
|
|
|
|
registerNotebookContentProvider(
|
|
extension: IExtensionDescription,
|
|
viewType: string,
|
|
provider: vscode.NotebookContentProvider,
|
|
options?: vscode.NotebookDocumentContentOptions,
|
|
registration?: vscode.NotebookRegistrationData
|
|
): vscode.Disposable {
|
|
if (isFalsyOrWhitespace(viewType)) {
|
|
throw new Error(`viewType cannot be empty or just whitespace`);
|
|
}
|
|
if (this._notebookContentProviders.has(viewType)) {
|
|
throw new Error(`Notebook provider for '${viewType}' already registered`);
|
|
}
|
|
|
|
this._notebookContentProviders.set(viewType, { extension, provider });
|
|
|
|
let listener: IDisposable | undefined;
|
|
if (provider.onDidChangeNotebookContentOptions) {
|
|
listener = provider.onDidChangeNotebookContentOptions(() => {
|
|
const internalOptions = typeConverters.NotebookDocumentContentOptions.from(provider.options);
|
|
this._notebookProxy.$updateNotebookProviderOptions(viewType, internalOptions);
|
|
});
|
|
}
|
|
|
|
this._notebookProxy.$registerNotebookProvider(
|
|
{ id: extension.identifier, location: extension.extensionLocation },
|
|
viewType,
|
|
typeConverters.NotebookDocumentContentOptions.from(options),
|
|
ExtHostNotebookController._convertNotebookRegistrationData(extension, registration)
|
|
);
|
|
|
|
return new extHostTypes.Disposable(() => {
|
|
listener?.dispose();
|
|
this._notebookContentProviders.delete(viewType);
|
|
this._notebookProxy.$unregisterNotebookProvider(viewType);
|
|
});
|
|
}
|
|
|
|
private static _convertNotebookRegistrationData(extension: IExtensionDescription, registration: vscode.NotebookRegistrationData | undefined): INotebookContributionData | undefined {
|
|
if (!registration) {
|
|
return undefined; // {{SQL CARBON EDIT}} Strict nulls
|
|
}
|
|
const viewOptionsFilenamePattern = registration.filenamePattern
|
|
.map(pattern => typeConverters.NotebookExclusiveDocumentPattern.from(pattern))
|
|
.filter(pattern => pattern !== undefined) as (string | IRelativePattern | INotebookExclusiveDocumentFilter)[];
|
|
if (registration.filenamePattern && !viewOptionsFilenamePattern) {
|
|
console.warn(`Notebook content provider view options file name pattern is invalid ${registration.filenamePattern}`);
|
|
return undefined;
|
|
}
|
|
return {
|
|
extension: extension.identifier,
|
|
providerDisplayName: extension.displayName || extension.name,
|
|
displayName: registration.displayName,
|
|
filenamePattern: viewOptionsFilenamePattern,
|
|
exclusive: registration.exclusive || false
|
|
};
|
|
}
|
|
|
|
registerNotebookCellStatusBarItemProvider(extension: IExtensionDescription, notebookType: string, provider: vscode.NotebookCellStatusBarItemProvider) {
|
|
|
|
const handle = ExtHostNotebookController._notebookStatusBarItemProviderHandlePool++;
|
|
const eventHandle = typeof provider.onDidChangeCellStatusBarItems === 'function' ? ExtHostNotebookController._notebookStatusBarItemProviderHandlePool++ : undefined;
|
|
|
|
this._notebookStatusBarItemProviders.set(handle, provider);
|
|
this._notebookProxy.$registerNotebookCellStatusBarItemProvider(handle, eventHandle, notebookType);
|
|
|
|
let subscription: vscode.Disposable | undefined;
|
|
if (eventHandle !== undefined) {
|
|
subscription = provider.onDidChangeCellStatusBarItems!(_ => this._notebookProxy.$emitCellStatusBarEvent(eventHandle));
|
|
}
|
|
|
|
return new extHostTypes.Disposable(() => {
|
|
this._notebookStatusBarItemProviders.delete(handle);
|
|
this._notebookProxy.$unregisterNotebookCellStatusBarItemProvider(handle, eventHandle);
|
|
if (subscription) {
|
|
subscription.dispose();
|
|
}
|
|
});
|
|
}
|
|
|
|
async createNotebookDocument(options: { viewType: string, content?: vscode.NotebookData }): Promise<URI> {
|
|
const canonicalUri = await this._notebookDocumentsProxy.$tryCreateNotebook({
|
|
viewType: options.viewType,
|
|
content: options.content && typeConverters.NotebookData.from(options.content)
|
|
});
|
|
return URI.revive(canonicalUri);
|
|
}
|
|
|
|
async openNotebookDocument(uri: URI): Promise<vscode.NotebookDocument> {
|
|
const cached = this._documents.get(uri);
|
|
if (cached) {
|
|
return cached.apiNotebook;
|
|
}
|
|
const canonicalUri = await this._notebookDocumentsProxy.$tryOpenNotebook(uri);
|
|
const document = this._documents.get(URI.revive(canonicalUri));
|
|
return assertIsDefined(document?.apiNotebook);
|
|
}
|
|
|
|
|
|
async showNotebookDocument(notebookOrUri: vscode.NotebookDocument | URI, options?: vscode.NotebookDocumentShowOptions): Promise<vscode.NotebookEditor> {
|
|
|
|
if (URI.isUri(notebookOrUri)) {
|
|
notebookOrUri = await this.openNotebookDocument(notebookOrUri);
|
|
}
|
|
|
|
let resolvedOptions: INotebookDocumentShowOptions;
|
|
if (typeof options === 'object') {
|
|
resolvedOptions = {
|
|
position: typeConverters.ViewColumn.from(options.viewColumn),
|
|
preserveFocus: options.preserveFocus,
|
|
selections: options.selections && options.selections.map(typeConverters.NotebookRange.from),
|
|
pinned: typeof options.preview === 'boolean' ? !options.preview : undefined
|
|
};
|
|
} else {
|
|
resolvedOptions = {
|
|
preserveFocus: false
|
|
};
|
|
}
|
|
|
|
const editorId = await this._notebookEditorsProxy.$tryShowNotebookDocument(notebookOrUri.uri, notebookOrUri.notebookType, resolvedOptions);
|
|
const editor = editorId && this._editors.get(editorId)?.apiEditor;
|
|
|
|
if (editor) {
|
|
return editor;
|
|
}
|
|
|
|
if (editorId) {
|
|
throw new Error(`Could NOT open editor for "${notebookOrUri.uri.toString()}" because another editor opened in the meantime.`);
|
|
} else {
|
|
throw new Error(`Could NOT open editor for "${notebookOrUri.uri.toString()}".`);
|
|
}
|
|
}
|
|
|
|
async $provideNotebookCellStatusBarItems(handle: number, uri: UriComponents, index: number, token: CancellationToken): Promise<INotebookCellStatusBarListDto | undefined> {
|
|
const provider = this._notebookStatusBarItemProviders.get(handle);
|
|
const revivedUri = URI.revive(uri);
|
|
const document = this._documents.get(revivedUri);
|
|
if (!document || !provider) {
|
|
return undefined; // {{SQL CARBON EDIT}} Strict nulls
|
|
}
|
|
|
|
const cell = document.getCellFromIndex(index);
|
|
if (!cell) {
|
|
return undefined; // {{SQL CARBON EDIT}} Strict nulls
|
|
}
|
|
|
|
const result = await provider.provideCellStatusBarItems(cell.apiCell, token);
|
|
if (!result) {
|
|
return undefined;
|
|
}
|
|
|
|
const disposables = new DisposableStore();
|
|
const cacheId = this._statusBarCache.add([disposables]);
|
|
const resultArr = Array.isArray(result) ? result : [result];
|
|
const items = resultArr.map(item => typeConverters.NotebookStatusBarItem.from(item, this._commandsConverter, disposables));
|
|
return {
|
|
cacheId,
|
|
items
|
|
};
|
|
}
|
|
|
|
$releaseNotebookCellStatusBarItems(cacheId: number): void {
|
|
this._statusBarCache.delete(cacheId);
|
|
}
|
|
|
|
// --- serialize/deserialize
|
|
|
|
private _handlePool = 0;
|
|
private readonly _notebookSerializer = new Map<number, vscode.NotebookSerializer>();
|
|
|
|
registerNotebookSerializer(extension: IExtensionDescription, viewType: string, serializer: vscode.NotebookSerializer, options?: vscode.NotebookDocumentContentOptions, registration?: vscode.NotebookRegistrationData): vscode.Disposable {
|
|
if (isFalsyOrWhitespace(viewType)) {
|
|
throw new Error(`viewType cannot be empty or just whitespace`);
|
|
}
|
|
const handle = this._handlePool++;
|
|
this._notebookSerializer.set(handle, serializer);
|
|
this._notebookProxy.$registerNotebookSerializer(
|
|
handle,
|
|
{ id: extension.identifier, location: extension.extensionLocation },
|
|
viewType,
|
|
typeConverters.NotebookDocumentContentOptions.from(options),
|
|
ExtHostNotebookController._convertNotebookRegistrationData(extension, registration)
|
|
);
|
|
return toDisposable(() => {
|
|
this._notebookProxy.$unregisterNotebookSerializer(handle);
|
|
});
|
|
}
|
|
|
|
async $dataToNotebook(handle: number, bytes: VSBuffer, token: CancellationToken): Promise<NotebookDataDto> {
|
|
const serializer = this._notebookSerializer.get(handle);
|
|
if (!serializer) {
|
|
throw new Error('NO serializer found');
|
|
}
|
|
const data = await serializer.deserializeNotebook(bytes.buffer, token);
|
|
return typeConverters.NotebookData.from(data);
|
|
}
|
|
|
|
async $notebookToData(handle: number, data: NotebookDataDto, token: CancellationToken): Promise<VSBuffer> {
|
|
const serializer = this._notebookSerializer.get(handle);
|
|
if (!serializer) {
|
|
throw new Error('NO serializer found');
|
|
}
|
|
const bytes = await serializer.serializeNotebook(typeConverters.NotebookData.to(data), token);
|
|
return VSBuffer.wrap(bytes);
|
|
}
|
|
|
|
// --- open, save, saveAs, backup
|
|
|
|
async $openNotebook(viewType: string, uri: UriComponents, backupId: string | undefined, untitledDocumentData: VSBuffer | undefined, token: CancellationToken): Promise<NotebookDataDto> {
|
|
const { provider } = this._getProviderData(viewType);
|
|
const data = await provider.openNotebook(URI.revive(uri), { backupId, untitledDocumentData: untitledDocumentData?.buffer }, token);
|
|
return {
|
|
metadata: data.metadata ?? Object.create(null),
|
|
cells: data.cells.map(typeConverters.NotebookCellData.from),
|
|
};
|
|
}
|
|
|
|
async $saveNotebook(viewType: string, uri: UriComponents, token: CancellationToken): Promise<boolean> {
|
|
const document = this.getNotebookDocument(URI.revive(uri));
|
|
const { provider } = this._getProviderData(viewType);
|
|
await provider.saveNotebook(document.apiNotebook, token);
|
|
return true;
|
|
}
|
|
|
|
async $saveNotebookAs(viewType: string, uri: UriComponents, target: UriComponents, token: CancellationToken): Promise<boolean> {
|
|
const document = this.getNotebookDocument(URI.revive(uri));
|
|
const { provider } = this._getProviderData(viewType);
|
|
await provider.saveNotebookAs(URI.revive(target), document.apiNotebook, token);
|
|
return true;
|
|
}
|
|
|
|
private _backupIdPool: number = 0;
|
|
|
|
async $backupNotebook(viewType: string, uri: UriComponents, cancellation: CancellationToken): Promise<string> {
|
|
const document = this.getNotebookDocument(URI.revive(uri));
|
|
const provider = this._getProviderData(viewType);
|
|
|
|
const storagePath = this._extensionStoragePaths.workspaceValue(provider.extension) ?? this._extensionStoragePaths.globalValue(provider.extension);
|
|
const fileName = String(hash([document.uri.toString(), this._backupIdPool++]));
|
|
const backupUri = URI.joinPath(storagePath, fileName);
|
|
|
|
const backup = await provider.provider.backupNotebook(document.apiNotebook, { destination: backupUri }, cancellation);
|
|
document.updateBackup(backup);
|
|
return backup.id;
|
|
}
|
|
|
|
|
|
private _createExtHostEditor(document: ExtHostNotebookDocument, editorId: string, data: INotebookEditorAddData) {
|
|
|
|
if (this._editors.has(editorId)) {
|
|
throw new Error(`editor with id ALREADY EXSIST: ${editorId}`);
|
|
}
|
|
|
|
const editor = new ExtHostNotebookEditor(
|
|
editorId,
|
|
this._notebookEditorsProxy,
|
|
document,
|
|
data.visibleRanges.map(typeConverters.NotebookRange.to),
|
|
data.selections.map(typeConverters.NotebookRange.to),
|
|
typeof data.viewColumn === 'number' ? typeConverters.ViewColumn.to(data.viewColumn) : undefined
|
|
);
|
|
|
|
this._editors.set(editorId, editor);
|
|
}
|
|
|
|
$acceptDocumentAndEditorsDelta(delta: INotebookDocumentsAndEditorsDelta): void {
|
|
|
|
if (delta.removedDocuments) {
|
|
for (const uri of delta.removedDocuments) {
|
|
const revivedUri = URI.revive(uri);
|
|
const document = this._documents.get(revivedUri);
|
|
|
|
if (document) {
|
|
document.dispose();
|
|
this._documents.delete(revivedUri);
|
|
this._textDocumentsAndEditors.$acceptDocumentsAndEditorsDelta({ removedDocuments: document.apiNotebook.getCells().map(cell => cell.document.uri) });
|
|
this._onDidCloseNotebookDocument.fire(document.apiNotebook);
|
|
}
|
|
|
|
for (const editor of this._editors.values()) {
|
|
if (editor.notebookData.uri.toString() === revivedUri.toString()) {
|
|
this._editors.delete(editor.id);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if (delta.addedDocuments) {
|
|
|
|
const addedCellDocuments: IModelAddedData[] = [];
|
|
|
|
for (const modelData of delta.addedDocuments) {
|
|
const uri = URI.revive(modelData.uri);
|
|
const viewType = modelData.viewType;
|
|
|
|
if (this._documents.has(uri)) {
|
|
throw new Error(`adding EXISTING notebook ${uri} `);
|
|
}
|
|
const that = this;
|
|
|
|
const document = new ExtHostNotebookDocument(
|
|
this._notebookDocumentsProxy,
|
|
this._textDocumentsAndEditors,
|
|
this._textDocuments,
|
|
{
|
|
emitModelChange(event: vscode.NotebookCellsChangeEvent): void {
|
|
that._onDidChangeNotebookCells.fire(event);
|
|
},
|
|
emitCellOutputsChange(event: vscode.NotebookCellOutputsChangeEvent): void {
|
|
that._onDidChangeCellOutputs.fire(event);
|
|
},
|
|
emitCellMetadataChange(event: vscode.NotebookCellMetadataChangeEvent): void {
|
|
that._onDidChangeCellMetadata.fire(event);
|
|
},
|
|
emitCellExecutionStateChange(event: vscode.NotebookCellExecutionStateChangeEvent): void {
|
|
that._onDidChangeCellExecutionState.fire(event);
|
|
}
|
|
},
|
|
viewType,
|
|
modelData.metadata ?? Object.create({}),
|
|
uri,
|
|
);
|
|
|
|
document.acceptModelChanged({
|
|
versionId: modelData.versionId,
|
|
rawEvents: [{
|
|
kind: NotebookCellsChangeType.Initialize,
|
|
changes: [[0, 0, modelData.cells]]
|
|
}]
|
|
}, false);
|
|
|
|
// add cell document as vscode.TextDocument
|
|
addedCellDocuments.push(...modelData.cells.map(cell => ExtHostCell.asModelAddData(document.apiNotebook, cell)));
|
|
|
|
this._documents.get(uri)?.dispose();
|
|
this._documents.set(uri, document);
|
|
this._textDocumentsAndEditors.$acceptDocumentsAndEditorsDelta({ addedDocuments: addedCellDocuments });
|
|
|
|
this._onDidOpenNotebookDocument.fire(document.apiNotebook);
|
|
}
|
|
}
|
|
|
|
if (delta.addedEditors) {
|
|
for (const editorModelData of delta.addedEditors) {
|
|
if (this._editors.has(editorModelData.id)) {
|
|
return;
|
|
}
|
|
|
|
const revivedUri = URI.revive(editorModelData.documentUri);
|
|
const document = this._documents.get(revivedUri);
|
|
|
|
if (document) {
|
|
this._createExtHostEditor(document, editorModelData.id, editorModelData);
|
|
}
|
|
}
|
|
}
|
|
|
|
const removedEditors: ExtHostNotebookEditor[] = [];
|
|
|
|
if (delta.removedEditors) {
|
|
for (const editorid of delta.removedEditors) {
|
|
const editor = this._editors.get(editorid);
|
|
|
|
if (editor) {
|
|
this._editors.delete(editorid);
|
|
|
|
if (this._activeNotebookEditor?.id === editor.id) {
|
|
this._activeNotebookEditor = undefined;
|
|
}
|
|
|
|
removedEditors.push(editor);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (delta.visibleEditors) {
|
|
this._visibleNotebookEditors = delta.visibleEditors.map(id => this._editors.get(id)!).filter(editor => !!editor) as ExtHostNotebookEditor[];
|
|
const visibleEditorsSet = new Set<string>();
|
|
this._visibleNotebookEditors.forEach(editor => visibleEditorsSet.add(editor.id));
|
|
|
|
for (const editor of this._editors.values()) {
|
|
const newValue = visibleEditorsSet.has(editor.id);
|
|
editor._acceptVisibility(newValue);
|
|
}
|
|
|
|
this._visibleNotebookEditors = [...this._editors.values()].map(e => e).filter(e => e.visible);
|
|
this._onDidChangeVisibleNotebookEditors.fire(this.visibleNotebookEditors);
|
|
}
|
|
|
|
if (delta.newActiveEditor === null) {
|
|
// clear active notebook as current active editor is non-notebook editor
|
|
this._activeNotebookEditor = undefined;
|
|
} else if (delta.newActiveEditor) {
|
|
this._activeNotebookEditor = this._editors.get(delta.newActiveEditor);
|
|
}
|
|
if (delta.newActiveEditor !== undefined) {
|
|
this._onDidChangeActiveNotebookEditor.fire(this._activeNotebookEditor?.apiEditor);
|
|
}
|
|
}
|
|
}
|