@volar/typescript 2.1.6 → 2.2.0-alpha.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/index.d.ts +0 -1
- package/index.js +0 -1
- package/lib/node/decorateLanguageService.d.ts +2 -2
- package/lib/node/decorateLanguageService.js +102 -102
- package/lib/node/decorateLanguageServiceHost.d.ts +2 -2
- package/lib/node/decorateLanguageServiceHost.js +14 -15
- package/lib/node/decorateProgram.d.ts +2 -2
- package/lib/node/decorateProgram.js +6 -6
- package/lib/node/proxyCreateProgram.js +14 -13
- package/lib/node/transform.d.ts +10 -10
- package/lib/node/transform.js +30 -30
- package/lib/node/utils.d.ts +2 -2
- package/lib/node/utils.js +10 -11
- package/lib/protocol/createProject.d.ts +2 -7
- package/lib/protocol/createProject.js +66 -70
- package/lib/quickstart/createAsyncLanguageServicePlugin.js +5 -5
- package/lib/quickstart/createLanguageServicePlugin.js +5 -5
- package/lib/resolveModuleName.d.ts +2 -2
- package/lib/resolveModuleName.js +6 -6
- package/package.json +4 -4
|
@@ -3,7 +3,7 @@ Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
3
3
|
exports.decorateProgram = void 0;
|
|
4
4
|
const utils_1 = require("./utils");
|
|
5
5
|
const transform_1 = require("./transform");
|
|
6
|
-
function decorateProgram(
|
|
6
|
+
function decorateProgram(language, program) {
|
|
7
7
|
const emit = program.emit;
|
|
8
8
|
// for tsc --noEmit
|
|
9
9
|
const getSyntacticDiagnostics = program.getSyntacticDiagnostics;
|
|
@@ -17,29 +17,29 @@ function decorateProgram(files, program) {
|
|
|
17
17
|
return {
|
|
18
18
|
...result,
|
|
19
19
|
diagnostics: result.diagnostics
|
|
20
|
-
.map(d => (0, transform_1.transformDiagnostic)(
|
|
20
|
+
.map(d => (0, transform_1.transformDiagnostic)(language, d))
|
|
21
21
|
.filter(utils_1.notEmpty),
|
|
22
22
|
};
|
|
23
23
|
};
|
|
24
24
|
program.getSyntacticDiagnostics = (sourceFile, cancellationToken) => {
|
|
25
25
|
return getSyntacticDiagnostics(sourceFile, cancellationToken)
|
|
26
|
-
.map(d => (0, transform_1.transformDiagnostic)(
|
|
26
|
+
.map(d => (0, transform_1.transformDiagnostic)(language, d))
|
|
27
27
|
.filter(utils_1.notEmpty);
|
|
28
28
|
};
|
|
29
29
|
program.getSemanticDiagnostics = (sourceFile, cancellationToken) => {
|
|
30
30
|
return getSemanticDiagnostics(sourceFile, cancellationToken)
|
|
31
|
-
.map(d => (0, transform_1.transformDiagnostic)(
|
|
31
|
+
.map(d => (0, transform_1.transformDiagnostic)(language, d))
|
|
32
32
|
.filter(utils_1.notEmpty);
|
|
33
33
|
};
|
|
34
34
|
program.getGlobalDiagnostics = cancellationToken => {
|
|
35
35
|
return getGlobalDiagnostics(cancellationToken)
|
|
36
|
-
.map(d => (0, transform_1.transformDiagnostic)(
|
|
36
|
+
.map(d => (0, transform_1.transformDiagnostic)(language, d))
|
|
37
37
|
.filter(utils_1.notEmpty);
|
|
38
38
|
};
|
|
39
39
|
// @ts-ignore
|
|
40
40
|
program.getBindAndCheckDiagnostics = (sourceFile, cancellationToken) => {
|
|
41
41
|
return getBindAndCheckDiagnostics(sourceFile, cancellationToken)
|
|
42
|
-
.map(d => (0, transform_1.transformDiagnostic)(
|
|
42
|
+
.map(d => (0, transform_1.transformDiagnostic)(language, d))
|
|
43
43
|
.filter(utils_1.notEmpty);
|
|
44
44
|
};
|
|
45
45
|
}
|
|
@@ -9,7 +9,7 @@ function proxyCreateProgram(ts, original, extensions, getLanguagePlugins) {
|
|
|
9
9
|
const options = args[0];
|
|
10
10
|
assert(!!options.host, '!!options.host');
|
|
11
11
|
const sourceFileToSnapshotMap = new WeakMap();
|
|
12
|
-
const
|
|
12
|
+
const language = (0, language_core_1.createLanguage)(getLanguagePlugins(ts, options), ts.sys.useCaseSensitiveFileNames, fileName => {
|
|
13
13
|
let snapshot;
|
|
14
14
|
assert(originalSourceFiles.has(fileName), `originalSourceFiles.has(${fileName})`);
|
|
15
15
|
const sourceFile = originalSourceFiles.get(fileName);
|
|
@@ -31,10 +31,10 @@ function proxyCreateProgram(ts, original, extensions, getLanguagePlugins) {
|
|
|
31
31
|
}
|
|
32
32
|
}
|
|
33
33
|
if (snapshot) {
|
|
34
|
-
|
|
34
|
+
language.scripts.set(fileName, (0, language_core_1.resolveCommonLanguageId)(fileName), snapshot);
|
|
35
35
|
}
|
|
36
36
|
else {
|
|
37
|
-
|
|
37
|
+
language.scripts.delete(fileName);
|
|
38
38
|
}
|
|
39
39
|
});
|
|
40
40
|
const originalSourceFiles = new Map();
|
|
@@ -60,16 +60,16 @@ function proxyCreateProgram(ts, original, extensions, getLanguagePlugins) {
|
|
|
60
60
|
if (originalSourceFile && extensions.some(ext => fileName.endsWith(ext))) {
|
|
61
61
|
let sourceFile2 = parsedSourceFiles.get(originalSourceFile);
|
|
62
62
|
if (!sourceFile2) {
|
|
63
|
-
const
|
|
64
|
-
assert(!!
|
|
63
|
+
const sourceScript = language.scripts.get(fileName);
|
|
64
|
+
assert(!!sourceScript, '!!sourceScript');
|
|
65
65
|
let patchedText = originalSourceFile.text.split('\n').map(line => ' '.repeat(line.length)).join('\n');
|
|
66
66
|
let scriptKind = ts.ScriptKind.TS;
|
|
67
|
-
if (
|
|
68
|
-
const { getScript, getExtraScripts } =
|
|
69
|
-
const
|
|
70
|
-
if (
|
|
71
|
-
scriptKind =
|
|
72
|
-
patchedText +=
|
|
67
|
+
if (sourceScript.generated?.languagePlugin.typescript) {
|
|
68
|
+
const { getServiceScript: getScript, getExtraServiceScripts: getExtraScripts } = sourceScript.generated.languagePlugin.typescript;
|
|
69
|
+
const serviceScript = getScript(sourceScript.generated.root);
|
|
70
|
+
if (serviceScript) {
|
|
71
|
+
scriptKind = serviceScript.scriptKind;
|
|
72
|
+
patchedText += serviceScript.code.snapshot.getText(0, serviceScript.code.snapshot.getLength());
|
|
73
73
|
}
|
|
74
74
|
if (getExtraScripts) {
|
|
75
75
|
console.warn('getExtraScripts() is not available in this use case.');
|
|
@@ -95,8 +95,9 @@ function proxyCreateProgram(ts, original, extensions, getLanguagePlugins) {
|
|
|
95
95
|
});
|
|
96
96
|
};
|
|
97
97
|
const program = Reflect.apply(target, thisArg, [options]);
|
|
98
|
-
(0, decorateProgram_1.decorateProgram)(
|
|
99
|
-
|
|
98
|
+
(0, decorateProgram_1.decorateProgram)(language, program);
|
|
99
|
+
// TODO: #128
|
|
100
|
+
program.__volar__ = { files: language };
|
|
100
101
|
return program;
|
|
101
102
|
function resolveModuleName(name, containingFile, options, redirectedReference) {
|
|
102
103
|
const resolved = ts.resolveModuleName(name, containingFile, options, moduleResolutionHost, originalHost.getModuleResolutionCache?.(), redirectedReference);
|
package/lib/node/transform.d.ts
CHANGED
|
@@ -1,14 +1,14 @@
|
|
|
1
|
-
import {
|
|
1
|
+
import { Language, CodeInformation, SourceMap, SourceScript } from '@volar/language-core';
|
|
2
2
|
import type * as ts from 'typescript';
|
|
3
|
-
export declare function transformCallHierarchyItem(
|
|
4
|
-
export declare function transformDiagnostic<T extends ts.Diagnostic>(
|
|
5
|
-
export declare function transformFileTextChanges(
|
|
6
|
-
export declare function transformDocumentSpan<T extends ts.DocumentSpan>(
|
|
7
|
-
export declare function transformSpan(
|
|
3
|
+
export declare function transformCallHierarchyItem(language: Language, item: ts.CallHierarchyItem, filter: (data: CodeInformation) => boolean): ts.CallHierarchyItem;
|
|
4
|
+
export declare function transformDiagnostic<T extends ts.Diagnostic>(language: Language, diagnostic: T): T | undefined;
|
|
5
|
+
export declare function transformFileTextChanges(language: Language, changes: ts.FileTextChanges, filter: (data: CodeInformation) => boolean): ts.FileTextChanges | undefined;
|
|
6
|
+
export declare function transformDocumentSpan<T extends ts.DocumentSpan>(language: Language, documentSpan: T, filter: (data: CodeInformation) => boolean, shouldFallback?: boolean): T | undefined;
|
|
7
|
+
export declare function transformSpan(language: Language, fileName: string | undefined, textSpan: ts.TextSpan | undefined, filter: (data: CodeInformation) => boolean): {
|
|
8
8
|
fileName: string;
|
|
9
9
|
textSpan: ts.TextSpan;
|
|
10
10
|
} | undefined;
|
|
11
|
-
export declare function transformTextChange(
|
|
12
|
-
export declare function transformTextSpan(
|
|
13
|
-
export declare function toSourceOffset(
|
|
14
|
-
export declare function toGeneratedOffset(
|
|
11
|
+
export declare function transformTextChange(sourceScript: SourceScript, map: SourceMap<CodeInformation>, textChange: ts.TextChange, filter: (data: CodeInformation) => boolean): ts.TextChange | undefined;
|
|
12
|
+
export declare function transformTextSpan(sourceScript: SourceScript, map: SourceMap<CodeInformation>, textSpan: ts.TextSpan, filter: (data: CodeInformation) => boolean): ts.TextSpan | undefined;
|
|
13
|
+
export declare function toSourceOffset(sourceScript: SourceScript, map: SourceMap, position: number, filter: (data: CodeInformation) => boolean): number | undefined;
|
|
14
|
+
export declare function toGeneratedOffset(sourceScript: SourceScript, map: SourceMap, position: number, filter: (data: CodeInformation) => boolean): number | undefined;
|
package/lib/node/transform.js
CHANGED
|
@@ -4,9 +4,9 @@ exports.toGeneratedOffset = exports.toSourceOffset = exports.transformTextSpan =
|
|
|
4
4
|
const language_core_1 = require("@volar/language-core");
|
|
5
5
|
const utils_1 = require("./utils");
|
|
6
6
|
const transformedDiagnostics = new WeakMap();
|
|
7
|
-
function transformCallHierarchyItem(
|
|
8
|
-
const span = transformSpan(
|
|
9
|
-
const selectionSpan = transformSpan(
|
|
7
|
+
function transformCallHierarchyItem(language, item, filter) {
|
|
8
|
+
const span = transformSpan(language, item.file, item.span, filter);
|
|
9
|
+
const selectionSpan = transformSpan(language, item.file, item.selectionSpan, filter);
|
|
10
10
|
return {
|
|
11
11
|
...item,
|
|
12
12
|
span: span?.textSpan ?? { start: 0, length: 0 },
|
|
@@ -14,21 +14,21 @@ function transformCallHierarchyItem(files, item, filter) {
|
|
|
14
14
|
};
|
|
15
15
|
}
|
|
16
16
|
exports.transformCallHierarchyItem = transformCallHierarchyItem;
|
|
17
|
-
function transformDiagnostic(
|
|
17
|
+
function transformDiagnostic(language, diagnostic) {
|
|
18
18
|
if (!transformedDiagnostics.has(diagnostic)) {
|
|
19
19
|
transformedDiagnostics.set(diagnostic, undefined);
|
|
20
20
|
const { relatedInformation } = diagnostic;
|
|
21
21
|
if (relatedInformation) {
|
|
22
22
|
diagnostic.relatedInformation = relatedInformation
|
|
23
|
-
.map(d => transformDiagnostic(
|
|
23
|
+
.map(d => transformDiagnostic(language, d))
|
|
24
24
|
.filter(utils_1.notEmpty);
|
|
25
25
|
}
|
|
26
26
|
if (diagnostic.file !== undefined
|
|
27
27
|
&& diagnostic.start !== undefined
|
|
28
28
|
&& diagnostic.length !== undefined) {
|
|
29
|
-
const [
|
|
30
|
-
if (
|
|
31
|
-
const sourceSpan = transformTextSpan(
|
|
29
|
+
const [serviceScript, sourceScript, map] = (0, utils_1.getServiceScript)(language, diagnostic.file.fileName);
|
|
30
|
+
if (serviceScript) {
|
|
31
|
+
const sourceSpan = transformTextSpan(sourceScript, map, { start: diagnostic.start, length: diagnostic.length }, language_core_1.shouldReportDiagnostics);
|
|
32
32
|
if (sourceSpan) {
|
|
33
33
|
transformedDiagnostics.set(diagnostic, {
|
|
34
34
|
...diagnostic,
|
|
@@ -48,13 +48,13 @@ function transformDiagnostic(files, diagnostic) {
|
|
|
48
48
|
return transformedDiagnostics.get(diagnostic);
|
|
49
49
|
}
|
|
50
50
|
exports.transformDiagnostic = transformDiagnostic;
|
|
51
|
-
function transformFileTextChanges(
|
|
52
|
-
const [_, source] = (0, utils_1.
|
|
51
|
+
function transformFileTextChanges(language, changes, filter) {
|
|
52
|
+
const [_, source] = (0, utils_1.getServiceScript)(language, changes.fileName);
|
|
53
53
|
if (source) {
|
|
54
54
|
return {
|
|
55
55
|
...changes,
|
|
56
56
|
textChanges: changes.textChanges.map(c => {
|
|
57
|
-
const span = transformSpan(
|
|
57
|
+
const span = transformSpan(language, changes.fileName, c.span, filter);
|
|
58
58
|
if (span) {
|
|
59
59
|
return {
|
|
60
60
|
...c,
|
|
@@ -69,11 +69,11 @@ function transformFileTextChanges(files, changes, filter) {
|
|
|
69
69
|
}
|
|
70
70
|
}
|
|
71
71
|
exports.transformFileTextChanges = transformFileTextChanges;
|
|
72
|
-
function transformDocumentSpan(
|
|
73
|
-
let textSpan = transformSpan(
|
|
72
|
+
function transformDocumentSpan(language, documentSpan, filter, shouldFallback) {
|
|
73
|
+
let textSpan = transformSpan(language, documentSpan.fileName, documentSpan.textSpan, filter);
|
|
74
74
|
if (!textSpan && shouldFallback) {
|
|
75
|
-
const [
|
|
76
|
-
if (
|
|
75
|
+
const [serviceScript] = (0, utils_1.getServiceScript)(language, documentSpan.fileName);
|
|
76
|
+
if (serviceScript) {
|
|
77
77
|
textSpan = {
|
|
78
78
|
fileName: documentSpan.fileName,
|
|
79
79
|
textSpan: { start: 0, length: 0 },
|
|
@@ -83,9 +83,9 @@ function transformDocumentSpan(files, documentSpan, filter, shouldFallback) {
|
|
|
83
83
|
if (!textSpan) {
|
|
84
84
|
return;
|
|
85
85
|
}
|
|
86
|
-
const contextSpan = transformSpan(
|
|
87
|
-
const originalTextSpan = transformSpan(
|
|
88
|
-
const originalContextSpan = transformSpan(
|
|
86
|
+
const contextSpan = transformSpan(language, documentSpan.fileName, documentSpan.contextSpan, filter);
|
|
87
|
+
const originalTextSpan = transformSpan(language, documentSpan.originalFileName, documentSpan.originalTextSpan, filter);
|
|
88
|
+
const originalContextSpan = transformSpan(language, documentSpan.originalFileName, documentSpan.originalContextSpan, filter);
|
|
89
89
|
return {
|
|
90
90
|
...documentSpan,
|
|
91
91
|
fileName: textSpan.fileName,
|
|
@@ -97,13 +97,13 @@ function transformDocumentSpan(files, documentSpan, filter, shouldFallback) {
|
|
|
97
97
|
};
|
|
98
98
|
}
|
|
99
99
|
exports.transformDocumentSpan = transformDocumentSpan;
|
|
100
|
-
function transformSpan(
|
|
100
|
+
function transformSpan(language, fileName, textSpan, filter) {
|
|
101
101
|
if (!fileName || !textSpan) {
|
|
102
102
|
return;
|
|
103
103
|
}
|
|
104
|
-
const [virtualFile,
|
|
104
|
+
const [virtualFile, sourceScript, map] = (0, utils_1.getServiceScript)(language, fileName);
|
|
105
105
|
if (virtualFile) {
|
|
106
|
-
const sourceSpan = transformTextSpan(
|
|
106
|
+
const sourceSpan = transformTextSpan(sourceScript, map, textSpan, filter);
|
|
107
107
|
if (sourceSpan) {
|
|
108
108
|
return {
|
|
109
109
|
fileName,
|
|
@@ -119,8 +119,8 @@ function transformSpan(files, fileName, textSpan, filter) {
|
|
|
119
119
|
}
|
|
120
120
|
}
|
|
121
121
|
exports.transformSpan = transformSpan;
|
|
122
|
-
function transformTextChange(
|
|
123
|
-
const sourceSpan = transformTextSpan(
|
|
122
|
+
function transformTextChange(sourceScript, map, textChange, filter) {
|
|
123
|
+
const sourceSpan = transformTextSpan(sourceScript, map, textChange.span, filter);
|
|
124
124
|
if (sourceSpan) {
|
|
125
125
|
return {
|
|
126
126
|
newText: textChange.newText,
|
|
@@ -129,11 +129,11 @@ function transformTextChange(sourceFile, map, textChange, filter) {
|
|
|
129
129
|
}
|
|
130
130
|
}
|
|
131
131
|
exports.transformTextChange = transformTextChange;
|
|
132
|
-
function transformTextSpan(
|
|
132
|
+
function transformTextSpan(sourceScript, map, textSpan, filter) {
|
|
133
133
|
const start = textSpan.start;
|
|
134
134
|
const end = textSpan.start + textSpan.length;
|
|
135
|
-
const sourceStart = toSourceOffset(
|
|
136
|
-
const sourceEnd = toSourceOffset(
|
|
135
|
+
const sourceStart = toSourceOffset(sourceScript, map, start, filter);
|
|
136
|
+
const sourceEnd = toSourceOffset(sourceScript, map, end, filter);
|
|
137
137
|
if (sourceStart !== undefined && sourceEnd !== undefined && sourceEnd >= sourceStart) {
|
|
138
138
|
return {
|
|
139
139
|
start: sourceStart,
|
|
@@ -142,18 +142,18 @@ function transformTextSpan(sourceFile, map, textSpan, filter) {
|
|
|
142
142
|
}
|
|
143
143
|
}
|
|
144
144
|
exports.transformTextSpan = transformTextSpan;
|
|
145
|
-
function toSourceOffset(
|
|
146
|
-
for (const [sourceOffset, mapping] of map.getSourceOffsets(position -
|
|
145
|
+
function toSourceOffset(sourceScript, map, position, filter) {
|
|
146
|
+
for (const [sourceOffset, mapping] of map.getSourceOffsets(position - sourceScript.snapshot.getLength())) {
|
|
147
147
|
if (filter(mapping.data)) {
|
|
148
148
|
return sourceOffset;
|
|
149
149
|
}
|
|
150
150
|
}
|
|
151
151
|
}
|
|
152
152
|
exports.toSourceOffset = toSourceOffset;
|
|
153
|
-
function toGeneratedOffset(
|
|
153
|
+
function toGeneratedOffset(sourceScript, map, position, filter) {
|
|
154
154
|
for (const [generateOffset, mapping] of map.getGeneratedOffsets(position)) {
|
|
155
155
|
if (filter(mapping.data)) {
|
|
156
|
-
return generateOffset +
|
|
156
|
+
return generateOffset + sourceScript.snapshot.getLength();
|
|
157
157
|
}
|
|
158
158
|
}
|
|
159
159
|
}
|
package/lib/node/utils.d.ts
CHANGED
|
@@ -1,3 +1,3 @@
|
|
|
1
|
-
import type {
|
|
1
|
+
import type { Language } from '@volar/language-core';
|
|
2
2
|
export declare function notEmpty<T>(value: T | null | undefined): value is T;
|
|
3
|
-
export declare function
|
|
3
|
+
export declare function getServiceScript(language: Language, fileName: string): readonly [import("@volar/language-core").ServiceScript, import("@volar/language-core").SourceScript, import("@volar/language-core").SourceMap<import("@volar/language-core").CodeInformation>] | readonly [undefined, undefined, undefined];
|
package/lib/node/utils.js
CHANGED
|
@@ -1,23 +1,22 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.
|
|
3
|
+
exports.getServiceScript = exports.notEmpty = void 0;
|
|
4
4
|
function notEmpty(value) {
|
|
5
5
|
return value !== null && value !== undefined;
|
|
6
6
|
}
|
|
7
7
|
exports.notEmpty = notEmpty;
|
|
8
|
-
function
|
|
9
|
-
const
|
|
10
|
-
if (
|
|
11
|
-
const
|
|
12
|
-
if (
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
}
|
|
8
|
+
function getServiceScript(language, fileName) {
|
|
9
|
+
const sourceScript = language.scripts.get(fileName);
|
|
10
|
+
if (sourceScript?.generated) {
|
|
11
|
+
const serviceScript = sourceScript.generated.languagePlugin.typescript?.getServiceScript(sourceScript.generated.root);
|
|
12
|
+
if (serviceScript) {
|
|
13
|
+
const map = language.maps.get(serviceScript.code, sourceScript.id);
|
|
14
|
+
if (map) {
|
|
15
|
+
return [serviceScript, sourceScript, map];
|
|
17
16
|
}
|
|
18
17
|
}
|
|
19
18
|
}
|
|
20
19
|
return [undefined, undefined, undefined];
|
|
21
20
|
}
|
|
22
|
-
exports.
|
|
21
|
+
exports.getServiceScript = getServiceScript;
|
|
23
22
|
//# sourceMappingURL=utils.js.map
|
|
@@ -1,7 +1,2 @@
|
|
|
1
|
-
import { LanguagePlugin,
|
|
2
|
-
|
|
3
|
-
import type { createSys } from './createSys';
|
|
4
|
-
export declare function createLanguage(ts: typeof import('typescript'), sys: ReturnType<typeof createSys> | ts.System, languagePlugins: LanguagePlugin<any>[], configFileName: string | undefined, projectHost: TypeScriptProjectHost, { fileIdToFileName, fileNameToFileId }: {
|
|
5
|
-
fileIdToFileName: (uri: string) => string;
|
|
6
|
-
fileNameToFileId: (fileName: string) => string;
|
|
7
|
-
}): LanguageContext;
|
|
1
|
+
import { LanguagePlugin, Language, TypeScriptProjectHost } from '@volar/language-core';
|
|
2
|
+
export declare function createTypeScriptLanguage(ts: typeof import('typescript'), languagePlugins: LanguagePlugin[], projectHost: TypeScriptProjectHost): Language;
|
|
@@ -1,24 +1,24 @@
|
|
|
1
1
|
"use strict";
|
|
2
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
exports.
|
|
3
|
+
exports.createTypeScriptLanguage = void 0;
|
|
4
4
|
const language_core_1 = require("@volar/language-core");
|
|
5
5
|
const language_core_2 = require("@volar/language-core");
|
|
6
6
|
const path = require("path-browserify");
|
|
7
7
|
const resolveModuleName_1 = require("../resolveModuleName");
|
|
8
8
|
const scriptVersions = new Map();
|
|
9
9
|
const fsFileSnapshots = new Map();
|
|
10
|
-
function
|
|
11
|
-
const
|
|
12
|
-
const fileName =
|
|
10
|
+
function createTypeScriptLanguage(ts, languagePlugins, projectHost) {
|
|
11
|
+
const language = (0, language_core_1.createLanguage)(languagePlugins, projectHost.useCaseSensitiveFileNames, scriptId => {
|
|
12
|
+
const fileName = projectHost.scriptIdToFileName(scriptId);
|
|
13
13
|
// opened files
|
|
14
14
|
let snapshot = projectHost.getScriptSnapshot(fileName);
|
|
15
15
|
if (!snapshot) {
|
|
16
16
|
// fs files
|
|
17
17
|
const cache = fsFileSnapshots.get(fileName);
|
|
18
|
-
const modifiedTime =
|
|
18
|
+
const modifiedTime = projectHost.getModifiedTime?.(fileName)?.valueOf();
|
|
19
19
|
if (!cache || cache[0] !== modifiedTime) {
|
|
20
|
-
if (
|
|
21
|
-
const text =
|
|
20
|
+
if (projectHost.fileExists(fileName)) {
|
|
21
|
+
const text = projectHost.readFile(fileName);
|
|
22
22
|
const snapshot = text !== undefined ? ts.ScriptSnapshot.fromString(text) : undefined;
|
|
23
23
|
fsFileSnapshots.set(fileName, [modifiedTime, snapshot]);
|
|
24
24
|
}
|
|
@@ -29,10 +29,10 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
29
29
|
snapshot = fsFileSnapshots.get(fileName)?.[1];
|
|
30
30
|
}
|
|
31
31
|
if (snapshot) {
|
|
32
|
-
|
|
32
|
+
language.scripts.set(scriptId, projectHost.getLanguageId(scriptId), snapshot);
|
|
33
33
|
}
|
|
34
34
|
else {
|
|
35
|
-
|
|
35
|
+
language.scripts.delete(scriptId);
|
|
36
36
|
}
|
|
37
37
|
});
|
|
38
38
|
let { languageServiceHost, getExtraScript } = createLanguageServiceHost();
|
|
@@ -44,11 +44,11 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
44
44
|
if (languagePlugins.some(language => language.typescript?.extraFileExtensions.length)) {
|
|
45
45
|
// TODO: can this share between monorepo packages?
|
|
46
46
|
const moduleCache = ts.createModuleResolutionCache(languageServiceHost.getCurrentDirectory(), languageServiceHost.useCaseSensitiveFileNames ? s => s : s => s.toLowerCase(), languageServiceHost.getCompilationSettings());
|
|
47
|
-
const resolveModuleName = (0, resolveModuleName_1.createResolveModuleName)(ts, languageServiceHost, languagePlugins, fileName =>
|
|
48
|
-
let lastSysVersion =
|
|
47
|
+
const resolveModuleName = (0, resolveModuleName_1.createResolveModuleName)(ts, languageServiceHost, languagePlugins, fileName => language.scripts.get(projectHost.fileNameToScriptId(fileName)));
|
|
48
|
+
let lastSysVersion = projectHost.getSystemVersion?.();
|
|
49
49
|
languageServiceHost.resolveModuleNameLiterals = (moduleLiterals, containingFile, redirectedReference, options, sourceFile) => {
|
|
50
|
-
if (
|
|
51
|
-
lastSysVersion =
|
|
50
|
+
if (projectHost.getSystemVersion && lastSysVersion !== projectHost.getSystemVersion()) {
|
|
51
|
+
lastSysVersion = projectHost.getSystemVersion();
|
|
52
52
|
moduleCache.clear();
|
|
53
53
|
}
|
|
54
54
|
return moduleLiterals.map(moduleLiteral => {
|
|
@@ -56,8 +56,8 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
56
56
|
});
|
|
57
57
|
};
|
|
58
58
|
languageServiceHost.resolveModuleNames = (moduleNames, containingFile, _reusedNames, redirectedReference, options) => {
|
|
59
|
-
if (
|
|
60
|
-
lastSysVersion =
|
|
59
|
+
if (projectHost.getSystemVersion && lastSysVersion !== projectHost.getSystemVersion()) {
|
|
60
|
+
lastSysVersion = projectHost.getSystemVersion();
|
|
61
61
|
moduleCache.clear();
|
|
62
62
|
}
|
|
63
63
|
return moduleNames.map(moduleName => {
|
|
@@ -65,25 +65,21 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
65
65
|
});
|
|
66
66
|
};
|
|
67
67
|
}
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
sys,
|
|
73
|
-
projectHost,
|
|
74
|
-
languageServiceHost,
|
|
75
|
-
getExtraScript,
|
|
76
|
-
},
|
|
68
|
+
language.typescript = {
|
|
69
|
+
projectHost,
|
|
70
|
+
languageServiceHost,
|
|
71
|
+
getExtraServiceScript: getExtraScript,
|
|
77
72
|
};
|
|
73
|
+
return language;
|
|
78
74
|
function createLanguageServiceHost() {
|
|
79
75
|
let lastProjectVersion;
|
|
80
76
|
let tsProjectVersion = 0;
|
|
81
|
-
let tsFileRegistry = new language_core_1.FileMap(
|
|
82
|
-
let extraScriptRegistry = new language_core_1.FileMap(
|
|
77
|
+
let tsFileRegistry = new language_core_1.FileMap(projectHost.useCaseSensitiveFileNames);
|
|
78
|
+
let extraScriptRegistry = new language_core_1.FileMap(projectHost.useCaseSensitiveFileNames);
|
|
83
79
|
let lastTsVirtualFileSnapshots = new Set();
|
|
84
80
|
let lastOtherVirtualFileSnapshots = new Set();
|
|
85
81
|
const languageServiceHost = {
|
|
86
|
-
...
|
|
82
|
+
...projectHost,
|
|
87
83
|
getCurrentDirectory: projectHost.getCurrentDirectory,
|
|
88
84
|
getCompilationSettings() {
|
|
89
85
|
const options = projectHost.getCompilationSettings();
|
|
@@ -107,16 +103,16 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
107
103
|
}
|
|
108
104
|
},
|
|
109
105
|
useCaseSensitiveFileNames() {
|
|
110
|
-
return
|
|
106
|
+
return projectHost.useCaseSensitiveFileNames;
|
|
111
107
|
},
|
|
112
108
|
getNewLine() {
|
|
113
|
-
return
|
|
109
|
+
return projectHost.newLine;
|
|
114
110
|
},
|
|
115
111
|
getTypeRootsVersion: () => {
|
|
116
|
-
return
|
|
112
|
+
return projectHost.getSystemVersion?.() ?? -1; // TODO: only update for /node_modules changes?
|
|
117
113
|
},
|
|
118
114
|
getDirectories(dirName) {
|
|
119
|
-
return
|
|
115
|
+
return projectHost.getDirectories(dirName);
|
|
120
116
|
},
|
|
121
117
|
readDirectory(dirName, extensions, excludes, includes, depth) {
|
|
122
118
|
const exts = new Set(extensions);
|
|
@@ -126,7 +122,7 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
126
122
|
}
|
|
127
123
|
}
|
|
128
124
|
extensions = [...exts];
|
|
129
|
-
return
|
|
125
|
+
return projectHost.readDirectory(dirName, extensions, excludes, includes, depth);
|
|
130
126
|
},
|
|
131
127
|
readFile(fileName) {
|
|
132
128
|
const snapshot = getScriptSnapshot(fileName);
|
|
@@ -139,7 +135,7 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
139
135
|
},
|
|
140
136
|
getProjectVersion() {
|
|
141
137
|
sync();
|
|
142
|
-
return tsProjectVersion + (
|
|
138
|
+
return tsProjectVersion + (projectHost.getSystemVersion ? `:${projectHost.getSystemVersion()}` : '');
|
|
143
139
|
},
|
|
144
140
|
getScriptFileNames() {
|
|
145
141
|
sync();
|
|
@@ -150,11 +146,11 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
150
146
|
if (extraScriptRegistry.has(fileName)) {
|
|
151
147
|
return extraScriptRegistry.get(fileName).scriptKind;
|
|
152
148
|
}
|
|
153
|
-
const
|
|
154
|
-
if (
|
|
155
|
-
const
|
|
156
|
-
if (
|
|
157
|
-
return
|
|
149
|
+
const sourceScript = language.scripts.get(projectHost.fileNameToScriptId(fileName));
|
|
150
|
+
if (sourceScript?.generated) {
|
|
151
|
+
const serviceScript = sourceScript.generated.languagePlugin.typescript?.getServiceScript(sourceScript.generated.root);
|
|
152
|
+
if (serviceScript) {
|
|
153
|
+
return serviceScript.scriptKind;
|
|
158
154
|
}
|
|
159
155
|
}
|
|
160
156
|
switch (path.extname(fileName)) {
|
|
@@ -199,19 +195,19 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
199
195
|
const newOtherVirtualFileSnapshots = new Set();
|
|
200
196
|
const tsFileNamesSet = new Set();
|
|
201
197
|
for (const fileName of projectHost.getScriptFileNames()) {
|
|
202
|
-
const
|
|
203
|
-
if (
|
|
204
|
-
const
|
|
205
|
-
if (
|
|
206
|
-
newTsVirtualFileSnapshots.add(
|
|
198
|
+
const sourceScript = language.scripts.get(projectHost.fileNameToScriptId(fileName));
|
|
199
|
+
if (sourceScript?.generated) {
|
|
200
|
+
const serviceScript = sourceScript.generated.languagePlugin.typescript?.getServiceScript(sourceScript.generated.root);
|
|
201
|
+
if (serviceScript) {
|
|
202
|
+
newTsVirtualFileSnapshots.add(serviceScript.code.snapshot);
|
|
207
203
|
tsFileNamesSet.add(fileName);
|
|
208
204
|
}
|
|
209
|
-
for (const
|
|
210
|
-
newTsVirtualFileSnapshots.add(
|
|
211
|
-
tsFileNamesSet.add(
|
|
212
|
-
extraScriptRegistry.set(
|
|
205
|
+
for (const extraServiceScript of sourceScript.generated.languagePlugin.typescript?.getExtraServiceScripts?.(fileName, sourceScript.generated.root) ?? []) {
|
|
206
|
+
newTsVirtualFileSnapshots.add(extraServiceScript.code.snapshot);
|
|
207
|
+
tsFileNamesSet.add(extraServiceScript.fileName);
|
|
208
|
+
extraScriptRegistry.set(extraServiceScript.fileName, extraServiceScript);
|
|
213
209
|
}
|
|
214
|
-
for (const code of (0, language_core_2.forEachEmbeddedCode)(
|
|
210
|
+
for (const code of (0, language_core_2.forEachEmbeddedCode)(sourceScript.generated.root)) {
|
|
215
211
|
newOtherVirtualFileSnapshots.add(code.snapshot);
|
|
216
212
|
}
|
|
217
213
|
}
|
|
@@ -238,15 +234,15 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
238
234
|
if (extraScriptRegistry.has(fileName)) {
|
|
239
235
|
return extraScriptRegistry.get(fileName).code.snapshot;
|
|
240
236
|
}
|
|
241
|
-
const
|
|
242
|
-
if (
|
|
243
|
-
const
|
|
244
|
-
if (
|
|
245
|
-
return
|
|
237
|
+
const sourceScript = language.scripts.get(projectHost.fileNameToScriptId(fileName));
|
|
238
|
+
if (sourceScript?.generated) {
|
|
239
|
+
const serviceScript = sourceScript.generated.languagePlugin.typescript?.getServiceScript(sourceScript.generated.root);
|
|
240
|
+
if (serviceScript) {
|
|
241
|
+
return serviceScript.code.snapshot;
|
|
246
242
|
}
|
|
247
243
|
}
|
|
248
|
-
else if (
|
|
249
|
-
return
|
|
244
|
+
else if (sourceScript) {
|
|
245
|
+
return sourceScript.snapshot;
|
|
250
246
|
}
|
|
251
247
|
}
|
|
252
248
|
function getScriptVersion(fileName) {
|
|
@@ -262,34 +258,34 @@ function createLanguage(ts, sys, languagePlugins, configFileName, projectHost, {
|
|
|
262
258
|
}
|
|
263
259
|
return version.map.get(snapshot).toString();
|
|
264
260
|
}
|
|
265
|
-
const
|
|
266
|
-
if (
|
|
267
|
-
const
|
|
268
|
-
if (
|
|
269
|
-
if (!version.map.has(
|
|
270
|
-
version.map.set(
|
|
261
|
+
const sourceScript = language.scripts.get(projectHost.fileNameToScriptId(fileName));
|
|
262
|
+
if (sourceScript?.generated) {
|
|
263
|
+
const serviceScript = sourceScript.generated.languagePlugin.typescript?.getServiceScript(sourceScript.generated.root);
|
|
264
|
+
if (serviceScript) {
|
|
265
|
+
if (!version.map.has(serviceScript.code.snapshot)) {
|
|
266
|
+
version.map.set(serviceScript.code.snapshot, version.lastVersion++);
|
|
271
267
|
}
|
|
272
|
-
return version.map.get(
|
|
268
|
+
return version.map.get(serviceScript.code.snapshot).toString();
|
|
273
269
|
}
|
|
274
270
|
}
|
|
275
271
|
const isOpenedFile = !!projectHost.getScriptSnapshot(fileName);
|
|
276
272
|
if (isOpenedFile) {
|
|
277
|
-
const
|
|
278
|
-
if (
|
|
279
|
-
if (!version.map.has(
|
|
280
|
-
version.map.set(
|
|
273
|
+
const sourceScript = language.scripts.get(projectHost.fileNameToScriptId(fileName));
|
|
274
|
+
if (sourceScript && !sourceScript.generated) {
|
|
275
|
+
if (!version.map.has(sourceScript.snapshot)) {
|
|
276
|
+
version.map.set(sourceScript.snapshot, version.lastVersion++);
|
|
281
277
|
}
|
|
282
|
-
return version.map.get(
|
|
278
|
+
return version.map.get(sourceScript.snapshot).toString();
|
|
283
279
|
}
|
|
284
280
|
}
|
|
285
|
-
if (
|
|
286
|
-
return
|
|
281
|
+
if (projectHost.fileExists(fileName)) {
|
|
282
|
+
return projectHost.getModifiedTime?.(fileName)?.valueOf().toString() ?? '0';
|
|
287
283
|
}
|
|
288
284
|
return '';
|
|
289
285
|
}
|
|
290
286
|
}
|
|
291
287
|
}
|
|
292
|
-
exports.
|
|
288
|
+
exports.createTypeScriptLanguage = createTypeScriptLanguage;
|
|
293
289
|
function setEquals(a, b) {
|
|
294
290
|
if (a.size !== b.size) {
|
|
295
291
|
return false;
|
|
@@ -52,17 +52,17 @@ function createAsyncLanguageServicePlugin(extensions, scriptKind, loadLanguagePl
|
|
|
52
52
|
};
|
|
53
53
|
}
|
|
54
54
|
loadLanguagePlugins(ts, info).then(languagePlugins => {
|
|
55
|
-
const
|
|
55
|
+
const language = (0, language_core_1.createLanguage)(languagePlugins, ts.sys.useCaseSensitiveFileNames, fileName => {
|
|
56
56
|
const snapshot = getScriptSnapshot(fileName);
|
|
57
57
|
if (snapshot) {
|
|
58
|
-
|
|
58
|
+
language.scripts.set(fileName, (0, language_core_1.resolveCommonLanguageId)(fileName), snapshot);
|
|
59
59
|
}
|
|
60
60
|
else {
|
|
61
|
-
|
|
61
|
+
language.scripts.delete(fileName);
|
|
62
62
|
}
|
|
63
63
|
});
|
|
64
|
-
(0, decorateLanguageService_1.decorateLanguageService)(
|
|
65
|
-
(0, decorateLanguageServiceHost_1.decorateLanguageServiceHost)(
|
|
64
|
+
(0, decorateLanguageService_1.decorateLanguageService)(language, info.languageService);
|
|
65
|
+
(0, decorateLanguageServiceHost_1.decorateLanguageServiceHost)(language, info.languageServiceHost, ts);
|
|
66
66
|
info.project.markAsDirty();
|
|
67
67
|
initialized = true;
|
|
68
68
|
});
|