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 | ;
|
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 |