Path: blob/main/src/vs/workbench/api/common/extHostDocumentContentProviders.ts
3296 views
/*---------------------------------------------------------------------------------------------1* Copyright (c) Microsoft Corporation. All rights reserved.2* Licensed under the MIT License. See License.txt in the project root for license information.3*--------------------------------------------------------------------------------------------*/45import { onUnexpectedError } from '../../../base/common/errors.js';6import { URI, UriComponents } from '../../../base/common/uri.js';7import { IDisposable } from '../../../base/common/lifecycle.js';8import { Disposable } from './extHostTypes.js';9import type * as vscode from 'vscode';10import { MainContext, ExtHostDocumentContentProvidersShape, MainThreadDocumentContentProvidersShape, IMainContext } from './extHost.protocol.js';11import { ExtHostDocumentsAndEditors } from './extHostDocumentsAndEditors.js';12import { Schemas } from '../../../base/common/network.js';13import { ILogService } from '../../../platform/log/common/log.js';14import { CancellationToken } from '../../../base/common/cancellation.js';15import { splitLines } from '../../../base/common/strings.js';1617export class ExtHostDocumentContentProvider implements ExtHostDocumentContentProvidersShape {1819private static _handlePool = 0;2021private readonly _documentContentProviders = new Map<number, vscode.TextDocumentContentProvider>();22private readonly _proxy: MainThreadDocumentContentProvidersShape;2324constructor(25mainContext: IMainContext,26private readonly _documentsAndEditors: ExtHostDocumentsAndEditors,27private readonly _logService: ILogService,28) {29this._proxy = mainContext.getProxy(MainContext.MainThreadDocumentContentProviders);30}3132registerTextDocumentContentProvider(scheme: string, provider: vscode.TextDocumentContentProvider): vscode.Disposable {33// todo@remote34// check with scheme from fs-providers!35if (Object.keys(Schemas).indexOf(scheme) >= 0) {36throw new Error(`scheme '${scheme}' already registered`);37}3839const handle = ExtHostDocumentContentProvider._handlePool++;4041this._documentContentProviders.set(handle, provider);42this._proxy.$registerTextContentProvider(handle, scheme);4344let subscription: IDisposable | undefined;45if (typeof provider.onDidChange === 'function') {4647let lastEvent: Promise<void> | undefined;4849subscription = provider.onDidChange(async uri => {50if (uri.scheme !== scheme) {51this._logService.warn(`Provider for scheme '${scheme}' is firing event for schema '${uri.scheme}' which will be IGNORED`);52return;53}54if (!this._documentsAndEditors.getDocument(uri)) {55// ignore event if document isn't open56return;57}5859if (lastEvent) {60await lastEvent;61}6263const thisEvent = this.$provideTextDocumentContent(handle, uri)64.then(async value => {65if (!value && typeof value !== 'string') {66return;67}6869const document = this._documentsAndEditors.getDocument(uri);70if (!document) {71// disposed in the meantime72return;73}7475// create lines and compare76const lines = splitLines(value);7778// broadcast event when content changed79if (!document.equalLines(lines)) {80return this._proxy.$onVirtualDocumentChange(uri, value);81}82})83.catch(onUnexpectedError)84.finally(() => {85if (lastEvent === thisEvent) {86lastEvent = undefined;87}88});8990lastEvent = thisEvent;91});92}93return new Disposable(() => {94if (this._documentContentProviders.delete(handle)) {95this._proxy.$unregisterTextContentProvider(handle);96}97if (subscription) {98subscription.dispose();99subscription = undefined;100}101});102}103104$provideTextDocumentContent(handle: number, uri: UriComponents): Promise<string | null | undefined> {105const provider = this._documentContentProviders.get(handle);106if (!provider) {107return Promise.reject(new Error(`unsupported uri-scheme: ${uri.scheme}`));108}109return Promise.resolve(provider.provideTextDocumentContent(URI.revive(uri), CancellationToken.None));110}111}112113114