UNPKG

32.8 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/typecheck/api", "typescript", "@angular/language-service/ivy/hybrid_visitor", "@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 api_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/api");
23 var ts = require("typescript");
24 var hybrid_visitor_1 = require("@angular/language-service/ivy/hybrid_visitor");
25 var utils_1 = require("@angular/language-service/ivy/utils");
26 var DefinitionBuilder = /** @class */ (function () {
27 function DefinitionBuilder(tsLS, compiler) {
28 this.tsLS = tsLS;
29 this.compiler = compiler;
30 }
31 DefinitionBuilder.prototype.getDefinitionAndBoundSpan = function (fileName, position) {
32 var templateInfo = utils_1.getTemplateInfoAtPosition(fileName, position, this.compiler);
33 if (templateInfo === undefined) {
34 return;
35 }
36 var definitionMeta = this.getDefinitionMetaAtPosition(templateInfo, position);
37 // The `$event` of event handlers would point to the $event parameter in the shim file, as in
38 // `_outputHelper(_t3["x"]).subscribe(function ($event): any { $event }) ;`
39 // If we wanted to return something for this, it would be more appropriate for something like
40 // `getTypeDefinition`.
41 if (definitionMeta === undefined || utils_1.isDollarEvent(definitionMeta.node)) {
42 return undefined;
43 }
44 var definitions = this.getDefinitionsForSymbol(tslib_1.__assign(tslib_1.__assign({}, definitionMeta), templateInfo));
45 return { definitions: definitions, textSpan: utils_1.getTextSpanOfNode(definitionMeta.node) };
46 };
47 DefinitionBuilder.prototype.getDefinitionsForSymbol = function (_a) {
48 var symbol = _a.symbol, node = _a.node, path = _a.path, component = _a.component;
49 switch (symbol.kind) {
50 case api_1.SymbolKind.Directive:
51 case api_1.SymbolKind.Element:
52 case api_1.SymbolKind.Template:
53 case api_1.SymbolKind.DomBinding:
54 // Though it is generally more appropriate for the above symbol definitions to be
55 // associated with "type definitions" since the location in the template is the
56 // actual definition location, the better user experience would be to allow
57 // LS users to "go to definition" on an item in the template that maps to a class and be
58 // taken to the directive or HTML class.
59 return this.getTypeDefinitionsForTemplateInstance(symbol, node);
60 case api_1.SymbolKind.Output:
61 case api_1.SymbolKind.Input: {
62 var bindingDefs = this.getDefinitionsForSymbols.apply(this, tslib_1.__spread(symbol.bindings));
63 // Also attempt to get directive matches for the input name. If there is a directive that
64 // has the input name as part of the selector, we want to return that as well.
65 var directiveDefs = this.getDirectiveTypeDefsForBindingNode(node, path, component);
66 return tslib_1.__spread(bindingDefs, directiveDefs);
67 }
68 case api_1.SymbolKind.Variable:
69 case api_1.SymbolKind.Reference: {
70 var definitions = [];
71 if (symbol.declaration !== node) {
72 definitions.push({
73 name: symbol.declaration.name,
74 containerName: '',
75 containerKind: ts.ScriptElementKind.unknown,
76 kind: ts.ScriptElementKind.variableElement,
77 textSpan: utils_1.getTextSpanOfNode(symbol.declaration),
78 contextSpan: utils_1.toTextSpan(symbol.declaration.sourceSpan),
79 fileName: symbol.declaration.sourceSpan.start.file.url,
80 });
81 }
82 if (symbol.kind === api_1.SymbolKind.Variable) {
83 definitions.push.apply(definitions, tslib_1.__spread(this.getDefinitionsForSymbols(symbol)));
84 }
85 return definitions;
86 }
87 case api_1.SymbolKind.Expression: {
88 return this.getDefinitionsForSymbols(symbol);
89 }
90 }
91 };
92 DefinitionBuilder.prototype.getDefinitionsForSymbols = function () {
93 var _this = this;
94 var symbols = [];
95 for (var _i = 0; _i < arguments.length; _i++) {
96 symbols[_i] = arguments[_i];
97 }
98 return utils_1.flatMap(symbols, function (_a) {
99 var _b;
100 var shimLocation = _a.shimLocation;
101 var shimPath = shimLocation.shimPath, positionInShimFile = shimLocation.positionInShimFile;
102 return (_b = _this.tsLS.getDefinitionAtPosition(shimPath, positionInShimFile)) !== null && _b !== void 0 ? _b : [];
103 });
104 };
105 DefinitionBuilder.prototype.getTypeDefinitionsAtPosition = function (fileName, position) {
106 var templateInfo = utils_1.getTemplateInfoAtPosition(fileName, position, this.compiler);
107 if (templateInfo === undefined) {
108 return;
109 }
110 var definitionMeta = this.getDefinitionMetaAtPosition(templateInfo, position);
111 if (definitionMeta === undefined) {
112 return undefined;
113 }
114 var symbol = definitionMeta.symbol, node = definitionMeta.node;
115 switch (symbol.kind) {
116 case api_1.SymbolKind.Directive:
117 case api_1.SymbolKind.DomBinding:
118 case api_1.SymbolKind.Element:
119 case api_1.SymbolKind.Template:
120 return this.getTypeDefinitionsForTemplateInstance(symbol, node);
121 case api_1.SymbolKind.Output:
122 case api_1.SymbolKind.Input: {
123 var bindingDefs = this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(symbol.bindings));
124 // Also attempt to get directive matches for the input name. If there is a directive that
125 // has the input name as part of the selector, we want to return that as well.
126 var directiveDefs = this.getDirectiveTypeDefsForBindingNode(node, definitionMeta.path, templateInfo.component);
127 return tslib_1.__spread(bindingDefs, directiveDefs);
128 }
129 case api_1.SymbolKind.Reference:
130 case api_1.SymbolKind.Expression:
131 case api_1.SymbolKind.Variable:
132 return this.getTypeDefinitionsForSymbols(symbol);
133 }
134 };
135 DefinitionBuilder.prototype.getTypeDefinitionsForTemplateInstance = function (symbol, node) {
136 switch (symbol.kind) {
137 case api_1.SymbolKind.Template: {
138 var matches = utils_1.getDirectiveMatchesForElementTag(symbol.templateNode, symbol.directives);
139 return this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(matches));
140 }
141 case api_1.SymbolKind.Element: {
142 var matches = utils_1.getDirectiveMatchesForElementTag(symbol.templateNode, symbol.directives);
143 // If one of the directive matches is a component, we should not include the native element
144 // in the results because it is replaced by the component.
145 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]));
146 }
147 case api_1.SymbolKind.DomBinding: {
148 if (!(node instanceof compiler_1.TmplAstTextAttribute)) {
149 return [];
150 }
151 var dirs = utils_1.getDirectiveMatchesForAttribute(node.name, symbol.host.templateNode, symbol.host.directives);
152 return this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(dirs));
153 }
154 case api_1.SymbolKind.Directive:
155 return this.getTypeDefinitionsForSymbols(symbol);
156 }
157 };
158 DefinitionBuilder.prototype.getDirectiveTypeDefsForBindingNode = function (node, pathToNode, component) {
159 if (!(node instanceof compiler_1.TmplAstBoundAttribute) && !(node instanceof compiler_1.TmplAstTextAttribute) &&
160 !(node instanceof compiler_1.TmplAstBoundEvent)) {
161 return [];
162 }
163 var parent = pathToNode[pathToNode.length - 2];
164 if (!(parent instanceof compiler_1.TmplAstTemplate || parent instanceof compiler_1.TmplAstElement)) {
165 return [];
166 }
167 var templateOrElementSymbol = this.compiler.getTemplateTypeChecker().getSymbolOfNode(parent, component);
168 if (templateOrElementSymbol === null ||
169 (templateOrElementSymbol.kind !== api_1.SymbolKind.Template &&
170 templateOrElementSymbol.kind !== api_1.SymbolKind.Element)) {
171 return [];
172 }
173 var dirs = utils_1.getDirectiveMatchesForAttribute(node.name, parent, templateOrElementSymbol.directives);
174 return this.getTypeDefinitionsForSymbols.apply(this, tslib_1.__spread(dirs));
175 };
176 DefinitionBuilder.prototype.getTypeDefinitionsForSymbols = function () {
177 var _this = this;
178 var symbols = [];
179 for (var _i = 0; _i < arguments.length; _i++) {
180 symbols[_i] = arguments[_i];
181 }
182 return utils_1.flatMap(symbols, function (_a) {
183 var _b;
184 var shimLocation = _a.shimLocation;
185 var shimPath = shimLocation.shimPath, positionInShimFile = shimLocation.positionInShimFile;
186 return (_b = _this.tsLS.getTypeDefinitionAtPosition(shimPath, positionInShimFile)) !== null && _b !== void 0 ? _b : [];
187 });
188 };
189 DefinitionBuilder.prototype.getDefinitionMetaAtPosition = function (_a, position) {
190 var template = _a.template, component = _a.component;
191 var path = hybrid_visitor_1.getPathToNodeAtPosition(template, position);
192 if (path === undefined) {
193 return;
194 }
195 var node = path[path.length - 1];
196 var symbol = this.compiler.getTemplateTypeChecker().getSymbolOfNode(node, component);
197 if (symbol === null) {
198 return;
199 }
200 return { node: node, path: path, symbol: symbol };
201 };
202 return DefinitionBuilder;
203 }());
204 exports.DefinitionBuilder = DefinitionBuilder;
205});
206//# 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 {DirectiveSymbol, DomBindingSymbol, ElementSymbol, ShimLocation, Symbol, SymbolKind, TemplateSymbol} from '@angular/compiler-cli/src/ngtsc/typecheck/api';\nimport * as ts from 'typescript';\n\nimport {getPathToNodeAtPosition} from './hybrid_visitor';\nimport {flatMap, getDirectiveMatchesForAttribute, getDirectiveMatchesForElementTag, getTemplateInfoAtPosition, getTextSpanOfNode, isDollarEvent, TemplateInfo, toTextSpan} from './utils';\n\ninterface DefinitionMeta {\n  node: AST|TmplAstNode;\n  path: Array<AST|TmplAstNode>;\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      return;\n    }\n    const definitionMeta = this.getDefinitionMetaAtPosition(templateInfo, position);\n    // The `$event` of event handlers would point to the $event parameter in the shim file, as in\n    // `_outputHelper(_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 (definitionMeta === undefined || isDollarEvent(definitionMeta.node)) {\n      return undefined;\n    }\n\n    const definitions = this.getDefinitionsForSymbol({...definitionMeta, ...templateInfo});\n    return {definitions, textSpan: getTextSpanOfNode(definitionMeta.node)};\n  }\n\n  private getDefinitionsForSymbol({symbol, node, path, 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.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, path, 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          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: symbol.declaration.sourceSpan.start.file.url,\n          });\n        }\n        if (symbol.kind === SymbolKind.Variable) {\n          definitions.push(...this.getDefinitionsForSymbols(symbol));\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 definitionMeta = this.getDefinitionMetaAtPosition(templateInfo, position);\n    if (definitionMeta === undefined) {\n      return undefined;\n    }\n\n    const {symbol, node} = definitionMeta;\n    switch (symbol.kind) {\n      case SymbolKind.Directive:\n      case SymbolKind.DomBinding:\n      case SymbolKind.Element:\n      case SymbolKind.Template:\n        return this.getTypeDefinitionsForTemplateInstance(symbol, node);\n      case SymbolKind.Output:\n      case SymbolKind.Input: {\n        const bindingDefs = this.getTypeDefinitionsForSymbols(...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(\n            node, definitionMeta.path, templateInfo.component);\n        return [...bindingDefs, ...directiveDefs];\n      }\n      case SymbolKind.Reference:\n      case SymbolKind.Expression:\n      case SymbolKind.Variable:\n        return this.getTypeDefinitionsForSymbols(symbol);\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, pathToNode: Array<TmplAstNode|AST>, component: ts.ClassDeclaration) {\n    if (!(node instanceof TmplAstBoundAttribute) && !(node instanceof TmplAstTextAttribute) &&\n        !(node instanceof TmplAstBoundEvent)) {\n      return [];\n    }\n    const parent = pathToNode[pathToNode.length - 2];\n    if (!(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 path = getPathToNodeAtPosition(template, position);\n    if (path === undefined) {\n      return;\n    }\n\n    const node = path[path.length - 1];\n    const symbol = this.compiler.getTemplateTypeChecker().getSymbolOfNode(node, component);\n    if (symbol === null) {\n      return;\n    }\n    return {node, path, symbol};\n  }\n}\n"]}
\No newline at end of file