Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
microsoft
GitHub Repository: microsoft/vscode
Path: blob/main/src/vs/workbench/api/browser/mainThreadDocumentContentProviders.ts
3296 views
1
/*---------------------------------------------------------------------------------------------
2
* Copyright (c) Microsoft Corporation. All rights reserved.
3
* Licensed under the MIT License. See License.txt in the project root for license information.
4
*--------------------------------------------------------------------------------------------*/
5
6
import { onUnexpectedError } from '../../../base/common/errors.js';
7
import { dispose, DisposableMap } from '../../../base/common/lifecycle.js';
8
import { URI, UriComponents } from '../../../base/common/uri.js';
9
import { EditOperation } from '../../../editor/common/core/editOperation.js';
10
import { Range } from '../../../editor/common/core/range.js';
11
import { ITextModel } from '../../../editor/common/model.js';
12
import { IEditorWorkerService } from '../../../editor/common/services/editorWorker.js';
13
import { IModelService } from '../../../editor/common/services/model.js';
14
import { ILanguageService } from '../../../editor/common/languages/language.js';
15
import { ITextModelService } from '../../../editor/common/services/resolverService.js';
16
import { extHostNamedCustomer, IExtHostContext } from '../../services/extensions/common/extHostCustomers.js';
17
import { ExtHostContext, ExtHostDocumentContentProvidersShape, MainContext, MainThreadDocumentContentProvidersShape } from '../common/extHost.protocol.js';
18
import { CancellationTokenSource } from '../../../base/common/cancellation.js';
19
20
@extHostNamedCustomer(MainContext.MainThreadDocumentContentProviders)
21
export class MainThreadDocumentContentProviders implements MainThreadDocumentContentProvidersShape {
22
23
private readonly _resourceContentProvider = new DisposableMap<number>();
24
private readonly _pendingUpdate = new Map<string, CancellationTokenSource>();
25
private readonly _proxy: ExtHostDocumentContentProvidersShape;
26
27
constructor(
28
extHostContext: IExtHostContext,
29
@ITextModelService private readonly _textModelResolverService: ITextModelService,
30
@ILanguageService private readonly _languageService: ILanguageService,
31
@IModelService private readonly _modelService: IModelService,
32
@IEditorWorkerService private readonly _editorWorkerService: IEditorWorkerService
33
) {
34
this._proxy = extHostContext.getProxy(ExtHostContext.ExtHostDocumentContentProviders);
35
}
36
37
dispose(): void {
38
this._resourceContentProvider.dispose();
39
dispose(this._pendingUpdate.values());
40
}
41
42
$registerTextContentProvider(handle: number, scheme: string): void {
43
const registration = this._textModelResolverService.registerTextModelContentProvider(scheme, {
44
provideTextContent: (uri: URI): Promise<ITextModel | null> => {
45
return this._proxy.$provideTextDocumentContent(handle, uri).then(value => {
46
if (typeof value === 'string') {
47
const firstLineText = value.substr(0, 1 + value.search(/\r?\n/));
48
const languageSelection = this._languageService.createByFilepathOrFirstLine(uri, firstLineText);
49
return this._modelService.createModel(value, languageSelection, uri);
50
}
51
return null;
52
});
53
}
54
});
55
this._resourceContentProvider.set(handle, registration);
56
}
57
58
$unregisterTextContentProvider(handle: number): void {
59
this._resourceContentProvider.deleteAndDispose(handle);
60
}
61
62
async $onVirtualDocumentChange(uri: UriComponents, value: string): Promise<void> {
63
const model = this._modelService.getModel(URI.revive(uri));
64
if (!model) {
65
return;
66
}
67
68
// cancel and dispose an existing update
69
const pending = this._pendingUpdate.get(model.id);
70
pending?.cancel();
71
72
// create and keep update token
73
const myToken = new CancellationTokenSource();
74
this._pendingUpdate.set(model.id, myToken);
75
76
try {
77
const edits = await this._editorWorkerService.computeMoreMinimalEdits(model.uri, [{ text: value, range: model.getFullModelRange() }]);
78
79
// remove token
80
this._pendingUpdate.delete(model.id);
81
82
if (myToken.token.isCancellationRequested) {
83
// ignore this
84
return;
85
}
86
if (edits && edits.length > 0) {
87
// use the evil-edit as these models show in readonly-editor only
88
model.applyEdits(edits.map(edit => EditOperation.replace(Range.lift(edit.range), edit.text)));
89
}
90
} catch (error) {
91
onUnexpectedError(error);
92
}
93
}
94
}
95
96