@vue/language-server 3.0.0-alpha.8 → 3.0.0-beta.2
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/bin/vue-language-server.js +4 -4
- package/index.d.ts +1 -3
- package/index.js +309 -19
- package/lib/reactionsAnalyze.d.ts +27 -0
- package/lib/reactionsAnalyze.js +456 -0
- package/lib/reactionsAnalyzeLS.d.ts +5 -0
- package/lib/reactionsAnalyzeLS.js +35 -0
- package/package.json +11 -10
- package/lib/types.d.ts +0 -8
- package/lib/types.js +0 -19
- package/node.d.ts +0 -1
- package/node.js +0 -149
- package/protocol.d.ts +0 -1
- package/protocol.js +0 -18
package/index.d.ts
CHANGED
package/index.js
CHANGED
|
@@ -1,22 +1,312 @@
|
|
|
1
1
|
"use strict";
|
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
-
if (k2 === undefined) k2 = k;
|
|
4
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
-
}
|
|
8
|
-
Object.defineProperty(o, k2, desc);
|
|
9
|
-
}) : (function(o, m, k, k2) {
|
|
10
|
-
if (k2 === undefined) k2 = k;
|
|
11
|
-
o[k2] = m[k];
|
|
12
|
-
}));
|
|
13
|
-
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
-
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
-
};
|
|
16
2
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
3
|
+
const language_server_1 = require("@volar/language-server");
|
|
4
|
+
const simpleProject_1 = require("@volar/language-server/lib/project/simpleProject");
|
|
5
|
+
const node_1 = require("@volar/language-server/node");
|
|
6
|
+
const language_core_1 = require("@vue/language-core");
|
|
7
|
+
const language_service_1 = require("@vue/language-service");
|
|
8
|
+
const ts = require("typescript");
|
|
9
|
+
const vscode_uri_1 = require("vscode-uri");
|
|
10
|
+
const reactionsAnalyze_1 = require("./lib/reactionsAnalyze");
|
|
11
|
+
const reactionsAnalyzeLS_1 = require("./lib/reactionsAnalyzeLS");
|
|
12
|
+
const connection = (0, node_1.createConnection)();
|
|
13
|
+
const server = (0, node_1.createServer)(connection);
|
|
14
|
+
connection.listen();
|
|
15
|
+
connection.onInitialize(params => {
|
|
16
|
+
const tsconfigProjects = (0, language_service_1.createUriMap)();
|
|
17
|
+
const file2ProjectInfo = new Map();
|
|
18
|
+
server.fileWatcher.onDidChangeWatchedFiles(({ changes }) => {
|
|
19
|
+
for (const change of changes) {
|
|
20
|
+
const changeUri = vscode_uri_1.URI.parse(change.uri);
|
|
21
|
+
if (tsconfigProjects.has(changeUri)) {
|
|
22
|
+
tsconfigProjects.get(changeUri).dispose();
|
|
23
|
+
tsconfigProjects.delete(changeUri);
|
|
24
|
+
file2ProjectInfo.clear();
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
});
|
|
28
|
+
let simpleLs;
|
|
29
|
+
let tsserverRequestId = 0;
|
|
30
|
+
const tsserverRequestHandlers = new Map();
|
|
31
|
+
connection.onNotification('tsserver/response', ([id, res]) => {
|
|
32
|
+
tsserverRequestHandlers.get(id)?.(res);
|
|
33
|
+
tsserverRequestHandlers.delete(id);
|
|
34
|
+
});
|
|
35
|
+
return server.initialize(params, {
|
|
36
|
+
setup() { },
|
|
37
|
+
async getLanguageService(uri) {
|
|
38
|
+
if (uri.scheme === 'file') {
|
|
39
|
+
const fileName = uri.fsPath.replace(/\\/g, '/');
|
|
40
|
+
let projectInfoPromise = file2ProjectInfo.get(fileName);
|
|
41
|
+
if (!projectInfoPromise) {
|
|
42
|
+
projectInfoPromise = sendTsRequest(ts.server.protocol.CommandTypes.ProjectInfo, {
|
|
43
|
+
file: fileName,
|
|
44
|
+
needFileNameList: false,
|
|
45
|
+
});
|
|
46
|
+
file2ProjectInfo.set(fileName, projectInfoPromise);
|
|
47
|
+
}
|
|
48
|
+
const projectInfo = await projectInfoPromise;
|
|
49
|
+
if (projectInfo) {
|
|
50
|
+
const { configFileName } = projectInfo;
|
|
51
|
+
let ls = tsconfigProjects.get(vscode_uri_1.URI.file(configFileName));
|
|
52
|
+
if (!ls) {
|
|
53
|
+
ls = createLs(server, configFileName);
|
|
54
|
+
tsconfigProjects.set(vscode_uri_1.URI.file(configFileName), ls);
|
|
55
|
+
}
|
|
56
|
+
return ls;
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
return simpleLs ??= createLs(server, undefined);
|
|
60
|
+
},
|
|
61
|
+
getExistingLanguageServices() {
|
|
62
|
+
return Promise.all([
|
|
63
|
+
...tsconfigProjects.values(),
|
|
64
|
+
simpleLs,
|
|
65
|
+
].filter(promise => !!promise));
|
|
66
|
+
},
|
|
67
|
+
reload() {
|
|
68
|
+
for (const ls of tsconfigProjects.values()) {
|
|
69
|
+
ls.dispose();
|
|
70
|
+
}
|
|
71
|
+
tsconfigProjects.clear();
|
|
72
|
+
if (simpleLs) {
|
|
73
|
+
simpleLs.dispose();
|
|
74
|
+
simpleLs = undefined;
|
|
75
|
+
}
|
|
76
|
+
},
|
|
77
|
+
}, (0, language_service_1.createVueLanguageServicePlugins)(ts, {
|
|
78
|
+
collectExtractProps(...args) {
|
|
79
|
+
return sendTsRequest('vue:collectExtractProps', args);
|
|
80
|
+
},
|
|
81
|
+
getComponentDirectives(...args) {
|
|
82
|
+
return sendTsRequest('vue:getComponentDirectives', args);
|
|
83
|
+
},
|
|
84
|
+
getComponentEvents(...args) {
|
|
85
|
+
return sendTsRequest('vue:getComponentEvents', args);
|
|
86
|
+
},
|
|
87
|
+
getComponentNames(...args) {
|
|
88
|
+
return sendTsRequest('vue:getComponentNames', args);
|
|
89
|
+
},
|
|
90
|
+
getComponentProps(...args) {
|
|
91
|
+
return sendTsRequest('vue:getComponentProps', args);
|
|
92
|
+
},
|
|
93
|
+
getElementAttrs(...args) {
|
|
94
|
+
return sendTsRequest('vue:getElementAttrs', args);
|
|
95
|
+
},
|
|
96
|
+
getElementNames(...args) {
|
|
97
|
+
return sendTsRequest('vue:getElementNames', args);
|
|
98
|
+
},
|
|
99
|
+
getImportPathForFile(...args) {
|
|
100
|
+
return sendTsRequest('vue:getImportPathForFile', args);
|
|
101
|
+
},
|
|
102
|
+
getPropertiesAtLocation(...args) {
|
|
103
|
+
return sendTsRequest('vue:getPropertiesAtLocation', args);
|
|
104
|
+
},
|
|
105
|
+
getDocumentHighlights(fileName, position) {
|
|
106
|
+
return sendTsRequest('documentHighlights-full', // internal command
|
|
107
|
+
{
|
|
108
|
+
file: fileName,
|
|
109
|
+
...{ position },
|
|
110
|
+
filesToSearch: [fileName],
|
|
111
|
+
});
|
|
112
|
+
},
|
|
113
|
+
async getQuickInfoAtPosition(fileName, { line, character }) {
|
|
114
|
+
const result = await sendTsRequest(ts.server.protocol.CommandTypes.Quickinfo, {
|
|
115
|
+
file: fileName,
|
|
116
|
+
line: line + 1,
|
|
117
|
+
offset: character + 1,
|
|
118
|
+
});
|
|
119
|
+
return ts.displayPartsToString(result?.displayParts ?? []);
|
|
120
|
+
},
|
|
121
|
+
}));
|
|
122
|
+
async function sendTsRequest(command, args) {
|
|
123
|
+
return await new Promise(resolve => {
|
|
124
|
+
const requestId = ++tsserverRequestId;
|
|
125
|
+
tsserverRequestHandlers.set(requestId, resolve);
|
|
126
|
+
connection.sendNotification('tsserver/request', [requestId, command, args]);
|
|
127
|
+
});
|
|
128
|
+
}
|
|
129
|
+
function createLs(server, tsconfig) {
|
|
130
|
+
const commonLine = tsconfig
|
|
131
|
+
? (0, language_core_1.createParsedCommandLine)(ts, ts.sys, tsconfig)
|
|
132
|
+
: {
|
|
133
|
+
options: ts.getDefaultCompilerOptions(),
|
|
134
|
+
vueOptions: (0, language_core_1.getDefaultCompilerOptions)(),
|
|
135
|
+
};
|
|
136
|
+
const language = (0, language_core_1.createLanguage)([
|
|
137
|
+
{
|
|
138
|
+
getLanguageId: uri => server.documents.get(uri)?.languageId,
|
|
139
|
+
},
|
|
140
|
+
(0, language_core_1.createVueLanguagePlugin)(ts, commonLine.options, commonLine.vueOptions, uri => uri.fsPath.replace(/\\/g, '/')),
|
|
141
|
+
], (0, language_service_1.createUriMap)(), uri => {
|
|
142
|
+
const document = server.documents.get(uri);
|
|
143
|
+
if (document) {
|
|
144
|
+
language.scripts.set(uri, document.getSnapshot(), document.languageId);
|
|
145
|
+
}
|
|
146
|
+
else {
|
|
147
|
+
language.scripts.delete(uri);
|
|
148
|
+
}
|
|
149
|
+
});
|
|
150
|
+
return (0, language_service_1.createLanguageService)(language, server.languageServicePlugins, (0, simpleProject_1.createLanguageServiceEnvironment)(server, [...server.workspaceFolders.all]), { vue: { compilerOptions: commonLine.vueOptions } });
|
|
151
|
+
}
|
|
152
|
+
});
|
|
153
|
+
connection.onInitialized(server.initialized);
|
|
154
|
+
connection.onShutdown(server.shutdown);
|
|
155
|
+
connection.onRequest('vue/parseSfc', (params) => {
|
|
156
|
+
const document = server.documents.get(vscode_uri_1.URI.parse(params.textDocument.uri));
|
|
157
|
+
if (document) {
|
|
158
|
+
return (0, language_core_1.parse)(document.getText());
|
|
159
|
+
}
|
|
160
|
+
});
|
|
161
|
+
connection.onRequest('vue/interpolationRanges', async (params) => {
|
|
162
|
+
const uri = vscode_uri_1.URI.parse(params.textDocument.uri);
|
|
163
|
+
const languageService = await server.project.getLanguageService(uri);
|
|
164
|
+
if (languageService) {
|
|
165
|
+
const sourceFile = languageService.context.language.scripts.get(uri);
|
|
166
|
+
if (sourceFile?.generated) {
|
|
167
|
+
const ranges = [];
|
|
168
|
+
for (const code of (0, language_core_1.forEachEmbeddedCode)(sourceFile.generated.root)) {
|
|
169
|
+
const codeText = code.snapshot.getText(0, code.snapshot.getLength());
|
|
170
|
+
if ((code.id.startsWith('template_inline_ts_')
|
|
171
|
+
&& codeText.startsWith('0 +')
|
|
172
|
+
&& codeText.endsWith('+ 0;'))
|
|
173
|
+
|| (code.id.startsWith('style_') && code.id.endsWith('_inline_ts'))) {
|
|
174
|
+
for (const mapping of code.mappings) {
|
|
175
|
+
for (let i = 0; i < mapping.sourceOffsets.length; i++) {
|
|
176
|
+
ranges.push([
|
|
177
|
+
mapping.sourceOffsets[i],
|
|
178
|
+
mapping.sourceOffsets[i] + mapping.lengths[i],
|
|
179
|
+
]);
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
}
|
|
184
|
+
return ranges;
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
return [];
|
|
188
|
+
});
|
|
189
|
+
const cacheDocuments = new Map();
|
|
190
|
+
connection.onRequest('vue/reactionsAnalyze', async (params) => {
|
|
191
|
+
if (params.syncDocument) {
|
|
192
|
+
const document = language_server_1.TextDocument.create(params.textDocument.uri, params.syncDocument.languageId, 0, params.syncDocument.content);
|
|
193
|
+
const snapshot = ts.ScriptSnapshot.fromString(params.syncDocument.content);
|
|
194
|
+
cacheDocuments.set(params.textDocument.uri, [document, snapshot]);
|
|
195
|
+
}
|
|
196
|
+
const uri = vscode_uri_1.URI.parse(params.textDocument.uri);
|
|
197
|
+
const languageService = await server.project.getLanguageService(uri);
|
|
198
|
+
const sourceScript = languageService.context.language.scripts.get(uri);
|
|
199
|
+
let document;
|
|
200
|
+
let snapshot;
|
|
201
|
+
if (sourceScript) {
|
|
202
|
+
document = languageService.context.documents.get(sourceScript.id, sourceScript.languageId, sourceScript.snapshot);
|
|
203
|
+
snapshot = sourceScript.snapshot;
|
|
204
|
+
}
|
|
205
|
+
else if (cacheDocuments.has(params.textDocument.uri)) {
|
|
206
|
+
const [doc, snap] = cacheDocuments.get(params.textDocument.uri);
|
|
207
|
+
document = doc;
|
|
208
|
+
snapshot = snap;
|
|
209
|
+
}
|
|
210
|
+
if (!document || !snapshot) {
|
|
211
|
+
return;
|
|
212
|
+
}
|
|
213
|
+
let offset = document.offsetAt(params.position);
|
|
214
|
+
if (sourceScript?.generated) {
|
|
215
|
+
const serviceScript = sourceScript.generated.languagePlugin.typescript?.getServiceScript(sourceScript.generated.root);
|
|
216
|
+
if (!serviceScript) {
|
|
217
|
+
return;
|
|
218
|
+
}
|
|
219
|
+
const map = languageService.context.language.maps.get(serviceScript.code, sourceScript);
|
|
220
|
+
let embeddedOffset;
|
|
221
|
+
for (const [mapped, mapping] of map.toGeneratedLocation(offset)) {
|
|
222
|
+
if ((0, language_core_1.isReferencesEnabled)(mapping.data)) {
|
|
223
|
+
embeddedOffset = mapped;
|
|
224
|
+
break;
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
if (embeddedOffset === undefined) {
|
|
228
|
+
return;
|
|
229
|
+
}
|
|
230
|
+
offset = embeddedOffset;
|
|
231
|
+
const embeddedUri = languageService.context.encodeEmbeddedDocumentUri(sourceScript.id, serviceScript.code.id);
|
|
232
|
+
document = languageService.context.documents.get(embeddedUri, serviceScript.code.languageId, serviceScript.code.snapshot);
|
|
233
|
+
snapshot = serviceScript.code.snapshot;
|
|
234
|
+
}
|
|
235
|
+
const { languageService: tsLs, fileName } = (0, reactionsAnalyzeLS_1.getLanguageService)(ts, snapshot, document.languageId);
|
|
236
|
+
const result = (0, reactionsAnalyze_1.analyze)(ts, tsLs, fileName, offset);
|
|
237
|
+
if (!result) {
|
|
238
|
+
return;
|
|
239
|
+
}
|
|
240
|
+
const subscribers = [];
|
|
241
|
+
const dependencies = [];
|
|
242
|
+
if (sourceScript?.generated) {
|
|
243
|
+
const serviceScript = sourceScript.generated.languagePlugin.typescript?.getServiceScript(sourceScript.generated.root);
|
|
244
|
+
if (!serviceScript) {
|
|
245
|
+
return;
|
|
246
|
+
}
|
|
247
|
+
const docs = [
|
|
248
|
+
languageService.context.documents.get(sourceScript.id, sourceScript.languageId, sourceScript.snapshot),
|
|
249
|
+
document,
|
|
250
|
+
languageService.context.language.maps.get(serviceScript.code, sourceScript),
|
|
251
|
+
];
|
|
252
|
+
for (const dependency of result.dependencies) {
|
|
253
|
+
let start = document.positionAt(dependency.getStart(result.sourceFile));
|
|
254
|
+
let end = document.positionAt(dependency.getEnd());
|
|
255
|
+
if (ts.isBlock(dependency) && dependency.statements.length) {
|
|
256
|
+
const { statements } = dependency;
|
|
257
|
+
start = document.positionAt(statements[0].getStart(result.sourceFile));
|
|
258
|
+
end = document.positionAt(statements[statements.length - 1].getEnd());
|
|
259
|
+
}
|
|
260
|
+
const sourceRange = (0, language_service_1.getSourceRange)(docs, { start, end });
|
|
261
|
+
if (sourceRange) {
|
|
262
|
+
dependencies.push(sourceRange);
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
for (const subscriber of result.subscribers) {
|
|
266
|
+
if (!subscriber.sideEffectInfo) {
|
|
267
|
+
continue;
|
|
268
|
+
}
|
|
269
|
+
let start = document.positionAt(subscriber.sideEffectInfo.handler.getStart(result.sourceFile));
|
|
270
|
+
let end = document.positionAt(subscriber.sideEffectInfo.handler.getEnd());
|
|
271
|
+
if (ts.isBlock(subscriber.sideEffectInfo.handler) && subscriber.sideEffectInfo.handler.statements.length) {
|
|
272
|
+
const { statements } = subscriber.sideEffectInfo.handler;
|
|
273
|
+
start = document.positionAt(statements[0].getStart(result.sourceFile));
|
|
274
|
+
end = document.positionAt(statements[statements.length - 1].getEnd());
|
|
275
|
+
}
|
|
276
|
+
const sourceRange = (0, language_service_1.getSourceRange)(docs, { start, end });
|
|
277
|
+
if (sourceRange) {
|
|
278
|
+
subscribers.push(sourceRange);
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
}
|
|
282
|
+
else {
|
|
283
|
+
for (const dependency of result.dependencies) {
|
|
284
|
+
let start = document.positionAt(dependency.getStart(result.sourceFile));
|
|
285
|
+
let end = document.positionAt(dependency.getEnd());
|
|
286
|
+
if (ts.isBlock(dependency) && dependency.statements.length) {
|
|
287
|
+
const { statements } = dependency;
|
|
288
|
+
start = document.positionAt(statements[0].getStart(result.sourceFile));
|
|
289
|
+
end = document.positionAt(statements[statements.length - 1].getEnd());
|
|
290
|
+
}
|
|
291
|
+
dependencies.push({ start, end });
|
|
292
|
+
}
|
|
293
|
+
for (const subscriber of result.subscribers) {
|
|
294
|
+
if (!subscriber.sideEffectInfo) {
|
|
295
|
+
continue;
|
|
296
|
+
}
|
|
297
|
+
let start = document.positionAt(subscriber.sideEffectInfo.handler.getStart(result.sourceFile));
|
|
298
|
+
let end = document.positionAt(subscriber.sideEffectInfo.handler.getEnd());
|
|
299
|
+
if (ts.isBlock(subscriber.sideEffectInfo.handler) && subscriber.sideEffectInfo.handler.statements.length) {
|
|
300
|
+
const { statements } = subscriber.sideEffectInfo.handler;
|
|
301
|
+
start = document.positionAt(statements[0].getStart(result.sourceFile));
|
|
302
|
+
end = document.positionAt(statements[statements.length - 1].getEnd());
|
|
303
|
+
}
|
|
304
|
+
subscribers.push({ start, end });
|
|
305
|
+
}
|
|
306
|
+
}
|
|
307
|
+
return {
|
|
308
|
+
subscribers,
|
|
309
|
+
dependencies,
|
|
310
|
+
};
|
|
311
|
+
});
|
|
22
312
|
//# sourceMappingURL=index.js.map
|
|
@@ -0,0 +1,27 @@
|
|
|
1
|
+
import type * as ts from 'typescript';
|
|
2
|
+
declare const enum TrackKind {
|
|
3
|
+
AccessDotValue = 0,
|
|
4
|
+
AccessAnyValue = 1,
|
|
5
|
+
Call = 2
|
|
6
|
+
}
|
|
7
|
+
interface SignalNode {
|
|
8
|
+
bindingInfo?: {
|
|
9
|
+
isRef: boolean;
|
|
10
|
+
name: ts.BindingName;
|
|
11
|
+
trackKinds: TrackKind[];
|
|
12
|
+
};
|
|
13
|
+
trackInfo?: {
|
|
14
|
+
depsHandler: ts.Node;
|
|
15
|
+
needToUse: boolean;
|
|
16
|
+
};
|
|
17
|
+
sideEffectInfo?: {
|
|
18
|
+
isEffect: boolean;
|
|
19
|
+
handler: ts.Node;
|
|
20
|
+
};
|
|
21
|
+
}
|
|
22
|
+
export declare function analyze(ts: typeof import('typescript'), languageService: ts.LanguageService, fileName: string, position: number): {
|
|
23
|
+
sourceFile: ts.SourceFile;
|
|
24
|
+
subscribers: SignalNode[];
|
|
25
|
+
dependencies: ts.Node[];
|
|
26
|
+
} | undefined;
|
|
27
|
+
export {};
|
|
@@ -0,0 +1,456 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.analyze = analyze;
|
|
4
|
+
const language_core_1 = require("@vue/language-core");
|
|
5
|
+
function analyze(ts, languageService, fileName, position) {
|
|
6
|
+
const sourceFile = languageService.getProgram().getSourceFile(fileName);
|
|
7
|
+
const { signals, dotValueAccesses, dotAnyAccesses, functionCalls } = collect(ts, sourceFile);
|
|
8
|
+
let signal = findSignalByNamePosition(position);
|
|
9
|
+
if (!signal) {
|
|
10
|
+
signal = findEffectByEffectHandlerPosition(position);
|
|
11
|
+
if (signal?.bindingInfo) {
|
|
12
|
+
position = signal.bindingInfo.name.getStart(sourceFile);
|
|
13
|
+
}
|
|
14
|
+
}
|
|
15
|
+
if (!signal) {
|
|
16
|
+
return;
|
|
17
|
+
}
|
|
18
|
+
const subscribers = signal.bindingInfo
|
|
19
|
+
? findSubscribers(signal.bindingInfo.name, signal.bindingInfo.trackKinds)
|
|
20
|
+
: [];
|
|
21
|
+
const dependencies = signal
|
|
22
|
+
? findDependencies(signal)
|
|
23
|
+
: [];
|
|
24
|
+
if ((!signal.sideEffectInfo?.isEffect && !subscribers.length)
|
|
25
|
+
|| (!signal.bindingInfo?.isRef && !dependencies.length)) {
|
|
26
|
+
return;
|
|
27
|
+
}
|
|
28
|
+
return {
|
|
29
|
+
sourceFile,
|
|
30
|
+
subscribers: [...new Set(subscribers)],
|
|
31
|
+
dependencies: [...new Set(dependencies)],
|
|
32
|
+
};
|
|
33
|
+
function findDependencies(signal, visited = new Set()) {
|
|
34
|
+
if (visited.has(signal)) {
|
|
35
|
+
return [];
|
|
36
|
+
}
|
|
37
|
+
visited.add(signal);
|
|
38
|
+
const nodes = [];
|
|
39
|
+
let hasRef = signal.bindingInfo?.isRef ?? false;
|
|
40
|
+
if (signal.trackInfo) {
|
|
41
|
+
const { needToUse } = signal.trackInfo;
|
|
42
|
+
visit(signal.trackInfo.depsHandler, needToUse);
|
|
43
|
+
signal.trackInfo.depsHandler.forEachChild(child => visit(child, needToUse));
|
|
44
|
+
}
|
|
45
|
+
if (!hasRef) {
|
|
46
|
+
return [];
|
|
47
|
+
}
|
|
48
|
+
return nodes;
|
|
49
|
+
function visit(node, needToUse, parentIsPropertyAccess = false) {
|
|
50
|
+
if (!needToUse) {
|
|
51
|
+
if (!parentIsPropertyAccess && ts.isIdentifier(node)) {
|
|
52
|
+
const definition = languageService.getDefinitionAtPosition(fileName, node.getStart(sourceFile));
|
|
53
|
+
for (const info of definition ?? []) {
|
|
54
|
+
if (info.fileName !== fileName) {
|
|
55
|
+
continue;
|
|
56
|
+
}
|
|
57
|
+
const signal = findSignalByNamePosition(info.textSpan.start);
|
|
58
|
+
if (!signal) {
|
|
59
|
+
continue;
|
|
60
|
+
}
|
|
61
|
+
if (signal.bindingInfo) {
|
|
62
|
+
nodes.push(signal.bindingInfo.name);
|
|
63
|
+
hasRef ||= signal.bindingInfo.isRef;
|
|
64
|
+
}
|
|
65
|
+
if (signal.sideEffectInfo) {
|
|
66
|
+
nodes.push(signal.sideEffectInfo.handler);
|
|
67
|
+
}
|
|
68
|
+
const deps = findDependencies(signal, visited);
|
|
69
|
+
nodes.push(...deps);
|
|
70
|
+
hasRef ||= deps.length > 0;
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
}
|
|
74
|
+
else if (ts.isPropertyAccessExpression(node) || ts.isElementAccessExpression(node) || ts.isCallExpression(node)) {
|
|
75
|
+
const definition = languageService.getDefinitionAtPosition(fileName, node.expression.getStart(sourceFile));
|
|
76
|
+
for (const info of definition ?? []) {
|
|
77
|
+
if (info.fileName !== fileName) {
|
|
78
|
+
continue;
|
|
79
|
+
}
|
|
80
|
+
const signal = findSignalByNamePosition(info.textSpan.start);
|
|
81
|
+
if (!signal) {
|
|
82
|
+
continue;
|
|
83
|
+
}
|
|
84
|
+
const oldSize = nodes.length;
|
|
85
|
+
if (signal.bindingInfo) {
|
|
86
|
+
for (const trackKind of signal.bindingInfo.trackKinds) {
|
|
87
|
+
if (ts.isPropertyAccessExpression(node)) {
|
|
88
|
+
if (trackKind === 0 /* TrackKind.AccessDotValue */ && node.name.text === 'value') {
|
|
89
|
+
nodes.push(signal.bindingInfo.name);
|
|
90
|
+
hasRef ||= signal.bindingInfo.isRef;
|
|
91
|
+
}
|
|
92
|
+
if (trackKind === 1 /* TrackKind.AccessAnyValue */ && node.name.text !== '') {
|
|
93
|
+
nodes.push(signal.bindingInfo.name);
|
|
94
|
+
hasRef ||= signal.bindingInfo.isRef;
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
else if (ts.isElementAccessExpression(node)) {
|
|
98
|
+
if (trackKind === 1 /* TrackKind.AccessAnyValue */) {
|
|
99
|
+
nodes.push(signal.bindingInfo.name);
|
|
100
|
+
hasRef ||= signal.bindingInfo.isRef;
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
else if (ts.isCallExpression(node)) {
|
|
104
|
+
if (trackKind === 2 /* TrackKind.Call */) {
|
|
105
|
+
nodes.push(signal.bindingInfo.name);
|
|
106
|
+
hasRef ||= signal.bindingInfo.isRef;
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
const signalDetected = nodes.length > oldSize;
|
|
112
|
+
if (signalDetected) {
|
|
113
|
+
if (signal.sideEffectInfo) {
|
|
114
|
+
nodes.push(signal.sideEffectInfo.handler);
|
|
115
|
+
}
|
|
116
|
+
const deps = findDependencies(signal, visited);
|
|
117
|
+
nodes.push(...deps);
|
|
118
|
+
hasRef ||= deps.length > 0;
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
}
|
|
122
|
+
node.forEachChild(child => visit(child, needToUse, ts.isPropertyAccessExpression(node) || ts.isElementAccessExpression(node)));
|
|
123
|
+
}
|
|
124
|
+
;
|
|
125
|
+
}
|
|
126
|
+
function findSubscribers(refName, trackKinds, visited = new Set()) {
|
|
127
|
+
return (0, language_core_1.findBindingVars)(ts, refName, sourceFile)
|
|
128
|
+
.map(binding => findSubscribersWorker(binding, trackKinds, visited))
|
|
129
|
+
.flat();
|
|
130
|
+
}
|
|
131
|
+
function findSubscribersWorker(refName, trackKinds, visited = new Set()) {
|
|
132
|
+
if (visited.has(refName.start)) {
|
|
133
|
+
return [];
|
|
134
|
+
}
|
|
135
|
+
visited.add(refName.start);
|
|
136
|
+
const references = languageService.findReferences(fileName, refName.start);
|
|
137
|
+
if (!references) {
|
|
138
|
+
return [];
|
|
139
|
+
}
|
|
140
|
+
const result = [];
|
|
141
|
+
for (const reference of references) {
|
|
142
|
+
for (const reference2 of reference.references) {
|
|
143
|
+
if (reference2.fileName !== fileName) {
|
|
144
|
+
continue;
|
|
145
|
+
}
|
|
146
|
+
const effect = findEffectByDepsHandlerPosition(reference2.textSpan.start);
|
|
147
|
+
if (effect?.trackInfo) {
|
|
148
|
+
if (effect.trackInfo.needToUse) {
|
|
149
|
+
let match = false;
|
|
150
|
+
for (const trackKind of trackKinds) {
|
|
151
|
+
if (trackKind === 1 /* TrackKind.AccessAnyValue */) {
|
|
152
|
+
match = dotAnyAccesses.has(reference2.textSpan.start + reference2.textSpan.length);
|
|
153
|
+
}
|
|
154
|
+
else if (trackKind === 0 /* TrackKind.AccessDotValue */) {
|
|
155
|
+
match = dotValueAccesses.has(reference2.textSpan.start + reference2.textSpan.length);
|
|
156
|
+
}
|
|
157
|
+
else if (trackKind === 2 /* TrackKind.Call */) {
|
|
158
|
+
match = functionCalls.has(reference2.textSpan.start + reference2.textSpan.length);
|
|
159
|
+
}
|
|
160
|
+
if (match) {
|
|
161
|
+
break;
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
if (!match) {
|
|
165
|
+
continue;
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
let hasEffect = effect.sideEffectInfo?.isEffect;
|
|
169
|
+
if (effect.bindingInfo) {
|
|
170
|
+
const subs = findSubscribers(effect.bindingInfo.name, effect.bindingInfo.trackKinds, visited);
|
|
171
|
+
result.push(...subs);
|
|
172
|
+
hasEffect ||= subs.length > 0;
|
|
173
|
+
}
|
|
174
|
+
if (hasEffect) {
|
|
175
|
+
result.push(effect);
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
return result;
|
|
181
|
+
}
|
|
182
|
+
function findSignalByNamePosition(position) {
|
|
183
|
+
return signals.find(ref => ref.bindingInfo && ref.bindingInfo.name.getStart(sourceFile) <= position && ref.bindingInfo.name.getEnd() >= position);
|
|
184
|
+
}
|
|
185
|
+
function findEffectByEffectHandlerPosition(position) {
|
|
186
|
+
return signals.find(ref => ref.sideEffectInfo && ref.sideEffectInfo.handler.getStart(sourceFile) <= position && ref.sideEffectInfo.handler.getEnd() >= position);
|
|
187
|
+
}
|
|
188
|
+
function findEffectByDepsHandlerPosition(position) {
|
|
189
|
+
return signals.find(ref => ref.trackInfo && ref.trackInfo.depsHandler.getStart(sourceFile) <= position && ref.trackInfo.depsHandler.getEnd() >= position);
|
|
190
|
+
}
|
|
191
|
+
}
|
|
192
|
+
function collect(ts, sourceFile) {
|
|
193
|
+
const signals = [];
|
|
194
|
+
const dotValueAccesses = new Set();
|
|
195
|
+
const dotAnyAccesses = new Set();
|
|
196
|
+
const functionCalls = new Set();
|
|
197
|
+
sourceFile.forEachChild(function visit(node) {
|
|
198
|
+
if (ts.isVariableDeclaration(node)) {
|
|
199
|
+
if (node.initializer && ts.isCallExpression(node.initializer)) {
|
|
200
|
+
const call = node.initializer;
|
|
201
|
+
if (ts.isIdentifier(call.expression)) {
|
|
202
|
+
const callName = call.expression.escapedText;
|
|
203
|
+
if (callName === 'ref' || callName === 'shallowRef' || callName === 'toRef' || callName === 'useTemplateRef' || callName === 'defineModel') {
|
|
204
|
+
signals.push({
|
|
205
|
+
bindingInfo: {
|
|
206
|
+
isRef: true,
|
|
207
|
+
name: node.name,
|
|
208
|
+
trackKinds: [0 /* TrackKind.AccessDotValue */],
|
|
209
|
+
},
|
|
210
|
+
});
|
|
211
|
+
}
|
|
212
|
+
else if (callName === 'reactive' || callName === 'shallowReactive' || callName === 'defineProps' || callName === 'withDefaults') {
|
|
213
|
+
signals.push({
|
|
214
|
+
bindingInfo: {
|
|
215
|
+
isRef: true,
|
|
216
|
+
name: node.name,
|
|
217
|
+
trackKinds: [1 /* TrackKind.AccessAnyValue */],
|
|
218
|
+
},
|
|
219
|
+
});
|
|
220
|
+
}
|
|
221
|
+
// TODO: toRefs
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
}
|
|
225
|
+
else if (ts.isFunctionDeclaration(node)) {
|
|
226
|
+
if (node.name && node.body) {
|
|
227
|
+
signals.push({
|
|
228
|
+
bindingInfo: {
|
|
229
|
+
isRef: false,
|
|
230
|
+
name: node.name,
|
|
231
|
+
trackKinds: [2 /* TrackKind.Call */],
|
|
232
|
+
},
|
|
233
|
+
trackInfo: {
|
|
234
|
+
depsHandler: node.body,
|
|
235
|
+
needToUse: true,
|
|
236
|
+
},
|
|
237
|
+
sideEffectInfo: {
|
|
238
|
+
isEffect: false,
|
|
239
|
+
handler: node.body,
|
|
240
|
+
},
|
|
241
|
+
});
|
|
242
|
+
}
|
|
243
|
+
}
|
|
244
|
+
else if (ts.isVariableStatement(node)) {
|
|
245
|
+
for (const declaration of node.declarationList.declarations) {
|
|
246
|
+
const name = declaration.name;
|
|
247
|
+
const callback = declaration.initializer;
|
|
248
|
+
if (callback && ts.isIdentifier(name) && (ts.isArrowFunction(callback) || ts.isFunctionExpression(callback))) {
|
|
249
|
+
signals.push({
|
|
250
|
+
bindingInfo: {
|
|
251
|
+
isRef: false,
|
|
252
|
+
name: name,
|
|
253
|
+
trackKinds: [2 /* TrackKind.Call */],
|
|
254
|
+
},
|
|
255
|
+
trackInfo: {
|
|
256
|
+
depsHandler: callback,
|
|
257
|
+
needToUse: true,
|
|
258
|
+
},
|
|
259
|
+
sideEffectInfo: {
|
|
260
|
+
isEffect: false,
|
|
261
|
+
handler: callback,
|
|
262
|
+
},
|
|
263
|
+
});
|
|
264
|
+
}
|
|
265
|
+
}
|
|
266
|
+
}
|
|
267
|
+
else if (ts.isParameter(node)) {
|
|
268
|
+
if (node.type && node.type && ts.isTypeReferenceNode(node.type)) {
|
|
269
|
+
const typeName = node.type.typeName.getText(sourceFile);
|
|
270
|
+
if (typeName.endsWith('Ref')) {
|
|
271
|
+
signals.push({
|
|
272
|
+
bindingInfo: {
|
|
273
|
+
isRef: true,
|
|
274
|
+
name: node.name,
|
|
275
|
+
trackKinds: [0 /* TrackKind.AccessDotValue */],
|
|
276
|
+
},
|
|
277
|
+
});
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
}
|
|
281
|
+
else if (ts.isCallExpression(node) && ts.isIdentifier(node.expression)) {
|
|
282
|
+
const call = node;
|
|
283
|
+
const callName = node.expression.escapedText;
|
|
284
|
+
if ((callName === 'effect' || callName === 'watchEffect') && call.arguments.length) {
|
|
285
|
+
const callback = call.arguments[0];
|
|
286
|
+
if (ts.isArrowFunction(callback) || ts.isFunctionExpression(callback)) {
|
|
287
|
+
signals.push({
|
|
288
|
+
trackInfo: {
|
|
289
|
+
depsHandler: callback.body,
|
|
290
|
+
needToUse: true,
|
|
291
|
+
},
|
|
292
|
+
sideEffectInfo: {
|
|
293
|
+
isEffect: true,
|
|
294
|
+
handler: callback.body,
|
|
295
|
+
},
|
|
296
|
+
});
|
|
297
|
+
}
|
|
298
|
+
}
|
|
299
|
+
if (callName === 'watch' && call.arguments.length >= 2) {
|
|
300
|
+
const depsCallback = call.arguments[0];
|
|
301
|
+
const effectCallback = call.arguments[1];
|
|
302
|
+
if (ts.isArrowFunction(effectCallback) || ts.isFunctionExpression(effectCallback)) {
|
|
303
|
+
if (ts.isArrowFunction(depsCallback) || ts.isFunctionExpression(depsCallback)) {
|
|
304
|
+
signals.push({
|
|
305
|
+
trackInfo: {
|
|
306
|
+
depsHandler: depsCallback.body,
|
|
307
|
+
needToUse: true,
|
|
308
|
+
},
|
|
309
|
+
sideEffectInfo: {
|
|
310
|
+
isEffect: true,
|
|
311
|
+
handler: effectCallback.body,
|
|
312
|
+
},
|
|
313
|
+
});
|
|
314
|
+
}
|
|
315
|
+
else {
|
|
316
|
+
signals.push({
|
|
317
|
+
trackInfo: {
|
|
318
|
+
depsHandler: depsCallback,
|
|
319
|
+
needToUse: false,
|
|
320
|
+
},
|
|
321
|
+
sideEffectInfo: {
|
|
322
|
+
isEffect: true,
|
|
323
|
+
handler: effectCallback.body,
|
|
324
|
+
}
|
|
325
|
+
});
|
|
326
|
+
}
|
|
327
|
+
}
|
|
328
|
+
}
|
|
329
|
+
else if ((0, language_core_1.hyphenateAttr)(callName).startsWith('use-')) {
|
|
330
|
+
signals.push({
|
|
331
|
+
bindingInfo: ts.isVariableDeclaration(call.parent)
|
|
332
|
+
? {
|
|
333
|
+
isRef: true,
|
|
334
|
+
name: call.parent.name,
|
|
335
|
+
trackKinds: [1 /* TrackKind.AccessAnyValue */, 2 /* TrackKind.Call */],
|
|
336
|
+
}
|
|
337
|
+
: undefined,
|
|
338
|
+
trackInfo: {
|
|
339
|
+
depsHandler: call,
|
|
340
|
+
needToUse: false,
|
|
341
|
+
},
|
|
342
|
+
});
|
|
343
|
+
}
|
|
344
|
+
else if ((callName === 'computed' || (0, language_core_1.hyphenateAttr)(callName).endsWith('-computed')) && call.arguments.length) {
|
|
345
|
+
const arg = call.arguments[0];
|
|
346
|
+
if (ts.isArrowFunction(arg) || ts.isFunctionExpression(arg)) {
|
|
347
|
+
signals.push({
|
|
348
|
+
bindingInfo: ts.isVariableDeclaration(call.parent)
|
|
349
|
+
? {
|
|
350
|
+
isRef: true,
|
|
351
|
+
name: call.parent.name,
|
|
352
|
+
trackKinds: [0 /* TrackKind.AccessDotValue */],
|
|
353
|
+
}
|
|
354
|
+
: undefined,
|
|
355
|
+
trackInfo: {
|
|
356
|
+
depsHandler: arg.body,
|
|
357
|
+
needToUse: true,
|
|
358
|
+
},
|
|
359
|
+
sideEffectInfo: {
|
|
360
|
+
isEffect: true,
|
|
361
|
+
handler: arg.body,
|
|
362
|
+
},
|
|
363
|
+
});
|
|
364
|
+
}
|
|
365
|
+
else if (ts.isIdentifier(arg)) {
|
|
366
|
+
signals.push({
|
|
367
|
+
bindingInfo: ts.isVariableDeclaration(call.parent)
|
|
368
|
+
? {
|
|
369
|
+
isRef: true,
|
|
370
|
+
name: call.parent.name,
|
|
371
|
+
trackKinds: [0 /* TrackKind.AccessDotValue */],
|
|
372
|
+
}
|
|
373
|
+
: undefined,
|
|
374
|
+
trackInfo: {
|
|
375
|
+
depsHandler: arg,
|
|
376
|
+
needToUse: false,
|
|
377
|
+
},
|
|
378
|
+
});
|
|
379
|
+
}
|
|
380
|
+
else if (ts.isObjectLiteralExpression(arg)) {
|
|
381
|
+
for (const prop of arg.properties) {
|
|
382
|
+
if (prop.name?.getText(sourceFile) === 'get') {
|
|
383
|
+
if (ts.isPropertyAssignment(prop)) {
|
|
384
|
+
const callback = prop.initializer;
|
|
385
|
+
if (ts.isArrowFunction(callback) || ts.isFunctionExpression(callback)) {
|
|
386
|
+
signals.push({
|
|
387
|
+
bindingInfo: ts.isVariableDeclaration(call.parent)
|
|
388
|
+
? {
|
|
389
|
+
isRef: true,
|
|
390
|
+
name: call.parent.name,
|
|
391
|
+
trackKinds: [0 /* TrackKind.AccessDotValue */],
|
|
392
|
+
}
|
|
393
|
+
: undefined,
|
|
394
|
+
trackInfo: {
|
|
395
|
+
depsHandler: callback.body,
|
|
396
|
+
needToUse: true,
|
|
397
|
+
},
|
|
398
|
+
sideEffectInfo: {
|
|
399
|
+
isEffect: true,
|
|
400
|
+
handler: callback.body,
|
|
401
|
+
},
|
|
402
|
+
});
|
|
403
|
+
}
|
|
404
|
+
}
|
|
405
|
+
else if (ts.isMethodDeclaration(prop) && prop.body) {
|
|
406
|
+
signals.push({
|
|
407
|
+
bindingInfo: ts.isVariableDeclaration(call.parent)
|
|
408
|
+
? {
|
|
409
|
+
isRef: true,
|
|
410
|
+
name: call.parent.name,
|
|
411
|
+
trackKinds: [0 /* TrackKind.AccessDotValue */],
|
|
412
|
+
}
|
|
413
|
+
: undefined,
|
|
414
|
+
trackInfo: {
|
|
415
|
+
depsHandler: prop.body,
|
|
416
|
+
needToUse: true,
|
|
417
|
+
},
|
|
418
|
+
sideEffectInfo: {
|
|
419
|
+
isEffect: true,
|
|
420
|
+
handler: prop.body,
|
|
421
|
+
},
|
|
422
|
+
});
|
|
423
|
+
}
|
|
424
|
+
}
|
|
425
|
+
}
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
}
|
|
429
|
+
node.forEachChild(visit);
|
|
430
|
+
});
|
|
431
|
+
sourceFile.forEachChild(function visit(node) {
|
|
432
|
+
if (ts.isPropertyAccessExpression(node)) {
|
|
433
|
+
if (node.name.text === 'value') {
|
|
434
|
+
dotValueAccesses.add(node.expression.getEnd());
|
|
435
|
+
dotAnyAccesses.add(node.expression.getEnd());
|
|
436
|
+
}
|
|
437
|
+
else if (node.name.text !== '') {
|
|
438
|
+
dotAnyAccesses.add(node.expression.getEnd());
|
|
439
|
+
}
|
|
440
|
+
}
|
|
441
|
+
else if (ts.isElementAccessExpression(node)) {
|
|
442
|
+
dotAnyAccesses.add(node.expression.getEnd());
|
|
443
|
+
}
|
|
444
|
+
else if (ts.isCallExpression(node)) {
|
|
445
|
+
functionCalls.add(node.expression.getEnd());
|
|
446
|
+
}
|
|
447
|
+
node.forEachChild(visit);
|
|
448
|
+
});
|
|
449
|
+
return {
|
|
450
|
+
signals,
|
|
451
|
+
dotValueAccesses,
|
|
452
|
+
dotAnyAccesses,
|
|
453
|
+
functionCalls,
|
|
454
|
+
};
|
|
455
|
+
}
|
|
456
|
+
//# sourceMappingURL=reactionsAnalyze.js.map
|
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
+
exports.getLanguageService = getLanguageService;
|
|
4
|
+
let currentProjectVersion = -1;
|
|
5
|
+
let currentFileName = '';
|
|
6
|
+
let currentSnapshot;
|
|
7
|
+
let languageService;
|
|
8
|
+
const host = {
|
|
9
|
+
getProjectVersion: () => currentProjectVersion.toString(),
|
|
10
|
+
getScriptFileNames: () => [currentFileName],
|
|
11
|
+
getScriptVersion: () => currentProjectVersion.toString(),
|
|
12
|
+
getScriptSnapshot: fileName => fileName === currentFileName ? currentSnapshot : undefined,
|
|
13
|
+
getCompilationSettings: () => ({}),
|
|
14
|
+
getCurrentDirectory: () => '',
|
|
15
|
+
getDefaultLibFileName: () => '',
|
|
16
|
+
readFile: () => undefined,
|
|
17
|
+
fileExists: fileName => fileName === currentFileName,
|
|
18
|
+
};
|
|
19
|
+
// TODO: share with volar-service-typescript
|
|
20
|
+
function getLanguageService(ts, snapshot, languageId) {
|
|
21
|
+
if (currentSnapshot !== snapshot) {
|
|
22
|
+
currentSnapshot = snapshot;
|
|
23
|
+
currentFileName = '/tmp.' + (languageId === 'javascript' ? 'js' :
|
|
24
|
+
languageId === 'typescriptreact' ? 'tsx' :
|
|
25
|
+
languageId === 'javascriptreact' ? 'jsx' :
|
|
26
|
+
'ts');
|
|
27
|
+
currentProjectVersion++;
|
|
28
|
+
}
|
|
29
|
+
languageService ??= ts.createLanguageService(host);
|
|
30
|
+
return {
|
|
31
|
+
languageService,
|
|
32
|
+
fileName: currentFileName,
|
|
33
|
+
};
|
|
34
|
+
}
|
|
35
|
+
//# sourceMappingURL=reactionsAnalyzeLS.js.map
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@vue/language-server",
|
|
3
|
-
"version": "3.0.0-
|
|
3
|
+
"version": "3.0.0-beta.2",
|
|
4
4
|
"license": "MIT",
|
|
5
5
|
"files": [
|
|
6
6
|
"**/*.js",
|
|
@@ -16,17 +16,18 @@
|
|
|
16
16
|
"directory": "packages/language-server"
|
|
17
17
|
},
|
|
18
18
|
"dependencies": {
|
|
19
|
-
"@volar/language-
|
|
20
|
-
"@
|
|
21
|
-
"@
|
|
22
|
-
"@vue/
|
|
23
|
-
"@vue/language-service": "3.0.0-alpha.8",
|
|
24
|
-
"@vue/typescript-plugin": "3.0.0-alpha.8",
|
|
25
|
-
"vscode-languageserver-protocol": "^3.17.5",
|
|
19
|
+
"@volar/language-server": "2.4.14",
|
|
20
|
+
"@vue/language-core": "3.0.0-beta.2",
|
|
21
|
+
"@vue/language-service": "3.0.0-beta.2",
|
|
22
|
+
"@vue/typescript-plugin": "3.0.0-beta.2",
|
|
26
23
|
"vscode-uri": "^3.0.8"
|
|
27
24
|
},
|
|
25
|
+
"peerDependencies": {
|
|
26
|
+
"typescript": "*"
|
|
27
|
+
},
|
|
28
28
|
"devDependencies": {
|
|
29
|
-
"@typescript/server-harness": "latest"
|
|
29
|
+
"@typescript/server-harness": "latest",
|
|
30
|
+
"@volar/test-utils": "2.4.14"
|
|
30
31
|
},
|
|
31
|
-
"gitHead": "
|
|
32
|
+
"gitHead": "ea40288f6fceebb65346732b6de5859c300cf1ee"
|
|
32
33
|
}
|
package/lib/types.d.ts
DELETED
package/lib/types.js
DELETED
|
@@ -1,19 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
-
if (k2 === undefined) k2 = k;
|
|
4
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
-
}
|
|
8
|
-
Object.defineProperty(o, k2, desc);
|
|
9
|
-
}) : (function(o, m, k, k2) {
|
|
10
|
-
if (k2 === undefined) k2 = k;
|
|
11
|
-
o[k2] = m[k];
|
|
12
|
-
}));
|
|
13
|
-
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
-
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
-
};
|
|
16
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
-
__exportStar(require("@volar/language-server/lib/types"), exports);
|
|
18
|
-
__exportStar(require("@vue/language-service/lib/types"), exports);
|
|
19
|
-
//# sourceMappingURL=types.js.map
|
package/node.d.ts
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export {};
|
package/node.js
DELETED
|
@@ -1,149 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
3
|
-
const simpleProject_1 = require("@volar/language-server/lib/project/simpleProject");
|
|
4
|
-
const node_1 = require("@volar/language-server/node");
|
|
5
|
-
const language_core_1 = require("@vue/language-core");
|
|
6
|
-
const language_service_1 = require("@vue/language-service");
|
|
7
|
-
const vscode_uri_1 = require("vscode-uri");
|
|
8
|
-
const connection = (0, node_1.createConnection)();
|
|
9
|
-
const server = (0, node_1.createServer)(connection);
|
|
10
|
-
connection.listen();
|
|
11
|
-
connection.onInitialize(params => {
|
|
12
|
-
const options = params.initializationOptions;
|
|
13
|
-
if (!options.typescript?.tsdk) {
|
|
14
|
-
throw new Error('typescript.tsdk is required');
|
|
15
|
-
}
|
|
16
|
-
if (!options.typescript?.tsserverRequestCommand) {
|
|
17
|
-
connection.console.warn('typescript.tsserverRequestCommand is required since >= 3.0 for complete TS features');
|
|
18
|
-
}
|
|
19
|
-
const { typescript: ts } = (0, node_1.loadTsdkByPath)(options.typescript.tsdk, params.locale);
|
|
20
|
-
const tsconfigProjects = (0, language_service_1.createUriMap)();
|
|
21
|
-
const file2ProjectInfo = new Map();
|
|
22
|
-
server.fileWatcher.onDidChangeWatchedFiles(({ changes }) => {
|
|
23
|
-
for (const change of changes) {
|
|
24
|
-
const changeUri = vscode_uri_1.URI.parse(change.uri);
|
|
25
|
-
if (tsconfigProjects.has(changeUri)) {
|
|
26
|
-
tsconfigProjects.get(changeUri).dispose();
|
|
27
|
-
tsconfigProjects.delete(changeUri);
|
|
28
|
-
file2ProjectInfo.clear();
|
|
29
|
-
}
|
|
30
|
-
}
|
|
31
|
-
});
|
|
32
|
-
let simpleLs;
|
|
33
|
-
return server.initialize(params, {
|
|
34
|
-
setup() { },
|
|
35
|
-
async getLanguageService(uri) {
|
|
36
|
-
if (uri.scheme === 'file' && options.typescript.tsserverRequestCommand) {
|
|
37
|
-
const fileName = uri.fsPath.replace(/\\/g, '/');
|
|
38
|
-
let projectInfoPromise = file2ProjectInfo.get(fileName);
|
|
39
|
-
if (!projectInfoPromise) {
|
|
40
|
-
projectInfoPromise = sendTsRequest(ts.server.protocol.CommandTypes.ProjectInfo, {
|
|
41
|
-
file: fileName,
|
|
42
|
-
needFileNameList: false,
|
|
43
|
-
});
|
|
44
|
-
file2ProjectInfo.set(fileName, projectInfoPromise);
|
|
45
|
-
}
|
|
46
|
-
const projectInfo = await projectInfoPromise;
|
|
47
|
-
if (projectInfo) {
|
|
48
|
-
const { configFileName } = projectInfo;
|
|
49
|
-
let ls = tsconfigProjects.get(vscode_uri_1.URI.file(configFileName));
|
|
50
|
-
if (!ls) {
|
|
51
|
-
ls = createLs(server, configFileName);
|
|
52
|
-
tsconfigProjects.set(vscode_uri_1.URI.file(configFileName), ls);
|
|
53
|
-
}
|
|
54
|
-
return ls;
|
|
55
|
-
}
|
|
56
|
-
}
|
|
57
|
-
return simpleLs ??= createLs(server, undefined);
|
|
58
|
-
},
|
|
59
|
-
getExistingLanguageServices() {
|
|
60
|
-
return Promise.all([
|
|
61
|
-
...tsconfigProjects.values(),
|
|
62
|
-
simpleLs,
|
|
63
|
-
].filter(promise => !!promise));
|
|
64
|
-
},
|
|
65
|
-
reload() {
|
|
66
|
-
for (const ls of [
|
|
67
|
-
...tsconfigProjects.values(),
|
|
68
|
-
simpleLs,
|
|
69
|
-
]) {
|
|
70
|
-
ls?.dispose();
|
|
71
|
-
}
|
|
72
|
-
tsconfigProjects.clear();
|
|
73
|
-
simpleLs = undefined;
|
|
74
|
-
},
|
|
75
|
-
}, (0, language_service_1.getHybridModeLanguageServicePlugins)(ts, options.typescript.tsserverRequestCommand ? {
|
|
76
|
-
collectExtractProps(...args) {
|
|
77
|
-
return sendTsRequest('vue:collectExtractProps', args);
|
|
78
|
-
},
|
|
79
|
-
getComponentDirectives(...args) {
|
|
80
|
-
return sendTsRequest('vue:getComponentDirectives', args);
|
|
81
|
-
},
|
|
82
|
-
getComponentEvents(...args) {
|
|
83
|
-
return sendTsRequest('vue:getComponentEvents', args);
|
|
84
|
-
},
|
|
85
|
-
getComponentNames(...args) {
|
|
86
|
-
return sendTsRequest('vue:getComponentNames', args);
|
|
87
|
-
},
|
|
88
|
-
getComponentProps(...args) {
|
|
89
|
-
return sendTsRequest('vue:getComponentProps', args);
|
|
90
|
-
},
|
|
91
|
-
getElementAttrs(...args) {
|
|
92
|
-
return sendTsRequest('vue:getElementAttrs', args);
|
|
93
|
-
},
|
|
94
|
-
getElementNames(...args) {
|
|
95
|
-
return sendTsRequest('vue:getElementNames', args);
|
|
96
|
-
},
|
|
97
|
-
getImportPathForFile(...args) {
|
|
98
|
-
return sendTsRequest('vue:getImportPathForFile', args);
|
|
99
|
-
},
|
|
100
|
-
getPropertiesAtLocation(...args) {
|
|
101
|
-
return sendTsRequest('vue:getPropertiesAtLocation', args);
|
|
102
|
-
},
|
|
103
|
-
getDocumentHighlights(fileName, position) {
|
|
104
|
-
return sendTsRequest('documentHighlights-full', // internal command
|
|
105
|
-
{
|
|
106
|
-
file: fileName,
|
|
107
|
-
...{ position },
|
|
108
|
-
filesToSearch: [fileName],
|
|
109
|
-
});
|
|
110
|
-
},
|
|
111
|
-
async getQuickInfoAtPosition(fileName, { line, character }) {
|
|
112
|
-
const result = await sendTsRequest(ts.server.protocol.CommandTypes.Quickinfo, {
|
|
113
|
-
file: fileName,
|
|
114
|
-
line: line + 1,
|
|
115
|
-
offset: character + 1,
|
|
116
|
-
});
|
|
117
|
-
return ts.displayPartsToString(result?.displayParts ?? []);
|
|
118
|
-
},
|
|
119
|
-
} : undefined));
|
|
120
|
-
function sendTsRequest(command, args) {
|
|
121
|
-
return connection.sendRequest(options.typescript.tsserverRequestCommand, [command, args]);
|
|
122
|
-
}
|
|
123
|
-
function createLs(server, tsconfig) {
|
|
124
|
-
const commonLine = tsconfig
|
|
125
|
-
? (0, language_core_1.createParsedCommandLine)(ts, ts.sys, tsconfig)
|
|
126
|
-
: {
|
|
127
|
-
options: ts.getDefaultCompilerOptions(),
|
|
128
|
-
vueOptions: (0, language_core_1.getDefaultCompilerOptions)(),
|
|
129
|
-
};
|
|
130
|
-
const language = (0, language_core_1.createLanguage)([
|
|
131
|
-
{
|
|
132
|
-
getLanguageId: uri => server.documents.get(uri)?.languageId,
|
|
133
|
-
},
|
|
134
|
-
(0, language_core_1.createVueLanguagePlugin)(ts, commonLine.options, commonLine.vueOptions, uri => uri.fsPath.replace(/\\/g, '/')),
|
|
135
|
-
], (0, language_service_1.createUriMap)(), uri => {
|
|
136
|
-
const document = server.documents.get(uri);
|
|
137
|
-
if (document) {
|
|
138
|
-
language.scripts.set(uri, document.getSnapshot(), document.languageId);
|
|
139
|
-
}
|
|
140
|
-
else {
|
|
141
|
-
language.scripts.delete(uri);
|
|
142
|
-
}
|
|
143
|
-
});
|
|
144
|
-
return (0, language_service_1.createLanguageService)(language, server.languageServicePlugins, (0, simpleProject_1.createLanguageServiceEnvironment)(server, [...server.workspaceFolders.all]), { vue: { compilerOptions: commonLine.vueOptions } });
|
|
145
|
-
}
|
|
146
|
-
});
|
|
147
|
-
connection.onInitialized(server.initialized);
|
|
148
|
-
connection.onShutdown(server.shutdown);
|
|
149
|
-
//# sourceMappingURL=node.js.map
|
package/protocol.d.ts
DELETED
|
@@ -1 +0,0 @@
|
|
|
1
|
-
export * from '@volar/language-server/protocol';
|
package/protocol.js
DELETED
|
@@ -1,18 +0,0 @@
|
|
|
1
|
-
"use strict";
|
|
2
|
-
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
3
|
-
if (k2 === undefined) k2 = k;
|
|
4
|
-
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
5
|
-
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
6
|
-
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
7
|
-
}
|
|
8
|
-
Object.defineProperty(o, k2, desc);
|
|
9
|
-
}) : (function(o, m, k, k2) {
|
|
10
|
-
if (k2 === undefined) k2 = k;
|
|
11
|
-
o[k2] = m[k];
|
|
12
|
-
}));
|
|
13
|
-
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
14
|
-
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
15
|
-
};
|
|
16
|
-
Object.defineProperty(exports, "__esModule", { value: true });
|
|
17
|
-
__exportStar(require("@volar/language-server/protocol"), exports);
|
|
18
|
-
//# sourceMappingURL=protocol.js.map
|