UNPKG

66.4 kBJavaScriptView Raw
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 "use strict";
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