1 | /**
|
2 | * @license
|
3 | * Copyright Google LLC All Rights Reserved.
|
4 | *
|
5 | * Use of this source code is governed by an MIT-style license that can be
|
6 | * found in the LICENSE file at https://angular.io/license
|
7 | */
|
8 | (function (factory) {
|
9 | if (typeof module === "object" && typeof module.exports === "object") {
|
10 | var v = factory(require, exports);
|
11 | if (v !== undefined) module.exports = v;
|
12 | }
|
13 | else if (typeof define === "function" && define.amd) {
|
14 | define("@angular/language-service/ivy/definitions", ["require", "exports", "tslib", "@angular/compiler", "@angular/compiler-cli/src/ngtsc/metadata", "@angular/compiler-cli/src/ngtsc/typecheck/api", "typescript", "@angular/language-service/ivy/template_target", "@angular/language-service/ivy/ts_utils", "@angular/language-service/ivy/utils"], factory);
|
15 | }
|
16 | })(function (require, exports) {
|
17 | ;
|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
19 | exports.DefinitionBuilder = void 0;
|
20 | var tslib_1 = require("tslib");
|
21 | var compiler_1 = require("@angular/compiler");
|
22 | var metadata_1 = require("@angular/compiler-cli/src/ngtsc/metadata");
|
23 | var api_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/api");
|
24 | var ts = require("typescript");
|
25 | var template_target_1 = require("@angular/language-service/ivy/template_target");
|
26 | var ts_utils_1 = require("@angular/language-service/ivy/ts_utils");
|
27 | var utils_1 = require("@angular/language-service/ivy/utils");
|
28 | var DefinitionBuilder = /** @class */ (function () {
|
29 | function DefinitionBuilder(tsLS, compiler) {
|
30 | this.tsLS = tsLS;
|
31 | this.compiler = compiler;
|
32 | }
|
33 | DefinitionBuilder.prototype.getDefinitionAndBoundSpan = function (fileName, position) {
|
34 | var e_1, _a;
|
35 | var _b;
|
36 | var templateInfo = utils_1.getTemplateInfoAtPosition(fileName, position, this.compiler);
|
37 | if (templateInfo === undefined) {
|
38 | // We were unable to get a template at the given position. If we are in a TS file, instead
|
39 | // attempt to get an Angular definition at the location inside a TS file (examples of this
|
40 | // would be templateUrl or a url in styleUrls).
|
41 | if (!utils_1.isTypeScriptFile(fileName)) {
|
42 | return;
|
43 | }
|
44 | return getDefinitionForExpressionAtPosition(fileName, position, this.compiler);
|
45 | }
|
46 | var definitionMetas = this.getDefinitionMetaAtPosition(templateInfo, position);
|
47 | if (definitionMetas === undefined) {
|
48 | return undefined;
|
49 | }
|
50 | var definitions = [];
|
51 | try {
|
52 | for (var definitionMetas_1 = tslib_1.__values(definitionMetas), definitionMetas_1_1 = definitionMetas_1.next(); !definitionMetas_1_1.done; definitionMetas_1_1 = definitionMetas_1.next()) {
|
53 | var definitionMeta = definitionMetas_1_1.value;
|
54 | // The `$event` of event handlers would point to the $event parameter in the shim file, as in
|
55 | // `_t3["x"].subscribe(function ($event): any { $event }) ;`
|
56 | // If we wanted to return something for this, it would be more appropriate for something like
|
57 | // `getTypeDefinition`.
|
58 | if (utils_1.isDollarEvent(definitionMeta.node)) {
|
59 | continue;
|
60 | }
|
61 | definitions.push.apply(definitions, tslib_1.__spread(((_b = this.getDefinitionsForSymbol(tslib_1.__assign(tslib_1.__assign({}, definitionMeta), templateInfo))) !== null && _b !== void 0 ? _b : [])));
|
62 | }
|
63 | }
|
64 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
65 | finally {
|
66 | try {
|
67 | if (definitionMetas_1_1 && !definitionMetas_1_1.done && (_a = definitionMetas_1.return)) _a.call(definitionMetas_1);
|
68 | }
|
69 | finally { if (e_1) throw e_1.error; }
|
70 | }
|
71 | if (definitions.length === 0) {
|
72 | return undefined;
|
73 | }
|
74 | return { definitions: definitions, textSpan: utils_1.getTextSpanOfNode(definitionMetas[0].node) };
|
75 | };
|
76 | DefinitionBuilder.prototype.getDefinitionsForSymbol = function (_a) {
|
77 | var symbol = _a.symbol, node = _a.node, parent = _a.parent, component = _a.component;
|
78 | switch (symbol.kind) {
|
79 | case api_1.SymbolKind.Directive:
|
80 | case api_1.SymbolKind.Element:
|
81 | case api_1.SymbolKind.Template:
|
82 | case api_1.SymbolKind.DomBinding:
|
83 | // Though it is generally more appropriate for the above symbol definitions to be
|
84 | // associated with "type definitions" since the location in the template is the
|
85 | // actual definition location, the better user experience would be to allow
|
86 | // LS users to "go to definition" on an item in the template that maps to a class and be
|
87 | // taken to the directive or HTML class.
|
88 | return this.getTypeDefinitionsForTemplateInstance(symbol, node);
|
89 | case api_1.SymbolKind.Pipe: {
|
90 | if (symbol.tsSymbol !== null) {
|
91 | return this.getDefinitionsForSymbols(symbol);
|
92 | }
|
93 | else {
|
94 | // If there is no `ts.Symbol` for the pipe transform, we want to return the
|
95 | // type definition (the pipe class).
|
96 | return this.getTypeDefinitionsForSymbols(symbol.classSymbol);
|
97 | }
|
98 | }
|
99 | case api_1.SymbolKind.Output:
|
100 | case api_1.SymbolKind.Input: {
|
101 | var bindingDefs = this.getDefinitionsForSymbols.apply(this, tslib_1.__spread(symbol.bindings));
|
102 | // Also attempt to get directive matches for the input name. If there is a directive that
|
103 | // has the input name as part of the selector, we want to return that as well.
|
104 | var directiveDefs = this.getDirectiveTypeDefsForBindingNode(node, parent, component);
|
105 | return tslib_1.__spread(bindingDefs, directiveDefs);
|
106 | }
|
107 | case api_1.SymbolKind.Variable:
|
108 | case api_1.SymbolKind.Reference: {
|
109 | var definitions = [];
|
110 | if (symbol.declaration !== node) {
|
111 | var shimLocation = symbol.kind === api_1.SymbolKind.Variable ? symbol.localVarLocation :
|
112 | symbol.referenceVarLocation;
|
113 | var mapping = utils_1.getTemplateLocationFromShimLocation(this.compiler.getTemplateTypeChecker(), shimLocation.shimPath, shimLocation.positionInShimFile);
|
114 | if (mapping !== null) {
|
115 | definitions.push({
|
116 | name: symbol.declaration.name,
|
117 | containerName: '',
|
118 | containerKind: ts.ScriptElementKind.unknown,
|
119 | kind: ts.ScriptElementKind.variableElement,
|
120 | textSpan: utils_1.getTextSpanOfNode(symbol.declaration),
|
121 | contextSpan: utils_1.toTextSpan(symbol.declaration.sourceSpan),
|
122 | fileName: mapping.templateUrl,
|
123 | });
|
124 | }
|
125 | }
|
126 | if (symbol.kind === api_1.SymbolKind.Variable) {
|
127 | definitions.push.apply(definitions, tslib_1.__spread(this.getDefinitionsForSymbols({ shimLocation: symbol.initializerLocation })));
|
128 | }
|
129 | return definitions;
|
130 | }
|
131 | case api_1.SymbolKind.Expression: {
|
132 | return this.getDefinitionsForSymbols(symbol);
|
133 | }
|
134 | }
|
135 | };
|
136 | DefinitionBuilder.prototype.getDefinitionsForSymbols = function () {
|
137 | var _this = this;
|
138 | var symbols = [];
|
139 | for (var _i = 0; _i < arguments.length; _i++) {
|
140 | symbols[_i] = arguments[_i];
|
141 | }
|
142 | return utils_1.flatMap(symbols, function (_a) {
|
143 | var _b;
|
144 | var shimLocation = _a.shimLocation;
|
145 | var shimPath = shimLocation.shimPath, positionInShimFile = shimLocation.positionInShimFile;
|
146 | return (_b = _this.tsLS.getDefinitionAtPosition(shimPath, positionInShimFile)) !== null && _b !== void 0 ? _b : [];
|
147 | });
|
148 | };
|
149 | DefinitionBuilder.prototype.getTypeDefinitionsAtPosition = function (fileName, position) {
|
150 | var e_2, _a;
|
151 | var templateInfo = utils_1.getTemplateInfoAtPosition(fileName, position, this.compiler);
|
152 | if (templateInfo === undefined) {
|
153 | return;
|
154 | }
|
155 | var definitionMetas = this.getDefinitionMetaAtPosition(templateInfo, position);
|
156 | if (definitionMetas === undefined) {
|
157 | return undefined;
|
158 | }
|
159 | var definitions = [];
|
160 | try {
|
161 | for (var definitionMetas_2 = tslib_1.__values(definitionMetas), definitionMetas_2_1 = definitionMetas_2.next(); !definitionMetas_2_1.done; definitionMetas_2_1 = definitionMetas_2.next()) {
|
162 | var _b = definitionMetas_2_1.value, symbol = _b.symbol, node = _b.node, parent_1 = _b.parent;
|
163 | switch (symbol.kind) {
|
164 | case api_1.SymbolKind.Directive:
|
165 | case api_1.SymbolKind.DomBinding:
|
166 | case api_1.SymbolKind.Element:
|
167 | case api_1.SymbolKind.Template:
|
168 | definitions.push.apply(definitions, tslib_1.__spread(this.getTypeDefinitionsForTemplateInstance(symbol, node)));
|
169 | break;
|
170 | case api_1.SymbolKind.Output:
|
171 | case api_1.SymbolKind.Input: {
|
172 | var bindingDefs = this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(symbol.bindings));
|
173 | definitions.push.apply(definitions, tslib_1.__spread(bindingDefs));
|
174 | // Also attempt to get directive matches for the input name. If there is a directive that
|
175 | // has the input name as part of the selector, we want to return that as well.
|
176 | var directiveDefs = this.getDirectiveTypeDefsForBindingNode(node, parent_1, templateInfo.component);
|
177 | definitions.push.apply(definitions, tslib_1.__spread(directiveDefs));
|
178 | break;
|
179 | }
|
180 | case api_1.SymbolKind.Pipe: {
|
181 | if (symbol.tsSymbol !== null) {
|
182 | definitions.push.apply(definitions, tslib_1.__spread(this.getTypeDefinitionsForSymbols(symbol)));
|
183 | }
|
184 | else {
|
185 | // If there is no `ts.Symbol` for the pipe transform, we want to return the
|
186 | // type definition (the pipe class).
|
187 | definitions.push.apply(definitions, tslib_1.__spread(this.getTypeDefinitionsForSymbols(symbol.classSymbol)));
|
188 | }
|
189 | break;
|
190 | }
|
191 | case api_1.SymbolKind.Reference:
|
192 | definitions.push.apply(definitions, tslib_1.__spread(this.getTypeDefinitionsForSymbols({ shimLocation: symbol.targetLocation })));
|
193 | break;
|
194 | case api_1.SymbolKind.Expression:
|
195 | definitions.push.apply(definitions, tslib_1.__spread(this.getTypeDefinitionsForSymbols(symbol)));
|
196 | break;
|
197 | case api_1.SymbolKind.Variable: {
|
198 | definitions.push.apply(definitions, tslib_1.__spread(this.getTypeDefinitionsForSymbols({ shimLocation: symbol.initializerLocation })));
|
199 | break;
|
200 | }
|
201 | }
|
202 | return definitions;
|
203 | }
|
204 | }
|
205 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
206 | finally {
|
207 | try {
|
208 | if (definitionMetas_2_1 && !definitionMetas_2_1.done && (_a = definitionMetas_2.return)) _a.call(definitionMetas_2);
|
209 | }
|
210 | finally { if (e_2) throw e_2.error; }
|
211 | }
|
212 | };
|
213 | DefinitionBuilder.prototype.getTypeDefinitionsForTemplateInstance = function (symbol, node) {
|
214 | switch (symbol.kind) {
|
215 | case api_1.SymbolKind.Template: {
|
216 | var matches = utils_1.getDirectiveMatchesForElementTag(symbol.templateNode, symbol.directives);
|
217 | return this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(matches));
|
218 | }
|
219 | case api_1.SymbolKind.Element: {
|
220 | var matches = utils_1.getDirectiveMatchesForElementTag(symbol.templateNode, symbol.directives);
|
221 | // If one of the directive matches is a component, we should not include the native element
|
222 | // in the results because it is replaced by the component.
|
223 | return Array.from(matches).some(function (dir) { return dir.isComponent; }) ? this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(matches)) : this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(matches, [symbol]));
|
224 | }
|
225 | case api_1.SymbolKind.DomBinding: {
|
226 | if (!(node instanceof compiler_1.TmplAstTextAttribute)) {
|
227 | return [];
|
228 | }
|
229 | var dirs = utils_1.getDirectiveMatchesForAttribute(node.name, symbol.host.templateNode, symbol.host.directives);
|
230 | return this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(dirs));
|
231 | }
|
232 | case api_1.SymbolKind.Directive:
|
233 | return this.getTypeDefinitionsForSymbols(symbol);
|
234 | }
|
235 | };
|
236 | DefinitionBuilder.prototype.getDirectiveTypeDefsForBindingNode = function (node, parent, component) {
|
237 | if (!(node instanceof compiler_1.TmplAstBoundAttribute) && !(node instanceof compiler_1.TmplAstTextAttribute) &&
|
238 | !(node instanceof compiler_1.TmplAstBoundEvent)) {
|
239 | return [];
|
240 | }
|
241 | if (parent === null ||
|
242 | !(parent instanceof compiler_1.TmplAstTemplate || parent instanceof compiler_1.TmplAstElement)) {
|
243 | return [];
|
244 | }
|
245 | var templateOrElementSymbol = this.compiler.getTemplateTypeChecker().getSymbolOfNode(parent, component);
|
246 | if (templateOrElementSymbol === null ||
|
247 | (templateOrElementSymbol.kind !== api_1.SymbolKind.Template &&
|
248 | templateOrElementSymbol.kind !== api_1.SymbolKind.Element)) {
|
249 | return [];
|
250 | }
|
251 | var dirs = utils_1.getDirectiveMatchesForAttribute(node.name, parent, templateOrElementSymbol.directives);
|
252 | return this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(dirs));
|
253 | };
|
254 | DefinitionBuilder.prototype.getTypeDefinitionsForSymbols = function () {
|
255 | var _this = this;
|
256 | var symbols = [];
|
257 | for (var _i = 0; _i < arguments.length; _i++) {
|
258 | symbols[_i] = arguments[_i];
|
259 | }
|
260 | return utils_1.flatMap(symbols, function (_a) {
|
261 | var _b;
|
262 | var shimLocation = _a.shimLocation;
|
263 | var shimPath = shimLocation.shimPath, positionInShimFile = shimLocation.positionInShimFile;
|
264 | return (_b = _this.tsLS.getTypeDefinitionAtPosition(shimPath, positionInShimFile)) !== null && _b !== void 0 ? _b : [];
|
265 | });
|
266 | };
|
267 | DefinitionBuilder.prototype.getDefinitionMetaAtPosition = function (_a, position) {
|
268 | var e_3, _b;
|
269 | var template = _a.template, component = _a.component;
|
270 | var target = template_target_1.getTargetAtPosition(template, position);
|
271 | if (target === null) {
|
272 | return undefined;
|
273 | }
|
274 | var context = target.context, parent = target.parent;
|
275 | var nodes = context.kind === template_target_1.TargetNodeKind.TwoWayBindingContext ? context.nodes : [context.node];
|
276 | var definitionMetas = [];
|
277 | try {
|
278 | for (var nodes_1 = tslib_1.__values(nodes), nodes_1_1 = nodes_1.next(); !nodes_1_1.done; nodes_1_1 = nodes_1.next()) {
|
279 | var node = nodes_1_1.value;
|
280 | var symbol = this.compiler.getTemplateTypeChecker().getSymbolOfNode(node, component);
|
281 | if (symbol === null) {
|
282 | continue;
|
283 | }
|
284 | definitionMetas.push({ node: node, parent: parent, symbol: symbol });
|
285 | }
|
286 | }
|
287 | catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
288 | finally {
|
289 | try {
|
290 | if (nodes_1_1 && !nodes_1_1.done && (_b = nodes_1.return)) _b.call(nodes_1);
|
291 | }
|
292 | finally { if (e_3) throw e_3.error; }
|
293 | }
|
294 | return definitionMetas.length > 0 ? definitionMetas : undefined;
|
295 | };
|
296 | return DefinitionBuilder;
|
297 | }());
|
298 | exports.DefinitionBuilder = DefinitionBuilder;
|
299 | /**
|
300 | * Gets an Angular-specific definition in a TypeScript source file.
|
301 | */
|
302 | function getDefinitionForExpressionAtPosition(fileName, position, compiler) {
|
303 | var sf = compiler.getNextProgram().getSourceFile(fileName);
|
304 | if (sf === undefined) {
|
305 | return;
|
306 | }
|
307 | var expression = ts_utils_1.findTightestNode(sf, position);
|
308 | if (expression === undefined) {
|
309 | return;
|
310 | }
|
311 | var classDeclaration = ts_utils_1.getParentClassDeclaration(expression);
|
312 | if (classDeclaration === undefined) {
|
313 | return;
|
314 | }
|
315 | var componentResources = compiler.getComponentResources(classDeclaration);
|
316 | if (componentResources === null) {
|
317 | return;
|
318 | }
|
319 | var allResources = tslib_1.__spread(componentResources.styles, [componentResources.template]);
|
320 | var resourceForExpression = allResources.find(function (resource) { return resource.expression === expression; });
|
321 | if (resourceForExpression === undefined || !metadata_1.isExternalResource(resourceForExpression)) {
|
322 | return;
|
323 | }
|
324 | var templateDefinitions = [{
|
325 | kind: ts.ScriptElementKind.externalModuleName,
|
326 | name: resourceForExpression.path,
|
327 | containerKind: ts.ScriptElementKind.unknown,
|
328 | containerName: '',
|
329 | // Reading the template is expensive, so don't provide a preview.
|
330 | // TODO(ayazhafiz): Consider providing an actual span:
|
331 | // 1. We're likely to read the template anyway
|
332 | // 2. We could show just the first 100 chars or so
|
333 | textSpan: { start: 0, length: 0 },
|
334 | fileName: resourceForExpression.path,
|
335 | }];
|
336 | return {
|
337 | definitions: templateDefinitions,
|
338 | textSpan: {
|
339 | // Exclude opening and closing quotes in the url span.
|
340 | start: expression.getStart() + 1,
|
341 | length: expression.getWidth() - 2,
|
342 | },
|
343 | };
|
344 | }
|
345 | });
|
346 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"definitions.js","sourceRoot":"","sources":["../../../../../../packages/language-service/ivy/definitions.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 */\n\nimport {AST, TmplAstBoundAttribute, TmplAstBoundEvent, TmplAstElement, TmplAstNode, TmplAstTemplate, TmplAstTextAttribute} from '@angular/compiler';\nimport {NgCompiler} from '@angular/compiler-cli/src/ngtsc/core';\nimport {isExternalResource} from '@angular/compiler-cli/src/ngtsc/metadata';\nimport {DirectiveSymbol, DomBindingSymbol, ElementSymbol, ShimLocation, Symbol, SymbolKind, TemplateSymbol} from '@angular/compiler-cli/src/ngtsc/typecheck/api';\nimport * as ts from 'typescript';\n\nimport {getTargetAtPosition, TargetNodeKind} from './template_target';\nimport {findTightestNode, getParentClassDeclaration} from './ts_utils';\nimport {flatMap, getDirectiveMatchesForAttribute, getDirectiveMatchesForElementTag, getTemplateInfoAtPosition, getTemplateLocationFromShimLocation, getTextSpanOfNode, isDollarEvent, isTypeScriptFile, TemplateInfo, toTextSpan} from './utils';\n\ninterface DefinitionMeta {\n  node: AST|TmplAstNode;\n  parent: AST|TmplAstNode|null;\n  symbol: Symbol;\n}\n\ninterface HasShimLocation {\n  shimLocation: ShimLocation;\n}\n\nexport class DefinitionBuilder {\n  constructor(private readonly tsLS: ts.LanguageService, private readonly compiler: NgCompiler) {}\n\n  getDefinitionAndBoundSpan(fileName: string, position: number): ts.DefinitionInfoAndBoundSpan\n      |undefined {\n    const templateInfo = getTemplateInfoAtPosition(fileName, position, this.compiler);\n    if (templateInfo === undefined) {\n      // We were unable to get a template at the given position. If we are in a TS file, instead\n      // attempt to get an Angular definition at the location inside a TS file (examples of this\n      // would be templateUrl or a url in styleUrls).\n      if (!isTypeScriptFile(fileName)) {\n        return;\n      }\n      return getDefinitionForExpressionAtPosition(fileName, position, this.compiler);\n    }\n    const definitionMetas = this.getDefinitionMetaAtPosition(templateInfo, position);\n    if (definitionMetas === undefined) {\n      return undefined;\n    }\n    const definitions: ts.DefinitionInfo[] = [];\n    for (const definitionMeta of definitionMetas) {\n      // The `$event` of event handlers would point to the $event parameter in the shim file, as in\n      // `_t3[\"x\"].subscribe(function ($event): any { $event }) ;`\n      // If we wanted to return something for this, it would be more appropriate for something like\n      // `getTypeDefinition`.\n      if (isDollarEvent(definitionMeta.node)) {\n        continue;\n      }\n\n      definitions.push(\n          ...(this.getDefinitionsForSymbol({...definitionMeta, ...templateInfo}) ?? []));\n    }\n\n    if (definitions.length === 0) {\n      return undefined;\n    }\n\n    return {definitions, textSpan: getTextSpanOfNode(definitionMetas[0].node)};\n  }\n\n  private getDefinitionsForSymbol({symbol, node, parent, component}: DefinitionMeta&\n                                  TemplateInfo): readonly ts.DefinitionInfo[]|undefined {\n    switch (symbol.kind) {\n      case SymbolKind.Directive:\n      case SymbolKind.Element:\n      case SymbolKind.Template:\n      case SymbolKind.DomBinding:\n        // Though it is generally more appropriate for the above symbol definitions to be\n        // associated with \"type definitions\" since the location in the template is the\n        // actual definition location, the better user experience would be to allow\n        // LS users to \"go to definition\" on an item in the template that maps to a class and be\n        // taken to the directive or HTML class.\n        return this.getTypeDefinitionsForTemplateInstance(symbol, node);\n      case SymbolKind.Pipe: {\n        if (symbol.tsSymbol !== null) {\n          return this.getDefinitionsForSymbols(symbol);\n        } else {\n          // If there is no `ts.Symbol` for the pipe transform, we want to return the\n          // type definition (the pipe class).\n          return this.getTypeDefinitionsForSymbols(symbol.classSymbol);\n        }\n      }\n      case SymbolKind.Output:\n      case SymbolKind.Input: {\n        const bindingDefs = this.getDefinitionsForSymbols(...symbol.bindings);\n        // Also attempt to get directive matches for the input name. If there is a directive that\n        // has the input name as part of the selector, we want to return that as well.\n        const directiveDefs = this.getDirectiveTypeDefsForBindingNode(node, parent, component);\n        return [...bindingDefs, ...directiveDefs];\n      }\n      case SymbolKind.Variable:\n      case SymbolKind.Reference: {\n        const definitions: ts.DefinitionInfo[] = [];\n        if (symbol.declaration !== node) {\n          const shimLocation = symbol.kind === SymbolKind.Variable ? symbol.localVarLocation :\n                                                                     symbol.referenceVarLocation;\n          const mapping = getTemplateLocationFromShimLocation(\n              this.compiler.getTemplateTypeChecker(), shimLocation.shimPath,\n              shimLocation.positionInShimFile);\n          if (mapping !== null) {\n            definitions.push({\n              name: symbol.declaration.name,\n              containerName: '',\n              containerKind: ts.ScriptElementKind.unknown,\n              kind: ts.ScriptElementKind.variableElement,\n              textSpan: getTextSpanOfNode(symbol.declaration),\n              contextSpan: toTextSpan(symbol.declaration.sourceSpan),\n              fileName: mapping.templateUrl,\n            });\n          }\n        }\n        if (symbol.kind === SymbolKind.Variable) {\n          definitions.push(\n              ...this.getDefinitionsForSymbols({shimLocation: symbol.initializerLocation}));\n        }\n        return definitions;\n      }\n      case SymbolKind.Expression: {\n        return this.getDefinitionsForSymbols(symbol);\n      }\n    }\n  }\n\n  private getDefinitionsForSymbols(...symbols: HasShimLocation[]): ts.DefinitionInfo[] {\n    return flatMap(symbols, ({shimLocation}) => {\n      const {shimPath, positionInShimFile} = shimLocation;\n      return this.tsLS.getDefinitionAtPosition(shimPath, positionInShimFile) ?? [];\n    });\n  }\n\n  getTypeDefinitionsAtPosition(fileName: string, position: number):\n      readonly ts.DefinitionInfo[]|undefined {\n    const templateInfo = getTemplateInfoAtPosition(fileName, position, this.compiler);\n    if (templateInfo === undefined) {\n      return;\n    }\n    const definitionMetas = this.getDefinitionMetaAtPosition(templateInfo, position);\n    if (definitionMetas === undefined) {\n      return undefined;\n    }\n\n    const definitions: ts.DefinitionInfo[] = [];\n    for (const {symbol, node, parent} of definitionMetas) {\n      switch (symbol.kind) {\n        case SymbolKind.Directive:\n        case SymbolKind.DomBinding:\n        case SymbolKind.Element:\n        case SymbolKind.Template:\n          definitions.push(...this.getTypeDefinitionsForTemplateInstance(symbol, node));\n          break;\n        case SymbolKind.Output:\n        case SymbolKind.Input: {\n          const bindingDefs = this.getTypeDefinitionsForSymbols(...symbol.bindings);\n          definitions.push(...bindingDefs);\n          // Also attempt to get directive matches for the input name. If there is a directive that\n          // has the input name as part of the selector, we want to return that as well.\n          const directiveDefs =\n              this.getDirectiveTypeDefsForBindingNode(node, parent, templateInfo.component);\n          definitions.push(...directiveDefs);\n          break;\n        }\n        case SymbolKind.Pipe: {\n          if (symbol.tsSymbol !== null) {\n            definitions.push(...this.getTypeDefinitionsForSymbols(symbol));\n          } else {\n            // If there is no `ts.Symbol` for the pipe transform, we want to return the\n            // type definition (the pipe class).\n            definitions.push(...this.getTypeDefinitionsForSymbols(symbol.classSymbol));\n          }\n          break;\n        }\n        case SymbolKind.Reference:\n          definitions.push(\n              ...this.getTypeDefinitionsForSymbols({shimLocation: symbol.targetLocation}));\n          break;\n        case SymbolKind.Expression:\n          definitions.push(...this.getTypeDefinitionsForSymbols(symbol));\n          break;\n        case SymbolKind.Variable: {\n          definitions.push(\n              ...this.getTypeDefinitionsForSymbols({shimLocation: symbol.initializerLocation}));\n          break;\n        }\n      }\n      return definitions;\n    }\n  }\n\n  private getTypeDefinitionsForTemplateInstance(\n      symbol: TemplateSymbol|ElementSymbol|DomBindingSymbol|DirectiveSymbol,\n      node: AST|TmplAstNode): ts.DefinitionInfo[] {\n    switch (symbol.kind) {\n      case SymbolKind.Template: {\n        const matches = getDirectiveMatchesForElementTag(symbol.templateNode, symbol.directives);\n        return this.getTypeDefinitionsForSymbols(...matches);\n      }\n      case SymbolKind.Element: {\n        const matches = getDirectiveMatchesForElementTag(symbol.templateNode, symbol.directives);\n        // If one of the directive matches is a component, we should not include the native element\n        // in the results because it is replaced by the component.\n        return Array.from(matches).some(dir => dir.isComponent) ?\n            this.getTypeDefinitionsForSymbols(...matches) :\n            this.getTypeDefinitionsForSymbols(...matches, symbol);\n      }\n      case SymbolKind.DomBinding: {\n        if (!(node instanceof TmplAstTextAttribute)) {\n          return [];\n        }\n        const dirs = getDirectiveMatchesForAttribute(\n            node.name, symbol.host.templateNode, symbol.host.directives);\n        return this.getTypeDefinitionsForSymbols(...dirs);\n      }\n      case SymbolKind.Directive:\n        return this.getTypeDefinitionsForSymbols(symbol);\n    }\n  }\n\n  private getDirectiveTypeDefsForBindingNode(\n      node: TmplAstNode|AST, parent: TmplAstNode|AST|null, component: ts.ClassDeclaration) {\n    if (!(node instanceof TmplAstBoundAttribute) && !(node instanceof TmplAstTextAttribute) &&\n        !(node instanceof TmplAstBoundEvent)) {\n      return [];\n    }\n    if (parent === null ||\n        !(parent instanceof TmplAstTemplate || parent instanceof TmplAstElement)) {\n      return [];\n    }\n    const templateOrElementSymbol =\n        this.compiler.getTemplateTypeChecker().getSymbolOfNode(parent, component);\n    if (templateOrElementSymbol === null ||\n        (templateOrElementSymbol.kind !== SymbolKind.Template &&\n         templateOrElementSymbol.kind !== SymbolKind.Element)) {\n      return [];\n    }\n    const dirs =\n        getDirectiveMatchesForAttribute(node.name, parent, templateOrElementSymbol.directives);\n    return this.getTypeDefinitionsForSymbols(...dirs);\n  }\n\n  private getTypeDefinitionsForSymbols(...symbols: HasShimLocation[]): ts.DefinitionInfo[] {\n    return flatMap(symbols, ({shimLocation}) => {\n      const {shimPath, positionInShimFile} = shimLocation;\n      return this.tsLS.getTypeDefinitionAtPosition(shimPath, positionInShimFile) ?? [];\n    });\n  }\n\n  private getDefinitionMetaAtPosition({template, component}: TemplateInfo, position: number):\n      DefinitionMeta[]|undefined {\n    const target = getTargetAtPosition(template, position);\n    if (target === null) {\n      return undefined;\n    }\n    const {context, parent} = target;\n\n    const nodes =\n        context.kind === TargetNodeKind.TwoWayBindingContext ? context.nodes : [context.node];\n\n\n    const definitionMetas: DefinitionMeta[] = [];\n    for (const node of nodes) {\n      const symbol = this.compiler.getTemplateTypeChecker().getSymbolOfNode(node, component);\n      if (symbol === null) {\n        continue;\n      }\n      definitionMetas.push({node, parent, symbol});\n    }\n    return definitionMetas.length > 0 ? definitionMetas : undefined;\n  }\n}\n\n/**\n * Gets an Angular-specific definition in a TypeScript source file.\n */\nfunction getDefinitionForExpressionAtPosition(\n    fileName: string, position: number, compiler: NgCompiler): ts.DefinitionInfoAndBoundSpan|\n    undefined {\n  const sf = compiler.getNextProgram().getSourceFile(fileName);\n  if (sf === undefined) {\n    return;\n  }\n\n  const expression = findTightestNode(sf, position);\n  if (expression === undefined) {\n    return;\n  }\n  const classDeclaration = getParentClassDeclaration(expression);\n  if (classDeclaration === undefined) {\n    return;\n  }\n  const componentResources = compiler.getComponentResources(classDeclaration);\n  if (componentResources === null) {\n    return;\n  }\n\n  const allResources = [...componentResources.styles, componentResources.template];\n\n  const resourceForExpression = allResources.find(resource => resource.expression === expression);\n  if (resourceForExpression === undefined || !isExternalResource(resourceForExpression)) {\n    return;\n  }\n\n  const templateDefinitions: ts.DefinitionInfo[] = [{\n    kind: ts.ScriptElementKind.externalModuleName,\n    name: resourceForExpression.path,\n    containerKind: ts.ScriptElementKind.unknown,\n    containerName: '',\n    // Reading the template is expensive, so don't provide a preview.\n    // TODO(ayazhafiz): Consider providing an actual span:\n    //  1. We're likely to read the template anyway\n    //  2. We could show just the first 100 chars or so\n    textSpan: {start: 0, length: 0},\n    fileName: resourceForExpression.path,\n  }];\n\n  return {\n    definitions: templateDefinitions,\n    textSpan: {\n      // Exclude opening and closing quotes in the url span.\n      start: expression.getStart() + 1,\n      length: expression.getWidth() - 2,\n    },\n  };\n}\n"]} |
\ | No newline at end of file |