UNPKG

29.4 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/src/expressions", ["require", "exports", "tslib", "@angular/compiler", "@angular/language-service/src/expression_type", "@angular/language-service/src/types", "@angular/language-service/src/utils"], factory);
15 }
16})(function (require, exports) {
17 "use strict";
18 Object.defineProperty(exports, "__esModule", { value: true });
19 exports.getExpressionSymbol = exports.getExpressionCompletions = void 0;
20 var tslib_1 = require("tslib");
21 var compiler_1 = require("@angular/compiler");
22 var expression_type_1 = require("@angular/language-service/src/expression_type");
23 var types_1 = require("@angular/language-service/src/types");
24 var utils_1 = require("@angular/language-service/src/utils");
25 function findAstAt(ast, position, excludeEmpty) {
26 if (excludeEmpty === void 0) { excludeEmpty = false; }
27 var path = [];
28 var visitor = new /** @class */ (function (_super) {
29 tslib_1.__extends(class_1, _super);
30 function class_1() {
31 return _super !== null && _super.apply(this, arguments) || this;
32 }
33 class_1.prototype.visit = function (ast) {
34 if ((!excludeEmpty || ast.sourceSpan.start < ast.sourceSpan.end) &&
35 utils_1.inSpan(position, ast.sourceSpan)) {
36 var isNotNarrower = path.length && !utils_1.isNarrower(ast.span, path[path.length - 1].span);
37 if (!isNotNarrower) {
38 path.push(ast);
39 }
40 ast.visit(this);
41 }
42 };
43 return class_1;
44 }(compiler_1.RecursiveAstVisitor));
45 // We never care about the ASTWithSource node and its visit() method calls its ast's visit so
46 // the visit() method above would never see it.
47 if (ast instanceof compiler_1.ASTWithSource) {
48 ast = ast.ast;
49 }
50 // `Interpolation` is useless here except the `expressions` of it.
51 if (ast instanceof compiler_1.Interpolation) {
52 ast = ast.expressions.filter(function (_ast) { return utils_1.inSpan(position, _ast.sourceSpan); })[0];
53 }
54 if (ast) {
55 visitor.visit(ast);
56 }
57 return new compiler_1.AstPath(path, position);
58 }
59 function getExpressionCompletions(scope, ast, position, templateInfo) {
60 var path = findAstAt(ast, position);
61 if (path.empty)
62 return undefined;
63 var tail = path.tail;
64 var result = scope;
65 function getType(ast) {
66 return new expression_type_1.AstType(scope, templateInfo.query, {}, templateInfo.source).getType(ast);
67 }
68 // If the completion request is in a not in a pipe or property access then the global scope
69 // (that is the scope of the implicit receiver) is the right scope as the user is typing the
70 // beginning of an expression.
71 tail.visit({
72 visitUnary: function (_ast) { },
73 visitBinary: function (_ast) { },
74 visitChain: function (_ast) { },
75 visitConditional: function (_ast) { },
76 visitFunctionCall: function (_ast) { },
77 visitImplicitReceiver: function (_ast) { },
78 visitThisReceiver: function (_ast) { },
79 visitInterpolation: function (_ast) {
80 result = undefined;
81 },
82 visitKeyedRead: function (_ast) { },
83 visitKeyedWrite: function (_ast) { },
84 visitLiteralArray: function (_ast) { },
85 visitLiteralMap: function (_ast) { },
86 visitLiteralPrimitive: function (ast) {
87 // The type `LiteralPrimitive` include the `ERROR`, and it's wrapped as `string`.
88 // packages/compiler/src/template_parser/binding_parser.ts#L308
89 // So exclude the `ERROR` here.
90 if (typeof ast.value === 'string' &&
91 ast.value ===
92 templateInfo.source.slice(ast.sourceSpan.start + 1, ast.sourceSpan.end - 1)) {
93 result = undefined;
94 }
95 },
96 visitMethodCall: function (_ast) { },
97 visitPipe: function (ast) {
98 if (position >= ast.exp.span.end &&
99 (!ast.args || !ast.args.length || position < ast.args[0].span.start)) {
100 // We are in a position a pipe name is expected.
101 result = templateInfo.query.getPipes();
102 }
103 },
104 visitPrefixNot: function (_ast) { },
105 visitNonNullAssert: function (_ast) { },
106 visitPropertyRead: function (ast) {
107 var receiverType = getType(ast.receiver);
108 result = receiverType ? receiverType.members() : scope;
109 },
110 visitPropertyWrite: function (ast) {
111 var receiverType = getType(ast.receiver);
112 result = receiverType ? receiverType.members() : scope;
113 },
114 visitQuote: function (_ast) {
115 // For a quote, return the members of any (if there are any).
116 result = templateInfo.query.getBuiltinType(types_1.BuiltinType.Any).members();
117 },
118 visitSafeMethodCall: function (ast) {
119 var receiverType = getType(ast.receiver);
120 result = receiverType ? receiverType.members() : scope;
121 },
122 visitSafePropertyRead: function (ast) {
123 var receiverType = getType(ast.receiver);
124 result = receiverType ? receiverType.members() : scope;
125 },
126 });
127 return result && result.values();
128 }
129 exports.getExpressionCompletions = getExpressionCompletions;
130 /**
131 * Retrieves the expression symbol at a particular position in a template.
132 *
133 * @param scope symbols in scope of the template
134 * @param ast template AST
135 * @param position absolute location in template to retrieve symbol at
136 * @param query type symbol query for the template scope
137 */
138 function getExpressionSymbol(scope, ast, position, templateInfo) {
139 var path = findAstAt(ast, position, /* excludeEmpty */ true);
140 if (path.empty)
141 return undefined;
142 var tail = path.tail;
143 function getType(ast) {
144 return new expression_type_1.AstType(scope, templateInfo.query, {}, templateInfo.source).getType(ast);
145 }
146 function spanFromName(ast) {
147 // `nameSpan` is an absolute span, but the span expected by the result of this method is
148 // relative to the start of the expression.
149 // TODO(ayazhafiz): migrate to only using absolute spans
150 var offset = ast.sourceSpan.start - ast.span.start;
151 return {
152 start: ast.nameSpan.start - offset,
153 end: ast.nameSpan.end - offset,
154 };
155 }
156 var symbol = undefined;
157 var span = undefined;
158 // If the completion request is in a not in a pipe or property access then the global scope
159 // (that is the scope of the implicit receiver) is the right scope as the user is typing the
160 // beginning of an expression.
161 tail.visit({
162 visitUnary: function (_ast) { },
163 visitBinary: function (_ast) { },
164 visitChain: function (_ast) { },
165 visitConditional: function (_ast) { },
166 visitFunctionCall: function (_ast) { },
167 visitImplicitReceiver: function (_ast) { },
168 visitThisReceiver: function (_ast) { },
169 visitInterpolation: function (_ast) { },
170 visitKeyedRead: function (_ast) { },
171 visitKeyedWrite: function (_ast) { },
172 visitLiteralArray: function (_ast) { },
173 visitLiteralMap: function (_ast) { },
174 visitLiteralPrimitive: function (_ast) { },
175 visitMethodCall: function (ast) {
176 var receiverType = getType(ast.receiver);
177 symbol = receiverType && receiverType.members().get(ast.name);
178 span = spanFromName(ast);
179 },
180 visitPipe: function (ast) {
181 if (utils_1.inSpan(position, ast.nameSpan, /* exclusive */ true)) {
182 // We are in a position a pipe name is expected.
183 var pipes = templateInfo.query.getPipes();
184 symbol = pipes.get(ast.name);
185 span = spanFromName(ast);
186 }
187 },
188 visitPrefixNot: function (_ast) { },
189 visitNonNullAssert: function (_ast) { },
190 visitPropertyRead: function (ast) {
191 var receiverType = getType(ast.receiver);
192 symbol = receiverType && receiverType.members().get(ast.name);
193 span = spanFromName(ast);
194 },
195 visitPropertyWrite: function (ast) {
196 var receiverType = getType(ast.receiver);
197 symbol = receiverType && receiverType.members().get(ast.name);
198 span = spanFromName(ast);
199 },
200 visitQuote: function (_ast) { },
201 visitSafeMethodCall: function (ast) {
202 var receiverType = getType(ast.receiver);
203 symbol = receiverType && receiverType.members().get(ast.name);
204 span = spanFromName(ast);
205 },
206 visitSafePropertyRead: function (ast) {
207 var receiverType = getType(ast.receiver);
208 symbol = receiverType && receiverType.members().get(ast.name);
209 span = spanFromName(ast);
210 },
211 });
212 if (symbol && span) {
213 return { symbol: symbol, span: span };
214 }
215 }
216 exports.getExpressionSymbol = getExpressionSymbol;
217});
218//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expressions.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/expressions.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, AstPath as AstPathBase, ASTWithName, ASTWithSource, Interpolation, RecursiveAstVisitor} from '@angular/compiler';\n\nimport {AstType} from './expression_type';\nimport {BuiltinType, Span, Symbol, SymbolTable, TemplateSource} from './types';\nimport {inSpan, isNarrower} from './utils';\n\ntype AstPath = AstPathBase<AST>;\n\nfunction findAstAt(ast: AST, position: number, excludeEmpty: boolean = false): AstPath {\n  const path: AST[] = [];\n  const visitor = new class extends RecursiveAstVisitor {\n    visit(ast: AST) {\n      if ((!excludeEmpty || ast.sourceSpan.start < ast.sourceSpan.end) &&\n          inSpan(position, ast.sourceSpan)) {\n        const isNotNarrower = path.length && !isNarrower(ast.span, path[path.length - 1].span);\n        if (!isNotNarrower) {\n          path.push(ast);\n        }\n        ast.visit(this);\n      }\n    }\n  };\n\n  // We never care about the ASTWithSource node and its visit() method calls its ast's visit so\n  // the visit() method above would never see it.\n  if (ast instanceof ASTWithSource) {\n    ast = ast.ast;\n  }\n\n  // `Interpolation` is useless here except the `expressions` of it.\n  if (ast instanceof Interpolation) {\n    ast = ast.expressions.filter((_ast: AST) => inSpan(position, _ast.sourceSpan))[0];\n  }\n\n  if (ast) {\n    visitor.visit(ast);\n  }\n\n  return new AstPathBase<AST>(path, position);\n}\n\nexport function getExpressionCompletions(\n    scope: SymbolTable, ast: AST, position: number, templateInfo: TemplateSource): Symbol[]|\n    undefined {\n  const path = findAstAt(ast, position);\n  if (path.empty) return undefined;\n  const tail = path.tail!;\n  let result: SymbolTable|undefined = scope;\n\n  function getType(ast: AST): Symbol {\n    return new AstType(scope, templateInfo.query, {}, templateInfo.source).getType(ast);\n  }\n\n  // If the completion request is in a not in a pipe or property access then the global scope\n  // (that is the scope of the implicit receiver) is the right scope as the user is typing the\n  // beginning of an expression.\n  tail.visit({\n    visitUnary(_ast) {},\n    visitBinary(_ast) {},\n    visitChain(_ast) {},\n    visitConditional(_ast) {},\n    visitFunctionCall(_ast) {},\n    visitImplicitReceiver(_ast) {},\n    visitThisReceiver(_ast) {},\n    visitInterpolation(_ast) {\n      result = undefined;\n    },\n    visitKeyedRead(_ast) {},\n    visitKeyedWrite(_ast) {},\n    visitLiteralArray(_ast) {},\n    visitLiteralMap(_ast) {},\n    visitLiteralPrimitive(ast) {\n      // The type `LiteralPrimitive` include the `ERROR`, and it's wrapped as `string`.\n      // packages/compiler/src/template_parser/binding_parser.ts#L308\n      // So exclude the `ERROR` here.\n      if (typeof ast.value === 'string' &&\n          ast.value ===\n              templateInfo.source.slice(ast.sourceSpan.start + 1, ast.sourceSpan.end - 1)) {\n        result = undefined;\n      }\n    },\n    visitMethodCall(_ast) {},\n    visitPipe(ast) {\n      if (position >= ast.exp.span.end &&\n          (!ast.args || !ast.args.length || position < (<AST>ast.args[0]).span.start)) {\n        // We are in a position a pipe name is expected.\n        result = templateInfo.query.getPipes();\n      }\n    },\n    visitPrefixNot(_ast) {},\n    visitNonNullAssert(_ast) {},\n    visitPropertyRead(ast) {\n      const receiverType = getType(ast.receiver);\n      result = receiverType ? receiverType.members() : scope;\n    },\n    visitPropertyWrite(ast) {\n      const receiverType = getType(ast.receiver);\n      result = receiverType ? receiverType.members() : scope;\n    },\n    visitQuote(_ast) {\n      // For a quote, return the members of any (if there are any).\n      result = templateInfo.query.getBuiltinType(BuiltinType.Any).members();\n    },\n    visitSafeMethodCall(ast) {\n      const receiverType = getType(ast.receiver);\n      result = receiverType ? receiverType.members() : scope;\n    },\n    visitSafePropertyRead(ast) {\n      const receiverType = getType(ast.receiver);\n      result = receiverType ? receiverType.members() : scope;\n    },\n  });\n\n  return result && result.values();\n}\n\n/**\n * Retrieves the expression symbol at a particular position in a template.\n *\n * @param scope symbols in scope of the template\n * @param ast template AST\n * @param position absolute location in template to retrieve symbol at\n * @param query type symbol query for the template scope\n */\nexport function getExpressionSymbol(\n    scope: SymbolTable, ast: AST, position: number,\n    templateInfo: TemplateSource): {symbol: Symbol, span: Span}|undefined {\n  const path = findAstAt(ast, position, /* excludeEmpty */ true);\n  if (path.empty) return undefined;\n  const tail = path.tail!;\n\n  function getType(ast: AST): Symbol {\n    return new AstType(scope, templateInfo.query, {}, templateInfo.source).getType(ast);\n  }\n\n  function spanFromName(ast: ASTWithName): Span {\n    // `nameSpan` is an absolute span, but the span expected by the result of this method is\n    // relative to the start of the expression.\n    // TODO(ayazhafiz): migrate to only using absolute spans\n    const offset = ast.sourceSpan.start - ast.span.start;\n    return {\n      start: ast.nameSpan.start - offset,\n      end: ast.nameSpan.end - offset,\n    };\n  }\n\n  let symbol: Symbol|undefined = undefined;\n  let span: Span|undefined = undefined;\n\n  // If the completion request is in a not in a pipe or property access then the global scope\n  // (that is the scope of the implicit receiver) is the right scope as the user is typing the\n  // beginning of an expression.\n  tail.visit({\n    visitUnary(_ast) {},\n    visitBinary(_ast) {},\n    visitChain(_ast) {},\n    visitConditional(_ast) {},\n    visitFunctionCall(_ast) {},\n    visitImplicitReceiver(_ast) {},\n    visitThisReceiver(_ast) {},\n    visitInterpolation(_ast) {},\n    visitKeyedRead(_ast) {},\n    visitKeyedWrite(_ast) {},\n    visitLiteralArray(_ast) {},\n    visitLiteralMap(_ast) {},\n    visitLiteralPrimitive(_ast) {},\n    visitMethodCall(ast) {\n      const receiverType = getType(ast.receiver);\n      symbol = receiverType && receiverType.members().get(ast.name);\n      span = spanFromName(ast);\n    },\n    visitPipe(ast) {\n      if (inSpan(position, ast.nameSpan, /* exclusive */ true)) {\n        // We are in a position a pipe name is expected.\n        const pipes = templateInfo.query.getPipes();\n        symbol = pipes.get(ast.name);\n        span = spanFromName(ast);\n      }\n    },\n    visitPrefixNot(_ast) {},\n    visitNonNullAssert(_ast) {},\n    visitPropertyRead(ast) {\n      const receiverType = getType(ast.receiver);\n      symbol = receiverType && receiverType.members().get(ast.name);\n      span = spanFromName(ast);\n    },\n    visitPropertyWrite(ast) {\n      const receiverType = getType(ast.receiver);\n      symbol = receiverType && receiverType.members().get(ast.name);\n      span = spanFromName(ast);\n    },\n    visitQuote(_ast) {},\n    visitSafeMethodCall(ast) {\n      const receiverType = getType(ast.receiver);\n      symbol = receiverType && receiverType.members().get(ast.name);\n      span = spanFromName(ast);\n    },\n    visitSafePropertyRead(ast) {\n      const receiverType = getType(ast.receiver);\n      symbol = receiverType && receiverType.members().get(ast.name);\n      span = spanFromName(ast);\n    },\n  });\n\n  if (symbol && span) {\n    return {symbol, span};\n  }\n}\n"]}
\No newline at end of file