UNPKG

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