1 | (function (factory) {
|
2 | if (typeof module === "object" && typeof module.exports === "object") {
|
3 | var v = factory(require, exports);
|
4 | if (v !== undefined) module.exports = v;
|
5 | }
|
6 | else if (typeof define === "function" && define.amd) {
|
7 | define("@angular/language-service/ivy/references", ["require", "exports", "tslib", "@angular/compiler", "@angular/compiler-cli/src/ngtsc/file_system", "@angular/compiler-cli/src/ngtsc/typecheck/api", "@angular/compiler-cli/src/ngtsc/typecheck/src/comments", "typescript", "@angular/language-service/ivy/template_target", "@angular/language-service/ivy/ts_utils", "@angular/language-service/ivy/utils"], factory);
|
8 | }
|
9 | })(function (require, exports) {
|
10 | ;
|
11 | Object.defineProperty(exports, "__esModule", { value: true });
|
12 | exports.ReferencesAndRenameBuilder = void 0;
|
13 | var tslib_1 = require("tslib");
|
14 | /**
|
15 | * @license
|
16 | * Copyright Google LLC All Rights Reserved.
|
17 | *
|
18 | * Use of this source code is governed by an MIT-style license that can be
|
19 | * found in the LICENSE file at https://angular.io/license
|
20 | */
|
21 | var compiler_1 = require("@angular/compiler");
|
22 | var file_system_1 = require("@angular/compiler-cli/src/ngtsc/file_system");
|
23 | var api_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/api");
|
24 | var comments_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/src/comments");
|
25 | var ts = require("typescript");
|
26 | var template_target_1 = require("@angular/language-service/ivy/template_target");
|
27 | var ts_utils_1 = require("@angular/language-service/ivy/ts_utils");
|
28 | var utils_1 = require("@angular/language-service/ivy/utils");
|
29 | function toFilePosition(shimLocation) {
|
30 | return { fileName: shimLocation.shimPath, position: shimLocation.positionInShimFile };
|
31 | }
|
32 | var RequestKind;
|
33 | (function (RequestKind) {
|
34 | RequestKind[RequestKind["Template"] = 0] = "Template";
|
35 | RequestKind[RequestKind["TypeScript"] = 1] = "TypeScript";
|
36 | })(RequestKind || (RequestKind = {}));
|
37 | var ReferencesAndRenameBuilder = /** @class */ (function () {
|
38 | function ReferencesAndRenameBuilder(strategy, tsLS, compiler) {
|
39 | this.strategy = strategy;
|
40 | this.tsLS = tsLS;
|
41 | this.compiler = compiler;
|
42 | this.ttc = this.compiler.getTemplateTypeChecker();
|
43 | }
|
44 | ReferencesAndRenameBuilder.prototype.getRenameInfo = function (filePath, position) {
|
45 | var templateInfo = utils_1.getTemplateInfoAtPosition(filePath, position, this.compiler);
|
46 | // We could not get a template at position so we assume the request came from outside the
|
47 | // template.
|
48 | if (templateInfo === undefined) {
|
49 | return this.tsLS.getRenameInfo(filePath, position);
|
50 | }
|
51 | var allTargetDetails = this.getTargetDetailsAtTemplatePosition(templateInfo, position);
|
52 | if (allTargetDetails === null) {
|
53 | return { canRename: false, localizedErrorMessage: 'Could not find template node at position.' };
|
54 | }
|
55 | var templateTarget = allTargetDetails[0].templateTarget;
|
56 | var templateTextAndSpan = getRenameTextAndSpanAtPosition(templateTarget, position);
|
57 | if (templateTextAndSpan === null) {
|
58 | return { canRename: false, localizedErrorMessage: 'Could not determine template node text.' };
|
59 | }
|
60 | var text = templateTextAndSpan.text, span = templateTextAndSpan.span;
|
61 | return {
|
62 | canRename: true,
|
63 | displayName: text,
|
64 | fullDisplayName: text,
|
65 | triggerSpan: span,
|
66 | };
|
67 | };
|
68 | ReferencesAndRenameBuilder.prototype.findRenameLocations = function (filePath, position) {
|
69 | this.ttc.generateAllTypeCheckBlocks();
|
70 | var templateInfo = utils_1.getTemplateInfoAtPosition(filePath, position, this.compiler);
|
71 | // We could not get a template at position so we assume the request came from outside the
|
72 | // template.
|
73 | if (templateInfo === undefined) {
|
74 | var requestNode = this.getTsNodeAtPosition(filePath, position);
|
75 | if (requestNode === null) {
|
76 | return undefined;
|
77 | }
|
78 | var requestOrigin = { kind: RequestKind.TypeScript, requestNode: requestNode };
|
79 | return this.findRenameLocationsAtTypescriptPosition(filePath, position, requestOrigin);
|
80 | }
|
81 | return this.findRenameLocationsAtTemplatePosition(templateInfo, position);
|
82 | };
|
83 | ReferencesAndRenameBuilder.prototype.findRenameLocationsAtTemplatePosition = function (templateInfo, position) {
|
84 | var e_1, _a, e_2, _b;
|
85 | var allTargetDetails = this.getTargetDetailsAtTemplatePosition(templateInfo, position);
|
86 | if (allTargetDetails === null) {
|
87 | return undefined;
|
88 | }
|
89 | var allRenameLocations = [];
|
90 | try {
|
91 | for (var allTargetDetails_1 = tslib_1.__values(allTargetDetails), allTargetDetails_1_1 = allTargetDetails_1.next(); !allTargetDetails_1_1.done; allTargetDetails_1_1 = allTargetDetails_1.next()) {
|
92 | var targetDetails = allTargetDetails_1_1.value;
|
93 | var requestOrigin = {
|
94 | kind: RequestKind.Template,
|
95 | requestNode: targetDetails.templateTarget,
|
96 | position: position,
|
97 | };
|
98 | try {
|
99 | for (var _c = (e_2 = void 0, tslib_1.__values(targetDetails.typescriptLocations)), _d = _c.next(); !_d.done; _d = _c.next()) {
|
100 | var location_1 = _d.value;
|
101 | var locations = this.findRenameLocationsAtTypescriptPosition(location_1.fileName, location_1.position, requestOrigin);
|
102 | // If we couldn't find rename locations for _any_ result, we should not allow renaming to
|
103 | // proceed instead of having a partially complete rename.
|
104 | if (locations === undefined) {
|
105 | return undefined;
|
106 | }
|
107 | allRenameLocations.push.apply(allRenameLocations, tslib_1.__spread(locations));
|
108 | }
|
109 | }
|
110 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
111 | finally {
|
112 | try {
|
113 | if (_d && !_d.done && (_b = _c.return)) _b.call(_c);
|
114 | }
|
115 | finally { if (e_2) throw e_2.error; }
|
116 | }
|
117 | }
|
118 | }
|
119 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
120 | finally {
|
121 | try {
|
122 | if (allTargetDetails_1_1 && !allTargetDetails_1_1.done && (_a = allTargetDetails_1.return)) _a.call(allTargetDetails_1);
|
123 | }
|
124 | finally { if (e_1) throw e_1.error; }
|
125 | }
|
126 | return allRenameLocations.length > 0 ? allRenameLocations : undefined;
|
127 | };
|
128 | ReferencesAndRenameBuilder.prototype.getTsNodeAtPosition = function (filePath, position) {
|
129 | var _a;
|
130 | var sf = this.strategy.getProgram().getSourceFile(filePath);
|
131 | if (!sf) {
|
132 | return null;
|
133 | }
|
134 | return (_a = ts_utils_1.findTightestNode(sf, position)) !== null && _a !== void 0 ? _a : null;
|
135 | };
|
136 | ReferencesAndRenameBuilder.prototype.findRenameLocationsAtTypescriptPosition = function (filePath, position, requestOrigin) {
|
137 | var e_3, _a;
|
138 | var originalNodeText;
|
139 | if (requestOrigin.kind === RequestKind.TypeScript) {
|
140 | originalNodeText = requestOrigin.requestNode.getText();
|
141 | }
|
142 | else {
|
143 | var templateNodeText = getRenameTextAndSpanAtPosition(requestOrigin.requestNode, requestOrigin.position);
|
144 | if (templateNodeText === null) {
|
145 | return undefined;
|
146 | }
|
147 | originalNodeText = templateNodeText.text;
|
148 | }
|
149 | var locations = this.tsLS.findRenameLocations(filePath, position, /*findInStrings*/ false, /*findInComments*/ false);
|
150 | if (locations === undefined) {
|
151 | return undefined;
|
152 | }
|
153 | var entries = new Map();
|
154 | try {
|
155 | for (var locations_1 = tslib_1.__values(locations), locations_1_1 = locations_1.next(); !locations_1_1.done; locations_1_1 = locations_1.next()) {
|
156 | var location_2 = locations_1_1.value;
|
157 | // TODO(atscott): Determine if a file is a shim file in a more robust way and make the API
|
158 | // available in an appropriate location.
|
159 | if (this.ttc.isTrackedTypeCheckFile(file_system_1.absoluteFrom(location_2.fileName))) {
|
160 | var entry = this.convertToTemplateDocumentSpan(location_2, this.ttc, originalNodeText);
|
161 | // There is no template node whose text matches the original rename request. Bail on
|
162 | // renaming completely rather than providing incomplete results.
|
163 | if (entry === null) {
|
164 | return undefined;
|
165 | }
|
166 | entries.set(createLocationKey(entry), entry);
|
167 | }
|
168 | else {
|
169 | // Ensure we only allow renaming a TS result with matching text
|
170 | var refNode = this.getTsNodeAtPosition(location_2.fileName, location_2.textSpan.start);
|
171 | if (refNode === null || refNode.getText() !== originalNodeText) {
|
172 | return undefined;
|
173 | }
|
174 | entries.set(createLocationKey(location_2), location_2);
|
175 | }
|
176 | }
|
177 | }
|
178 | catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
179 | finally {
|
180 | try {
|
181 | if (locations_1_1 && !locations_1_1.done && (_a = locations_1.return)) _a.call(locations_1);
|
182 | }
|
183 | finally { if (e_3) throw e_3.error; }
|
184 | }
|
185 | return Array.from(entries.values());
|
186 | };
|
187 | ReferencesAndRenameBuilder.prototype.getReferencesAtPosition = function (filePath, position) {
|
188 | this.ttc.generateAllTypeCheckBlocks();
|
189 | var templateInfo = utils_1.getTemplateInfoAtPosition(filePath, position, this.compiler);
|
190 | if (templateInfo === undefined) {
|
191 | return this.getReferencesAtTypescriptPosition(filePath, position);
|
192 | }
|
193 | return this.getReferencesAtTemplatePosition(templateInfo, position);
|
194 | };
|
195 | ReferencesAndRenameBuilder.prototype.getReferencesAtTemplatePosition = function (templateInfo, position) {
|
196 | var e_4, _a, e_5, _b;
|
197 | var allTargetDetails = this.getTargetDetailsAtTemplatePosition(templateInfo, position);
|
198 | if (allTargetDetails === null) {
|
199 | return undefined;
|
200 | }
|
201 | var allReferences = [];
|
202 | try {
|
203 | for (var allTargetDetails_2 = tslib_1.__values(allTargetDetails), allTargetDetails_2_1 = allTargetDetails_2.next(); !allTargetDetails_2_1.done; allTargetDetails_2_1 = allTargetDetails_2.next()) {
|
204 | var targetDetails = allTargetDetails_2_1.value;
|
205 | try {
|
206 | for (var _c = (e_5 = void 0, tslib_1.__values(targetDetails.typescriptLocations)), _d = _c.next(); !_d.done; _d = _c.next()) {
|
207 | var location_3 = _d.value;
|
208 | var refs = this.getReferencesAtTypescriptPosition(location_3.fileName, location_3.position);
|
209 | if (refs !== undefined) {
|
210 | allReferences.push.apply(allReferences, tslib_1.__spread(refs));
|
211 | }
|
212 | }
|
213 | }
|
214 | catch (e_5_1) { e_5 = { error: e_5_1 }; }
|
215 | finally {
|
216 | try {
|
217 | if (_d && !_d.done && (_b = _c.return)) _b.call(_c);
|
218 | }
|
219 | finally { if (e_5) throw e_5.error; }
|
220 | }
|
221 | }
|
222 | }
|
223 | catch (e_4_1) { e_4 = { error: e_4_1 }; }
|
224 | finally {
|
225 | try {
|
226 | if (allTargetDetails_2_1 && !allTargetDetails_2_1.done && (_a = allTargetDetails_2.return)) _a.call(allTargetDetails_2);
|
227 | }
|
228 | finally { if (e_4) throw e_4.error; }
|
229 | }
|
230 | return allReferences.length > 0 ? allReferences : undefined;
|
231 | };
|
232 | ReferencesAndRenameBuilder.prototype.getTargetDetailsAtTemplatePosition = function (_a, position) {
|
233 | var e_6, _b;
|
234 | var template = _a.template, component = _a.component;
|
235 | // Find the AST node in the template at the position.
|
236 | var positionDetails = template_target_1.getTargetAtPosition(template, position);
|
237 | if (positionDetails === null) {
|
238 | return null;
|
239 | }
|
240 | var nodes = positionDetails.context.kind === template_target_1.TargetNodeKind.TwoWayBindingContext ?
|
241 | positionDetails.context.nodes :
|
242 | [positionDetails.context.node];
|
243 | var details = [];
|
244 | try {
|
245 | for (var nodes_1 = tslib_1.__values(nodes), nodes_1_1 = nodes_1.next(); !nodes_1_1.done; nodes_1_1 = nodes_1.next()) {
|
246 | var node = nodes_1_1.value;
|
247 | // Get the information about the TCB at the template position.
|
248 | var symbol = this.ttc.getSymbolOfNode(node, component);
|
249 | if (symbol === null) {
|
250 | continue;
|
251 | }
|
252 | var templateTarget = node;
|
253 | switch (symbol.kind) {
|
254 | case api_1.SymbolKind.Directive:
|
255 | case api_1.SymbolKind.Template:
|
256 | // References to elements, templates, and directives will be through template references
|
257 | // (#ref). They shouldn't be used directly for a Language Service reference request.
|
258 | break;
|
259 | case api_1.SymbolKind.Element: {
|
260 | var matches = utils_1.getDirectiveMatchesForElementTag(symbol.templateNode, symbol.directives);
|
261 | details.push({ typescriptLocations: this.getPositionsForDirectives(matches), templateTarget: templateTarget });
|
262 | break;
|
263 | }
|
264 | case api_1.SymbolKind.DomBinding: {
|
265 | // Dom bindings aren't currently type-checked (see `checkTypeOfDomBindings`) so they don't
|
266 | // have a shim location. This means we can't match dom bindings to their lib.dom
|
267 | // reference, but we can still see if they match to a directive.
|
268 | if (!(node instanceof compiler_1.TmplAstTextAttribute) && !(node instanceof compiler_1.TmplAstBoundAttribute)) {
|
269 | return null;
|
270 | }
|
271 | var directives = utils_1.getDirectiveMatchesForAttribute(node.name, symbol.host.templateNode, symbol.host.directives);
|
272 | details.push({
|
273 | typescriptLocations: this.getPositionsForDirectives(directives),
|
274 | templateTarget: templateTarget,
|
275 | });
|
276 | break;
|
277 | }
|
278 | case api_1.SymbolKind.Reference: {
|
279 | details.push({
|
280 | typescriptLocations: [toFilePosition(symbol.referenceVarLocation)],
|
281 | templateTarget: templateTarget,
|
282 | });
|
283 | break;
|
284 | }
|
285 | case api_1.SymbolKind.Variable: {
|
286 | if ((templateTarget instanceof compiler_1.TmplAstVariable)) {
|
287 | if (templateTarget.valueSpan !== undefined &&
|
288 | utils_1.isWithin(position, templateTarget.valueSpan)) {
|
289 | // In the valueSpan of the variable, we want to get the reference of the initializer.
|
290 | details.push({
|
291 | typescriptLocations: [toFilePosition(symbol.initializerLocation)],
|
292 | templateTarget: templateTarget,
|
293 | });
|
294 | }
|
295 | else if (utils_1.isWithin(position, templateTarget.keySpan)) {
|
296 | // In the keySpan of the variable, we want to get the reference of the local variable.
|
297 | details.push({
|
298 | typescriptLocations: [toFilePosition(symbol.localVarLocation)],
|
299 | templateTarget: templateTarget,
|
300 | });
|
301 | }
|
302 | }
|
303 | else {
|
304 | // If the templateNode is not the `TmplAstVariable`, it must be a usage of the
|
305 | // variable somewhere in the template.
|
306 | details.push({
|
307 | typescriptLocations: [toFilePosition(symbol.localVarLocation)],
|
308 | templateTarget: templateTarget,
|
309 | });
|
310 | }
|
311 | break;
|
312 | }
|
313 | case api_1.SymbolKind.Input:
|
314 | case api_1.SymbolKind.Output: {
|
315 | details.push({
|
316 | typescriptLocations: symbol.bindings.map(function (binding) { return toFilePosition(binding.shimLocation); }),
|
317 | templateTarget: templateTarget,
|
318 | });
|
319 | break;
|
320 | }
|
321 | case api_1.SymbolKind.Pipe:
|
322 | case api_1.SymbolKind.Expression: {
|
323 | details.push({ typescriptLocations: [toFilePosition(symbol.shimLocation)], templateTarget: templateTarget });
|
324 | break;
|
325 | }
|
326 | }
|
327 | }
|
328 | }
|
329 | catch (e_6_1) { e_6 = { error: e_6_1 }; }
|
330 | finally {
|
331 | try {
|
332 | if (nodes_1_1 && !nodes_1_1.done && (_b = nodes_1.return)) _b.call(nodes_1);
|
333 | }
|
334 | finally { if (e_6) throw e_6.error; }
|
335 | }
|
336 | return details.length > 0 ? details : null;
|
337 | };
|
338 | ReferencesAndRenameBuilder.prototype.getPositionsForDirectives = function (directives) {
|
339 | var e_7, _a;
|
340 | var allDirectives = [];
|
341 | try {
|
342 | for (var _b = tslib_1.__values(directives.values()), _c = _b.next(); !_c.done; _c = _b.next()) {
|
343 | var dir = _c.value;
|
344 | var dirClass = dir.tsSymbol.valueDeclaration;
|
345 | if (dirClass === undefined || !ts.isClassDeclaration(dirClass) ||
|
346 | dirClass.name === undefined) {
|
347 | continue;
|
348 | }
|
349 | var fileName = dirClass.getSourceFile().fileName;
|
350 | var position = dirClass.name.getStart();
|
351 | allDirectives.push({ fileName: fileName, position: position });
|
352 | }
|
353 | }
|
354 | catch (e_7_1) { e_7 = { error: e_7_1 }; }
|
355 | finally {
|
356 | try {
|
357 | if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
358 | }
|
359 | finally { if (e_7) throw e_7.error; }
|
360 | }
|
361 | return allDirectives;
|
362 | };
|
363 | ReferencesAndRenameBuilder.prototype.getReferencesAtTypescriptPosition = function (fileName, position) {
|
364 | var e_8, _a;
|
365 | var refs = this.tsLS.getReferencesAtPosition(fileName, position);
|
366 | if (refs === undefined) {
|
367 | return undefined;
|
368 | }
|
369 | var entries = new Map();
|
370 | try {
|
371 | for (var refs_1 = tslib_1.__values(refs), refs_1_1 = refs_1.next(); !refs_1_1.done; refs_1_1 = refs_1.next()) {
|
372 | var ref = refs_1_1.value;
|
373 | if (this.ttc.isTrackedTypeCheckFile(file_system_1.absoluteFrom(ref.fileName))) {
|
374 | var entry = this.convertToTemplateDocumentSpan(ref, this.ttc);
|
375 | if (entry !== null) {
|
376 | entries.set(createLocationKey(entry), entry);
|
377 | }
|
378 | }
|
379 | else {
|
380 | entries.set(createLocationKey(ref), ref);
|
381 | }
|
382 | }
|
383 | }
|
384 | catch (e_8_1) { e_8 = { error: e_8_1 }; }
|
385 | finally {
|
386 | try {
|
387 | if (refs_1_1 && !refs_1_1.done && (_a = refs_1.return)) _a.call(refs_1);
|
388 | }
|
389 | finally { if (e_8) throw e_8.error; }
|
390 | }
|
391 | return Array.from(entries.values());
|
392 | };
|
393 | ReferencesAndRenameBuilder.prototype.convertToTemplateDocumentSpan = function (shimDocumentSpan, templateTypeChecker, requiredNodeText) {
|
394 | var sf = this.strategy.getProgram().getSourceFile(shimDocumentSpan.fileName);
|
395 | if (sf === undefined) {
|
396 | return null;
|
397 | }
|
398 | var tcbNode = ts_utils_1.findTightestNode(sf, shimDocumentSpan.textSpan.start);
|
399 | if (tcbNode === undefined ||
|
400 | comments_1.hasExpressionIdentifier(sf, tcbNode, comments_1.ExpressionIdentifier.EVENT_PARAMETER)) {
|
401 | // If the reference result is the $event parameter in the subscribe/addEventListener
|
402 | // function in the TCB, we want to filter this result out of the references. We really only
|
403 | // want to return references to the parameter in the template itself.
|
404 | return null;
|
405 | }
|
406 | // TODO(atscott): Determine how to consistently resolve paths. i.e. with the project
|
407 | // serverHost or LSParseConfigHost in the adapter. We should have a better defined way to
|
408 | // normalize paths.
|
409 | var mapping = utils_1.getTemplateLocationFromShimLocation(templateTypeChecker, file_system_1.absoluteFrom(shimDocumentSpan.fileName), shimDocumentSpan.textSpan.start);
|
410 | if (mapping === null) {
|
411 | return null;
|
412 | }
|
413 | var span = mapping.span, templateUrl = mapping.templateUrl;
|
414 | if (requiredNodeText !== undefined && span.toString() !== requiredNodeText) {
|
415 | return null;
|
416 | }
|
417 | return tslib_1.__assign(tslib_1.__assign({}, shimDocumentSpan), { fileName: templateUrl, textSpan: utils_1.toTextSpan(span) });
|
418 | };
|
419 | return ReferencesAndRenameBuilder;
|
420 | }());
|
421 | exports.ReferencesAndRenameBuilder = ReferencesAndRenameBuilder;
|
422 | function getRenameTextAndSpanAtPosition(node, position) {
|
423 | if (node instanceof compiler_1.TmplAstBoundAttribute || node instanceof compiler_1.TmplAstTextAttribute ||
|
424 | node instanceof compiler_1.TmplAstBoundEvent) {
|
425 | if (node.keySpan === undefined) {
|
426 | return null;
|
427 | }
|
428 | return { text: node.name, span: utils_1.toTextSpan(node.keySpan) };
|
429 | }
|
430 | else if (node instanceof compiler_1.TmplAstVariable || node instanceof compiler_1.TmplAstReference) {
|
431 | if (utils_1.isWithin(position, node.keySpan)) {
|
432 | return { text: node.keySpan.toString(), span: utils_1.toTextSpan(node.keySpan) };
|
433 | }
|
434 | else if (node.valueSpan && utils_1.isWithin(position, node.valueSpan)) {
|
435 | return { text: node.valueSpan.toString(), span: utils_1.toTextSpan(node.valueSpan) };
|
436 | }
|
437 | }
|
438 | if (node instanceof compiler_1.BindingPipe) {
|
439 | // TODO(atscott): Add support for renaming pipes
|
440 | return null;
|
441 | }
|
442 | if (node instanceof compiler_1.PropertyRead || node instanceof compiler_1.MethodCall || node instanceof compiler_1.PropertyWrite ||
|
443 | node instanceof compiler_1.SafePropertyRead || node instanceof compiler_1.SafeMethodCall) {
|
444 | return { text: node.name, span: utils_1.toTextSpan(node.nameSpan) };
|
445 | }
|
446 | else if (node instanceof compiler_1.LiteralPrimitive) {
|
447 | var span = utils_1.toTextSpan(node.sourceSpan);
|
448 | var text = node.value;
|
449 | if (typeof text === 'string') {
|
450 | // The span of a string literal includes the quotes but they should be removed for renaming.
|
451 | span.start += 1;
|
452 | span.length -= 2;
|
453 | }
|
454 | return { text: text, span: span };
|
455 | }
|
456 | return null;
|
457 | }
|
458 | /**
|
459 | * Creates a "key" for a rename/reference location by concatenating file name, span start, and span
|
460 | * length. This allows us to de-duplicate template results when an item may appear several times
|
461 | * in the TCB but map back to the same template location.
|
462 | */
|
463 | function createLocationKey(ds) {
|
464 | return ds.fileName + ds.textSpan.start + ds.textSpan.length;
|
465 | }
|
466 | });
|
467 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"references.js","sourceRoot":"","sources":["../../../../../../packages/language-service/ivy/references.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\nimport {AbsoluteSourceSpan, AST, BindingPipe, LiteralPrimitive, MethodCall, ParseSourceSpan, PropertyRead, PropertyWrite, SafeMethodCall, SafePropertyRead, TmplAstBoundAttribute, TmplAstBoundEvent, TmplAstNode, TmplAstReference, TmplAstTextAttribute, TmplAstVariable} from '@angular/compiler';\nimport {NgCompiler} from '@angular/compiler-cli/src/ngtsc/core';\nimport {absoluteFrom, absoluteFromSourceFile, AbsoluteFsPath} from '@angular/compiler-cli/src/ngtsc/file_system';\nimport {DirectiveSymbol, ShimLocation, SymbolKind, TemplateTypeChecker, TypeCheckingProgramStrategy} from '@angular/compiler-cli/src/ngtsc/typecheck/api';\nimport {ExpressionIdentifier, hasExpressionIdentifier} from '@angular/compiler-cli/src/ngtsc/typecheck/src/comments';\nimport * as ts from 'typescript';\n\nimport {getTargetAtPosition, TargetNodeKind} from './template_target';\nimport {findTightestNode} from './ts_utils';\nimport {getDirectiveMatchesForAttribute, getDirectiveMatchesForElementTag, getTemplateInfoAtPosition, getTemplateLocationFromShimLocation, isWithin, TemplateInfo, toTextSpan} from './utils';\n\ninterface FilePosition {\n  fileName: string;\n  position: number;\n}\n\nfunction toFilePosition(shimLocation: ShimLocation): FilePosition {\n  return {fileName: shimLocation.shimPath, position: shimLocation.positionInShimFile};\n}\n\nenum RequestKind {\n  Template,\n  TypeScript,\n}\n\ninterface TemplateRequest {\n  kind: RequestKind.Template;\n  requestNode: TmplAstNode|AST;\n  position: number;\n}\n\ninterface TypeScriptRequest {\n  kind: RequestKind.TypeScript;\n  requestNode: ts.Node;\n}\n\ntype RequestOrigin = TemplateRequest|TypeScriptRequest;\n\ninterface TemplateLocationDetails {\n  /**\n   * A target node in a template.\n   */\n  templateTarget: TmplAstNode|AST;\n\n  /**\n   * TypeScript locations which the template node maps to. A given template node might map to\n   * several TS nodes. For example, a template node for an attribute might resolve to several\n   * directives or a directive and one of its inputs.\n   */\n  typescriptLocations: FilePosition[];\n}\n\nexport class ReferencesAndRenameBuilder {\n  private readonly ttc = this.compiler.getTemplateTypeChecker();\n\n  constructor(\n      private readonly strategy: TypeCheckingProgramStrategy,\n      private readonly tsLS: ts.LanguageService, private readonly compiler: NgCompiler) {}\n\n  getRenameInfo(filePath: string, position: number):\n      Omit<ts.RenameInfoSuccess, 'kind'|'kindModifiers'>|ts.RenameInfoFailure {\n    const templateInfo = getTemplateInfoAtPosition(filePath, position, this.compiler);\n    // We could not get a template at position so we assume the request came from outside the\n    // template.\n    if (templateInfo === undefined) {\n      return this.tsLS.getRenameInfo(filePath, position);\n    }\n\n    const allTargetDetails = this.getTargetDetailsAtTemplatePosition(templateInfo, position);\n    if (allTargetDetails === null) {\n      return {canRename: false, localizedErrorMessage: 'Could not find template node at position.'};\n    }\n    const {templateTarget} = allTargetDetails[0];\n    const templateTextAndSpan = getRenameTextAndSpanAtPosition(templateTarget, position);\n    if (templateTextAndSpan === null) {\n      return {canRename: false, localizedErrorMessage: 'Could not determine template node text.'};\n    }\n    const {text, span} = templateTextAndSpan;\n    return {\n      canRename: true,\n      displayName: text,\n      fullDisplayName: text,\n      triggerSpan: span,\n    };\n  }\n\n  findRenameLocations(filePath: string, position: number): readonly ts.RenameLocation[]|undefined {\n    this.ttc.generateAllTypeCheckBlocks();\n    const templateInfo = getTemplateInfoAtPosition(filePath, position, this.compiler);\n    // We could not get a template at position so we assume the request came from outside the\n    // template.\n    if (templateInfo === undefined) {\n      const requestNode = this.getTsNodeAtPosition(filePath, position);\n      if (requestNode === null) {\n        return undefined;\n      }\n      const requestOrigin: TypeScriptRequest = {kind: RequestKind.TypeScript, requestNode};\n      return this.findRenameLocationsAtTypescriptPosition(filePath, position, requestOrigin);\n    }\n\n    return this.findRenameLocationsAtTemplatePosition(templateInfo, position);\n  }\n\n  private findRenameLocationsAtTemplatePosition(templateInfo: TemplateInfo, position: number):\n      readonly ts.RenameLocation[]|undefined {\n    const allTargetDetails = this.getTargetDetailsAtTemplatePosition(templateInfo, position);\n    if (allTargetDetails === null) {\n      return undefined;\n    }\n\n    const allRenameLocations: ts.RenameLocation[] = [];\n    for (const targetDetails of allTargetDetails) {\n      const requestOrigin: TemplateRequest = {\n        kind: RequestKind.Template,\n        requestNode: targetDetails.templateTarget,\n        position,\n      };\n\n      for (const location of targetDetails.typescriptLocations) {\n        const locations = this.findRenameLocationsAtTypescriptPosition(\n            location.fileName, location.position, requestOrigin);\n        // If we couldn't find rename locations for _any_ result, we should not allow renaming to\n        // proceed instead of having a partially complete rename.\n        if (locations === undefined) {\n          return undefined;\n        }\n        allRenameLocations.push(...locations);\n      }\n    }\n    return allRenameLocations.length > 0 ? allRenameLocations : undefined;\n  }\n\n  private getTsNodeAtPosition(filePath: string, position: number): ts.Node|null {\n    const sf = this.strategy.getProgram().getSourceFile(filePath);\n    if (!sf) {\n      return null;\n    }\n    return findTightestNode(sf, position) ?? null;\n  }\n\n  findRenameLocationsAtTypescriptPosition(\n      filePath: string, position: number,\n      requestOrigin: RequestOrigin): readonly ts.RenameLocation[]|undefined {\n    let originalNodeText: string;\n    if (requestOrigin.kind === RequestKind.TypeScript) {\n      originalNodeText = requestOrigin.requestNode.getText();\n    } else {\n      const templateNodeText =\n          getRenameTextAndSpanAtPosition(requestOrigin.requestNode, requestOrigin.position);\n      if (templateNodeText === null) {\n        return undefined;\n      }\n      originalNodeText = templateNodeText.text;\n    }\n\n    const locations = this.tsLS.findRenameLocations(\n        filePath, position, /*findInStrings*/ false, /*findInComments*/ false);\n    if (locations === undefined) {\n      return undefined;\n    }\n\n    const entries: Map<string, ts.RenameLocation> = new Map();\n    for (const location of locations) {\n      // TODO(atscott): Determine if a file is a shim file in a more robust way and make the API\n      // available in an appropriate location.\n      if (this.ttc.isTrackedTypeCheckFile(absoluteFrom(location.fileName))) {\n        const entry = this.convertToTemplateDocumentSpan(location, this.ttc, originalNodeText);\n        // There is no template node whose text matches the original rename request. Bail on\n        // renaming completely rather than providing incomplete results.\n        if (entry === null) {\n          return undefined;\n        }\n        entries.set(createLocationKey(entry), entry);\n      } else {\n        // Ensure we only allow renaming a TS result with matching text\n        const refNode = this.getTsNodeAtPosition(location.fileName, location.textSpan.start);\n        if (refNode === null || refNode.getText() !== originalNodeText) {\n          return undefined;\n        }\n        entries.set(createLocationKey(location), location);\n      }\n    }\n    return Array.from(entries.values());\n  }\n\n  getReferencesAtPosition(filePath: string, position: number): ts.ReferenceEntry[]|undefined {\n    this.ttc.generateAllTypeCheckBlocks();\n    const templateInfo = getTemplateInfoAtPosition(filePath, position, this.compiler);\n    if (templateInfo === undefined) {\n      return this.getReferencesAtTypescriptPosition(filePath, position);\n    }\n    return this.getReferencesAtTemplatePosition(templateInfo, position);\n  }\n\n  private getReferencesAtTemplatePosition(templateInfo: TemplateInfo, position: number):\n      ts.ReferenceEntry[]|undefined {\n    const allTargetDetails = this.getTargetDetailsAtTemplatePosition(templateInfo, position);\n    if (allTargetDetails === null) {\n      return undefined;\n    }\n    const allReferences: ts.ReferenceEntry[] = [];\n    for (const targetDetails of allTargetDetails) {\n      for (const location of targetDetails.typescriptLocations) {\n        const refs = this.getReferencesAtTypescriptPosition(location.fileName, location.position);\n        if (refs !== undefined) {\n          allReferences.push(...refs);\n        }\n      }\n    }\n    return allReferences.length > 0 ? allReferences : undefined;\n  }\n\n  private getTargetDetailsAtTemplatePosition({template, component}: TemplateInfo, position: number):\n      TemplateLocationDetails[]|null {\n    // Find the AST node in the template at the position.\n    const positionDetails = getTargetAtPosition(template, position);\n    if (positionDetails === null) {\n      return null;\n    }\n\n    const nodes = positionDetails.context.kind === TargetNodeKind.TwoWayBindingContext ?\n        positionDetails.context.nodes :\n        [positionDetails.context.node];\n\n    const details: TemplateLocationDetails[] = [];\n\n    for (const node of nodes) {\n      // Get the information about the TCB at the template position.\n      const symbol = this.ttc.getSymbolOfNode(node, component);\n      if (symbol === null) {\n        continue;\n      }\n\n      const templateTarget = node;\n      switch (symbol.kind) {\n        case SymbolKind.Directive:\n        case SymbolKind.Template:\n          // References to elements, templates, and directives will be through template references\n          // (#ref). They shouldn't be used directly for a Language Service reference request.\n          break;\n        case SymbolKind.Element: {\n          const matches = getDirectiveMatchesForElementTag(symbol.templateNode, symbol.directives);\n          details.push(\n              {typescriptLocations: this.getPositionsForDirectives(matches), templateTarget});\n          break;\n        }\n        case SymbolKind.DomBinding: {\n          // Dom bindings aren't currently type-checked (see `checkTypeOfDomBindings`) so they don't\n          // have a shim location. This means we can't match dom bindings to their lib.dom\n          // reference, but we can still see if they match to a directive.\n          if (!(node instanceof TmplAstTextAttribute) && !(node instanceof TmplAstBoundAttribute)) {\n            return null;\n          }\n          const directives = getDirectiveMatchesForAttribute(\n              node.name, symbol.host.templateNode, symbol.host.directives);\n          details.push({\n            typescriptLocations: this.getPositionsForDirectives(directives),\n            templateTarget,\n          });\n          break;\n        }\n        case SymbolKind.Reference: {\n          details.push({\n            typescriptLocations: [toFilePosition(symbol.referenceVarLocation)],\n            templateTarget,\n          });\n          break;\n        }\n        case SymbolKind.Variable: {\n          if ((templateTarget instanceof TmplAstVariable)) {\n            if (templateTarget.valueSpan !== undefined &&\n                isWithin(position, templateTarget.valueSpan)) {\n              // In the valueSpan of the variable, we want to get the reference of the initializer.\n              details.push({\n                typescriptLocations: [toFilePosition(symbol.initializerLocation)],\n                templateTarget,\n              });\n            } else if (isWithin(position, templateTarget.keySpan)) {\n              // In the keySpan of the variable, we want to get the reference of the local variable.\n              details.push({\n                typescriptLocations: [toFilePosition(symbol.localVarLocation)],\n                templateTarget,\n              });\n            }\n          } else {\n            // If the templateNode is not the `TmplAstVariable`, it must be a usage of the\n            // variable somewhere in the template.\n            details.push({\n              typescriptLocations: [toFilePosition(symbol.localVarLocation)],\n              templateTarget,\n            });\n          }\n          break;\n        }\n        case SymbolKind.Input:\n        case SymbolKind.Output: {\n          details.push({\n            typescriptLocations:\n                symbol.bindings.map(binding => toFilePosition(binding.shimLocation)),\n            templateTarget,\n          });\n          break;\n        }\n        case SymbolKind.Pipe:\n        case SymbolKind.Expression: {\n          details.push(\n              {typescriptLocations: [toFilePosition(symbol.shimLocation)], templateTarget});\n          break;\n        }\n      }\n    }\n\n    return details.length > 0 ? details : null;\n  }\n\n  private getPositionsForDirectives(directives: Set<DirectiveSymbol>): FilePosition[] {\n    const allDirectives: FilePosition[] = [];\n    for (const dir of directives.values()) {\n      const dirClass = dir.tsSymbol.valueDeclaration;\n      if (dirClass === undefined || !ts.isClassDeclaration(dirClass) ||\n          dirClass.name === undefined) {\n        continue;\n      }\n\n      const {fileName} = dirClass.getSourceFile();\n      const position = dirClass.name.getStart();\n      allDirectives.push({fileName, position});\n    }\n\n    return allDirectives;\n  }\n\n  private getReferencesAtTypescriptPosition(fileName: string, position: number):\n      ts.ReferenceEntry[]|undefined {\n    const refs = this.tsLS.getReferencesAtPosition(fileName, position);\n    if (refs === undefined) {\n      return undefined;\n    }\n\n    const entries: Map<string, ts.ReferenceEntry> = new Map();\n    for (const ref of refs) {\n      if (this.ttc.isTrackedTypeCheckFile(absoluteFrom(ref.fileName))) {\n        const entry = this.convertToTemplateDocumentSpan(ref, this.ttc);\n        if (entry !== null) {\n          entries.set(createLocationKey(entry), entry);\n        }\n      } else {\n        entries.set(createLocationKey(ref), ref);\n      }\n    }\n    return Array.from(entries.values());\n  }\n\n  private convertToTemplateDocumentSpan<T extends ts.DocumentSpan>(\n      shimDocumentSpan: T, templateTypeChecker: TemplateTypeChecker, requiredNodeText?: string): T\n      |null {\n    const sf = this.strategy.getProgram().getSourceFile(shimDocumentSpan.fileName);\n    if (sf === undefined) {\n      return null;\n    }\n    const tcbNode = findTightestNode(sf, shimDocumentSpan.textSpan.start);\n    if (tcbNode === undefined ||\n        hasExpressionIdentifier(sf, tcbNode, ExpressionIdentifier.EVENT_PARAMETER)) {\n      // If the reference result is the $event parameter in the subscribe/addEventListener\n      // function in the TCB, we want to filter this result out of the references. We really only\n      // want to return references to the parameter in the template itself.\n      return null;\n    }\n    // TODO(atscott): Determine how to consistently resolve paths. i.e. with the project\n    // serverHost or LSParseConfigHost in the adapter. We should have a better defined way to\n    // normalize paths.\n    const mapping = getTemplateLocationFromShimLocation(\n        templateTypeChecker, absoluteFrom(shimDocumentSpan.fileName),\n        shimDocumentSpan.textSpan.start);\n    if (mapping === null) {\n      return null;\n    }\n\n    const {span, templateUrl} = mapping;\n    if (requiredNodeText !== undefined && span.toString() !== requiredNodeText) {\n      return null;\n    }\n\n    return {\n      ...shimDocumentSpan,\n      fileName: templateUrl,\n      textSpan: toTextSpan(span),\n    };\n  }\n}\n\nfunction getRenameTextAndSpanAtPosition(\n    node: TmplAstNode|AST, position: number): {text: string, span: ts.TextSpan}|null {\n  if (node instanceof TmplAstBoundAttribute || node instanceof TmplAstTextAttribute ||\n      node instanceof TmplAstBoundEvent) {\n    if (node.keySpan === undefined) {\n      return null;\n    }\n    return {text: node.name, span: toTextSpan(node.keySpan)};\n  } else if (node instanceof TmplAstVariable || node instanceof TmplAstReference) {\n    if (isWithin(position, node.keySpan)) {\n      return {text: node.keySpan.toString(), span: toTextSpan(node.keySpan)};\n    } else if (node.valueSpan && isWithin(position, node.valueSpan)) {\n      return {text: node.valueSpan.toString(), span: toTextSpan(node.valueSpan)};\n    }\n  }\n\n  if (node instanceof BindingPipe) {\n    // TODO(atscott): Add support for renaming pipes\n    return null;\n  }\n  if (node instanceof PropertyRead || node instanceof MethodCall || node instanceof PropertyWrite ||\n      node instanceof SafePropertyRead || node instanceof SafeMethodCall) {\n    return {text: node.name, span: toTextSpan(node.nameSpan)};\n  } else if (node instanceof LiteralPrimitive) {\n    const span = toTextSpan(node.sourceSpan);\n    const text = node.value;\n    if (typeof text === 'string') {\n      // The span of a string literal includes the quotes but they should be removed for renaming.\n      span.start += 1;\n      span.length -= 2;\n    }\n    return {text, span};\n  }\n\n  return null;\n}\n\n\n/**\n * Creates a \"key\" for a rename/reference location by concatenating file name, span start, and span\n * length. This allows us to de-duplicate template results when an item may appear several times\n * in the TCB but map back to the same template location.\n */\nfunction createLocationKey(ds: ts.DocumentSpan) {\n  return ds.fileName + ds.textSpan.start + ds.textSpan.length;\n}"]} |
\ | No newline at end of file |