@volar/typescript 2.1.4 → 2.2.0-alpha.0

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.
@@ -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 files = (0, language_core_1.createFileRegistry)(getLanguagePlugins(ts, options), ts.sys.useCaseSensitiveFileNames, fileName => {
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
- files.set(fileName, (0, language_core_1.resolveCommonLanguageId)(fileName), snapshot);
34
+ language.scripts.set(fileName, (0, language_core_1.resolveCommonLanguageId)(fileName), snapshot);
35
35
  }
36
36
  else {
37
- files.delete(fileName);
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 sourceFile = files.get(fileName);
64
- assert(!!sourceFile, '!!sourceFile');
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 (sourceFile.generated?.languagePlugin.typescript) {
68
- const { getScript, getExtraScripts } = sourceFile.generated.languagePlugin.typescript;
69
- const script = getScript(sourceFile.generated.code);
70
- if (script) {
71
- scriptKind = script.scriptKind;
72
- patchedText += script.code.snapshot.getText(0, script.code.snapshot.getLength());
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)(files, program);
99
- program.__volar__ = { files };
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);
@@ -1,10 +1,10 @@
1
- import { FileRegistry, CodeInformation } from '@volar/language-core';
1
+ import { Language, CodeInformation } from '@volar/language-core';
2
2
  import type * as ts from 'typescript';
3
- export declare function transformCallHierarchyItem(files: FileRegistry, item: ts.CallHierarchyItem, filter: (data: CodeInformation) => boolean): ts.CallHierarchyItem;
4
- export declare function transformDiagnostic<T extends ts.Diagnostic>(files: FileRegistry, diagnostic: T): T | undefined;
5
- export declare function transformFileTextChanges(files: FileRegistry, changes: ts.FileTextChanges, filter: (data: CodeInformation) => boolean): ts.FileTextChanges | undefined;
6
- export declare function transformDocumentSpan<T extends ts.DocumentSpan>(files: FileRegistry, documentSpan: T, filter: (data: CodeInformation) => boolean, shouldFallback?: boolean): T | undefined;
7
- export declare function transformSpan(files: FileRegistry, fileName: string | undefined, textSpan: ts.TextSpan | undefined, filter: (data: CodeInformation) => boolean): {
3
+ export declare function transformCallHierarchyItem(files: Language, item: ts.CallHierarchyItem, filter: (data: CodeInformation) => boolean): ts.CallHierarchyItem;
4
+ export declare function transformDiagnostic<T extends ts.Diagnostic>(files: Language, diagnostic: T): T | undefined;
5
+ export declare function transformFileTextChanges(files: Language, changes: ts.FileTextChanges, filter: (data: CodeInformation) => boolean): ts.FileTextChanges | undefined;
6
+ export declare function transformDocumentSpan<T extends ts.DocumentSpan>(files: Language, documentSpan: T, filter: (data: CodeInformation) => boolean, shouldFallback?: boolean): T | undefined;
7
+ export declare function transformSpan(files: Language, fileName: string | undefined, textSpan: ts.TextSpan | undefined, filter: (data: CodeInformation) => boolean): {
8
8
  fileName: string;
9
9
  textSpan: ts.TextSpan;
10
10
  } | undefined;
@@ -26,9 +26,9 @@ function transformDiagnostic(files, diagnostic) {
26
26
  if (diagnostic.file !== undefined
27
27
  && diagnostic.start !== undefined
28
28
  && diagnostic.length !== undefined) {
29
- const [virtualCode, sourceFile, map] = (0, utils_1.getVirtualFileAndMap)(files, diagnostic.file.fileName);
29
+ const [virtualCode, sourceScript, map] = (0, utils_1.getServiceScript)(files, diagnostic.file.fileName);
30
30
  if (virtualCode) {
31
- const sourceRange = transformRange(sourceFile, map, diagnostic.start, diagnostic.start + diagnostic.length, language_core_1.shouldReportDiagnostics);
31
+ const sourceRange = transformRange(sourceScript, map, diagnostic.start, diagnostic.start + diagnostic.length, language_core_1.shouldReportDiagnostics);
32
32
  if (sourceRange) {
33
33
  transformedDiagnostics.set(diagnostic, {
34
34
  ...diagnostic,
@@ -49,7 +49,7 @@ function transformDiagnostic(files, diagnostic) {
49
49
  }
50
50
  exports.transformDiagnostic = transformDiagnostic;
51
51
  function transformFileTextChanges(files, changes, filter) {
52
- const [_, source] = (0, utils_1.getVirtualFileAndMap)(files, changes.fileName);
52
+ const [_, source] = (0, utils_1.getServiceScript)(files, changes.fileName);
53
53
  if (source) {
54
54
  return {
55
55
  ...changes,
@@ -72,7 +72,7 @@ exports.transformFileTextChanges = transformFileTextChanges;
72
72
  function transformDocumentSpan(files, documentSpan, filter, shouldFallback) {
73
73
  let textSpan = transformSpan(files, documentSpan.fileName, documentSpan.textSpan, filter);
74
74
  if (!textSpan && shouldFallback) {
75
- const [virtualCode] = (0, utils_1.getVirtualFileAndMap)(files, documentSpan.fileName);
75
+ const [virtualCode] = (0, utils_1.getServiceScript)(files, documentSpan.fileName);
76
76
  if (virtualCode) {
77
77
  textSpan = {
78
78
  fileName: documentSpan.fileName,
@@ -104,9 +104,9 @@ function transformSpan(files, fileName, textSpan, filter) {
104
104
  if (!textSpan) {
105
105
  return;
106
106
  }
107
- const [virtualFile, sourceFile, map] = (0, utils_1.getVirtualFileAndMap)(files, fileName);
107
+ const [virtualFile, sourceScript, map] = (0, utils_1.getServiceScript)(files, fileName);
108
108
  if (virtualFile) {
109
- const sourceRange = transformRange(sourceFile, map, textSpan.start, textSpan.start + textSpan.length, filter);
109
+ const sourceRange = transformRange(sourceScript, map, textSpan.start, textSpan.start + textSpan.length, filter);
110
110
  if (sourceRange) {
111
111
  return {
112
112
  fileName,
@@ -125,10 +125,10 @@ function transformSpan(files, fileName, textSpan, filter) {
125
125
  }
126
126
  }
127
127
  exports.transformSpan = transformSpan;
128
- function transformRange(sourceFile, map, start, end, filter) {
129
- for (const sourceStart of map.getSourceOffsets(start - sourceFile.snapshot.getLength())) {
128
+ function transformRange(sourceScript, map, start, end, filter) {
129
+ for (const sourceStart of map.getSourceOffsets(start - sourceScript.snapshot.getLength())) {
130
130
  if (filter(sourceStart[1].data)) {
131
- for (const sourceEnd of map.getSourceOffsets(end - sourceFile.snapshot.getLength())) {
131
+ for (const sourceEnd of map.getSourceOffsets(end - sourceScript.snapshot.getLength())) {
132
132
  if (sourceEnd[0] >= sourceStart[0] && filter(sourceEnd[1].data)) {
133
133
  return [sourceStart[0], sourceEnd[0]];
134
134
  }
@@ -1,3 +1,3 @@
1
- import type { FileRegistry } from '@volar/language-core';
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 getVirtualFileAndMap(files: FileRegistry, fileName: string): readonly [import("@volar/language-core").ServiceScript, import("@volar/language-core").SourceFile, import("@volar/language-core").SourceMap<import("@volar/language-core").CodeInformation>] | readonly [undefined, undefined, undefined];
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.getVirtualFileAndMap = exports.notEmpty = void 0;
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 getVirtualFileAndMap(files, fileName) {
9
- const sourceFile = files.get(fileName);
10
- if (sourceFile?.generated) {
11
- const script = sourceFile.generated.languagePlugin.typescript?.getScript(sourceFile.generated.code);
12
- if (script) {
13
- for (const map of files.getMaps(script.code)) {
14
- if (map[1][0] === sourceFile.snapshot) {
15
- return [script, sourceFile, map[1][1]];
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.getVirtualFileAndMap = getVirtualFileAndMap;
21
+ exports.getServiceScript = getServiceScript;
23
22
  //# sourceMappingURL=utils.js.map
@@ -1,7 +1,2 @@
1
- import { LanguagePlugin, LanguageContext, TypeScriptProjectHost } from '@volar/language-core';
2
- import type * as ts from 'typescript';
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.createLanguage = void 0;
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 createLanguage(ts, sys, languagePlugins, configFileName, projectHost, { fileIdToFileName, fileNameToFileId }) {
11
- const files = (0, language_core_1.createFileRegistry)(languagePlugins, sys.useCaseSensitiveFileNames, fileId => {
12
- const fileName = fileIdToFileName(fileId);
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 = sys.getModifiedTime?.(fileName)?.valueOf();
18
+ const modifiedTime = projectHost.getModifiedTime?.(fileName)?.valueOf();
19
19
  if (!cache || cache[0] !== modifiedTime) {
20
- if (sys.fileExists(fileName)) {
21
- const text = sys.readFile(fileName);
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
- files.set(fileId, projectHost.getLanguageId(fileId), snapshot);
32
+ language.scripts.set(scriptId, projectHost.getLanguageId(scriptId), snapshot);
33
33
  }
34
34
  else {
35
- files.delete(fileId);
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 => files.get(fileNameToFileId(fileName)));
48
- let lastSysVersion = 'version' in sys ? sys.version : undefined;
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 ('version' in sys && lastSysVersion !== sys.version) {
51
- lastSysVersion = sys.version;
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 ('version' in sys && lastSysVersion !== sys.version) {
60
- lastSysVersion = sys.version;
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
- return {
69
- files,
70
- typescript: {
71
- configFileName,
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(sys.useCaseSensitiveFileNames);
82
- let extraScriptRegistry = new language_core_1.FileMap(sys.useCaseSensitiveFileNames);
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
- ...sys,
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 sys.useCaseSensitiveFileNames;
106
+ return projectHost.useCaseSensitiveFileNames;
111
107
  },
112
108
  getNewLine() {
113
- return sys.newLine;
109
+ return projectHost.newLine;
114
110
  },
115
111
  getTypeRootsVersion: () => {
116
- return 'version' in sys ? sys.version : -1; // TODO: only update for /node_modules changes?
112
+ return projectHost.getSystemVersion?.() ?? -1; // TODO: only update for /node_modules changes?
117
113
  },
118
114
  getDirectories(dirName) {
119
- return sys.getDirectories(dirName);
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 sys.readDirectory(dirName, extensions, excludes, includes, depth);
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 + ('version' in sys ? `:${sys.version}` : '');
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 sourceFile = files.get(fileNameToFileId(fileName));
154
- if (sourceFile?.generated) {
155
- const tsCode = sourceFile.generated.languagePlugin.typescript?.getScript(sourceFile.generated.code);
156
- if (tsCode) {
157
- return tsCode.scriptKind;
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 sourceFile = files.get(fileNameToFileId(fileName));
203
- if (sourceFile?.generated) {
204
- const script = sourceFile.generated.languagePlugin.typescript?.getScript(sourceFile.generated.code);
205
- if (script) {
206
- newTsVirtualFileSnapshots.add(script.code.snapshot);
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 extraScript of sourceFile.generated.languagePlugin.typescript?.getExtraScripts?.(fileName, sourceFile.generated.code) ?? []) {
210
- newTsVirtualFileSnapshots.add(extraScript.code.snapshot);
211
- tsFileNamesSet.add(extraScript.fileName);
212
- extraScriptRegistry.set(extraScript.fileName, extraScript);
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)(sourceFile.generated.code)) {
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 sourceFile = files.get(fileNameToFileId(fileName));
242
- if (sourceFile?.generated) {
243
- const script = sourceFile.generated.languagePlugin.typescript?.getScript(sourceFile.generated.code);
244
- if (script) {
245
- return script.code.snapshot;
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 (sourceFile) {
249
- return sourceFile.snapshot;
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 sourceFile = files.get(fileNameToFileId(fileName));
266
- if (sourceFile?.generated) {
267
- const script = sourceFile.generated.languagePlugin.typescript?.getScript(sourceFile.generated.code);
268
- if (script) {
269
- if (!version.map.has(script.code.snapshot)) {
270
- version.map.set(script.code.snapshot, version.lastVersion++);
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(script.code.snapshot).toString();
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 sourceFile = files.get(fileNameToFileId(fileName));
278
- if (sourceFile && !sourceFile.generated) {
279
- if (!version.map.has(sourceFile.snapshot)) {
280
- version.map.set(sourceFile.snapshot, version.lastVersion++);
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(sourceFile.snapshot).toString();
278
+ return version.map.get(sourceScript.snapshot).toString();
283
279
  }
284
280
  }
285
- if (sys.fileExists(fileName)) {
286
- return sys.getModifiedTime?.(fileName)?.valueOf().toString() ?? '0';
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.createLanguage = createLanguage;
288
+ exports.createTypeScriptLanguage = createTypeScriptLanguage;
293
289
  function setEquals(a, b) {
294
290
  if (a.size !== b.size) {
295
291
  return false;
@@ -17,36 +17,31 @@ function createSys(ts, env, currentDirectory) {
17
17
  };
18
18
  const promises = new Set();
19
19
  const fileWatcher = env.onDidChangeWatchedFiles?.(({ changes }) => {
20
+ version++;
20
21
  for (const change of changes) {
21
22
  const fileName = env.typescript.uriToFileName(change.uri);
22
23
  const dirName = path.dirname(fileName);
23
24
  const baseName = path.basename(fileName);
24
- const dir = getDir(dirName);
25
- if (dir.files.has(baseName) || dir.requestedRead) { // is requested file or directory
26
- version++;
27
- if (change.type === 1 || change.type === 2) {
28
- dir.files.set(normalizeFileId(baseName), {
29
- name: baseName,
30
- stat: {
31
- type: 1,
32
- ctime: Date.now(),
33
- mtime: Date.now(),
34
- size: -1,
35
- },
36
- requestedStat: false,
37
- requestedText: false,
38
- });
39
- }
40
- else if (change.type === 3) {
41
- dir.files.set(normalizeFileId(baseName), {
42
- name: baseName,
43
- stat: undefined,
44
- text: undefined,
45
- requestedStat: true,
46
- requestedText: true,
47
- });
48
- }
49
- }
25
+ const fileExists = change.type === 1
26
+ || change.type === 2;
27
+ const dir = getDir(dirName, fileExists);
28
+ dir.files.set(normalizeFileId(baseName), fileExists ? {
29
+ name: baseName,
30
+ stat: {
31
+ type: 1,
32
+ ctime: Date.now(),
33
+ mtime: Date.now(),
34
+ size: -1,
35
+ },
36
+ requestedStat: false,
37
+ requestedText: false,
38
+ } : {
39
+ name: baseName,
40
+ stat: undefined,
41
+ text: undefined,
42
+ requestedStat: true,
43
+ requestedText: true,
44
+ });
50
45
  }
51
46
  });
52
47
  return {
@@ -335,7 +330,7 @@ function createSys(ts, env, currentDirectory) {
335
330
  }
336
331
  return updated;
337
332
  }
338
- function getDir(dirName) {
333
+ function getDir(dirName, markExists = false) {
339
334
  const dirNames = [];
340
335
  let currentDirPath = dirName;
341
336
  let currentDirName = path.basename(currentDirPath);
@@ -350,6 +345,10 @@ function createSys(ts, env, currentDirectory) {
350
345
  for (let i = dirNames.length - 1; i >= 0; i--) {
351
346
  const nextDirName = dirNames[i];
352
347
  currentDir = getDirFromDir(currentDir, nextDirName);
348
+ if (markExists && !currentDir.exists) {
349
+ currentDir.exists = true;
350
+ version++;
351
+ }
353
352
  }
354
353
  return currentDir;
355
354
  }
@@ -52,17 +52,17 @@ function createAsyncLanguageServicePlugin(extensions, scriptKind, loadLanguagePl
52
52
  };
53
53
  }
54
54
  loadLanguagePlugins(ts, info).then(languagePlugins => {
55
- const files = (0, language_core_1.createFileRegistry)(languagePlugins, ts.sys.useCaseSensitiveFileNames, fileName => {
55
+ const language = (0, language_core_1.createLanguage)(languagePlugins, ts.sys.useCaseSensitiveFileNames, fileName => {
56
56
  const snapshot = getScriptSnapshot(fileName);
57
57
  if (snapshot) {
58
- files.set(fileName, (0, language_core_1.resolveCommonLanguageId)(fileName), snapshot);
58
+ language.scripts.set(fileName, (0, language_core_1.resolveCommonLanguageId)(fileName), snapshot);
59
59
  }
60
60
  else {
61
- files.delete(fileName);
61
+ language.scripts.delete(fileName);
62
62
  }
63
63
  });
64
- (0, decorateLanguageService_1.decorateLanguageService)(files, info.languageService);
65
- (0, decorateLanguageServiceHost_1.decorateLanguageServiceHost)(files, info.languageServiceHost, ts);
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
  });
@@ -23,17 +23,17 @@ function createLanguageServicePlugin(loadLanguagePlugins) {
23
23
  .flat();
24
24
  projectExternalFileExtensions.set(info.project, extensions);
25
25
  const getScriptSnapshot = info.languageServiceHost.getScriptSnapshot.bind(info.languageServiceHost);
26
- const files = (0, language_core_1.createFileRegistry)(languagePlugins, ts.sys.useCaseSensitiveFileNames, fileName => {
26
+ const language = (0, language_core_1.createLanguage)(languagePlugins, ts.sys.useCaseSensitiveFileNames, fileName => {
27
27
  const snapshot = getScriptSnapshot(fileName);
28
28
  if (snapshot) {
29
- files.set(fileName, (0, language_core_1.resolveCommonLanguageId)(fileName), snapshot);
29
+ language.scripts.set(fileName, (0, language_core_1.resolveCommonLanguageId)(fileName), snapshot);
30
30
  }
31
31
  else {
32
- files.delete(fileName);
32
+ language.scripts.delete(fileName);
33
33
  }
34
34
  });
35
- (0, decorateLanguageService_1.decorateLanguageService)(files, info.languageService);
36
- (0, decorateLanguageServiceHost_1.decorateLanguageServiceHost)(files, info.languageServiceHost, ts);
35
+ (0, decorateLanguageService_1.decorateLanguageService)(language, info.languageService);
36
+ (0, decorateLanguageServiceHost_1.decorateLanguageServiceHost)(language, info.languageServiceHost, ts);
37
37
  }
38
38
  return info.languageService;
39
39
  },
@@ -1,3 +1,3 @@
1
- import type { LanguagePlugin, SourceFile } from '@volar/language-core';
1
+ import type { LanguagePlugin, SourceScript } from '@volar/language-core';
2
2
  import type * as ts from 'typescript';
3
- export declare function createResolveModuleName(ts: typeof import('typescript'), languageServiceHost: ts.LanguageServiceHost, languagePlugins: LanguagePlugin<any>[], getFile: (fileName: string) => SourceFile | undefined): (moduleName: string, containingFile: string, compilerOptions: ts.CompilerOptions, cache?: ts.ModuleResolutionCache, redirectedReference?: ts.ResolvedProjectReference, resolutionMode?: ts.ResolutionMode) => ts.ResolvedModuleWithFailedLookupLocations;
3
+ export declare function createResolveModuleName(ts: typeof import('typescript'), languageServiceHost: ts.LanguageServiceHost, languagePlugins: LanguagePlugin<any>[], getSourceScript: (fileName: string) => SourceScript | undefined): (moduleName: string, containingFile: string, compilerOptions: ts.CompilerOptions, cache?: ts.ModuleResolutionCache, redirectedReference?: ts.ResolvedProjectReference, resolutionMode?: ts.ResolutionMode) => ts.ResolvedModuleWithFailedLookupLocations;
@@ -1,7 +1,7 @@
1
1
  "use strict";
2
2
  Object.defineProperty(exports, "__esModule", { value: true });
3
3
  exports.createResolveModuleName = void 0;
4
- function createResolveModuleName(ts, languageServiceHost, languagePlugins, getFile) {
4
+ function createResolveModuleName(ts, languageServiceHost, languagePlugins, getSourceScript) {
5
5
  const toPatchResults = new Map();
6
6
  const moduleResolutionHost = {
7
7
  readFile: languageServiceHost.readFile.bind(languageServiceHost),
@@ -32,11 +32,11 @@ function createResolveModuleName(ts, languageServiceHost, languagePlugins, getFi
32
32
  const result = ts.resolveModuleName(moduleName, containingFile, compilerOptions, moduleResolutionHost, cache, redirectedReference, resolutionMode);
33
33
  if (result.resolvedModule && toPatchResults.has(result.resolvedModule.resolvedFileName)) {
34
34
  result.resolvedModule.resolvedFileName = toPatchResults.get(result.resolvedModule.resolvedFileName);
35
- const sourceFile = getFile(result.resolvedModule.resolvedFileName);
36
- if (sourceFile?.generated) {
37
- const tsCode = sourceFile.generated.languagePlugin.typescript?.getScript(sourceFile.generated.code);
38
- if (tsCode) {
39
- result.resolvedModule.extension = tsCode.extension;
35
+ const sourceScript = getSourceScript(result.resolvedModule.resolvedFileName);
36
+ if (sourceScript?.generated) {
37
+ const serviceScript = sourceScript.generated.languagePlugin.typescript?.getServiceScript(sourceScript.generated.root);
38
+ if (serviceScript) {
39
+ result.resolvedModule.extension = serviceScript.extension;
40
40
  }
41
41
  }
42
42
  }