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/hybrid_visitor", ["require", "exports", "tslib", "@angular/compiler", "@angular/compiler/src/expression_parser/ast", "@angular/compiler/src/render3/r3_ast", "@angular/language-service/ivy/utils"], factory);
|
15 | }
|
16 | })(function (require, exports) {
|
17 | ;
|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
19 | exports.findNodeAtPosition = exports.getPathToNodeAtPosition = void 0;
|
20 | var tslib_1 = require("tslib");
|
21 | var compiler_1 = require("@angular/compiler");
|
22 | var e = require("@angular/compiler/src/expression_parser/ast"); // e for expression AST
|
23 | var t = require("@angular/compiler/src/render3/r3_ast"); // t for template AST
|
24 | var utils_1 = require("@angular/language-service/ivy/utils");
|
25 | /**
|
26 | * Return the path to the template AST node or expression AST node that most accurately
|
27 | * represents the node at the specified cursor `position`.
|
28 | *
|
29 | * @param ast AST tree
|
30 | * @param position cursor position
|
31 | */
|
32 | function getPathToNodeAtPosition(ast, position) {
|
33 | var visitor = new R3Visitor(position);
|
34 | visitor.visitAll(ast);
|
35 | var candidate = visitor.path[visitor.path.length - 1];
|
36 | if (!candidate) {
|
37 | return;
|
38 | }
|
39 | if (utils_1.isTemplateNodeWithKeyAndValue(candidate)) {
|
40 | var keySpan = candidate.keySpan, valueSpan = candidate.valueSpan;
|
41 | var isWithinKeyValue = isWithin(position, keySpan) || (valueSpan && isWithin(position, valueSpan));
|
42 | if (!isWithinKeyValue) {
|
43 | // If cursor is within source span but not within key span or value span,
|
44 | // do not return the node.
|
45 | return;
|
46 | }
|
47 | }
|
48 | return visitor.path;
|
49 | }
|
50 | exports.getPathToNodeAtPosition = getPathToNodeAtPosition;
|
51 | /**
|
52 | * Return the template AST node or expression AST node that most accurately
|
53 | * represents the node at the specified cursor `position`.
|
54 | *
|
55 | * @param ast AST tree
|
56 | * @param position cursor position
|
57 | */
|
58 | function findNodeAtPosition(ast, position) {
|
59 | var path = getPathToNodeAtPosition(ast, position);
|
60 | if (!path) {
|
61 | return;
|
62 | }
|
63 | return path[path.length - 1];
|
64 | }
|
65 | exports.findNodeAtPosition = findNodeAtPosition;
|
66 | var R3Visitor = /** @class */ (function () {
|
67 | // Position must be absolute in the source file.
|
68 | function R3Visitor(position) {
|
69 | this.position = position;
|
70 | // We need to keep a path instead of the last node because we might need more
|
71 | // context for the last node, for example what is the parent node?
|
72 | this.path = [];
|
73 | }
|
74 | R3Visitor.prototype.visit = function (node) {
|
75 | var _a = getSpanIncludingEndTag(node), start = _a.start, end = _a.end;
|
76 | if (isWithin(this.position, { start: start, end: end })) {
|
77 | var length_1 = end - start;
|
78 | var last = this.path[this.path.length - 1];
|
79 | if (last) {
|
80 | var _b = utils_1.isTemplateNode(last) ? getSpanIncludingEndTag(last) : last.sourceSpan, start_1 = _b.start, end_1 = _b.end;
|
81 | var lastLength = end_1 - start_1;
|
82 | if (length_1 > lastLength) {
|
83 | // The current node has a span that is larger than the last node found
|
84 | // so we do not descend into it. This typically means we have found
|
85 | // a candidate in one of the root nodes so we do not need to visit
|
86 | // other root nodes.
|
87 | return;
|
88 | }
|
89 | }
|
90 | this.path.push(node);
|
91 | node.visit(this);
|
92 | }
|
93 | };
|
94 | R3Visitor.prototype.visitElement = function (element) {
|
95 | this.visitAll(element.attributes);
|
96 | this.visitAll(element.inputs);
|
97 | this.visitAll(element.outputs);
|
98 | this.visitAll(element.references);
|
99 | this.visitAll(element.children);
|
100 | };
|
101 | R3Visitor.prototype.visitTemplate = function (template) {
|
102 | this.visitAll(template.attributes);
|
103 | this.visitAll(template.inputs);
|
104 | this.visitAll(template.outputs);
|
105 | this.visitAll(template.templateAttrs);
|
106 | this.visitAll(template.references);
|
107 | this.visitAll(template.variables);
|
108 | this.visitAll(template.children);
|
109 | };
|
110 | R3Visitor.prototype.visitContent = function (content) {
|
111 | t.visitAll(this, content.attributes);
|
112 | };
|
113 | R3Visitor.prototype.visitVariable = function (variable) {
|
114 | // Variable has no template nodes or expression nodes.
|
115 | };
|
116 | R3Visitor.prototype.visitReference = function (reference) {
|
117 | // Reference has no template nodes or expression nodes.
|
118 | };
|
119 | R3Visitor.prototype.visitTextAttribute = function (attribute) {
|
120 | // Text attribute has no template nodes or expression nodes.
|
121 | };
|
122 | R3Visitor.prototype.visitBoundAttribute = function (attribute) {
|
123 | var visitor = new ExpressionVisitor(this.position);
|
124 | visitor.visit(attribute.value, this.path);
|
125 | };
|
126 | R3Visitor.prototype.visitBoundEvent = function (event) {
|
127 | var isTwoWayBinding = this.path.some(function (n) { return n instanceof t.BoundAttribute && event.name === n.name + 'Change'; });
|
128 | if (isTwoWayBinding) {
|
129 | // For two-way binding aka banana-in-a-box, there are two matches:
|
130 | // BoundAttribute and BoundEvent. Both have the same spans. We choose to
|
131 | // return BoundAttribute because it matches the identifier name verbatim.
|
132 | // TODO: For operations like go to definition, ideally we want to return
|
133 | // both.
|
134 | this.path.pop(); // remove bound event from the AST path
|
135 | return;
|
136 | }
|
137 | var visitor = new ExpressionVisitor(this.position);
|
138 | visitor.visit(event.handler, this.path);
|
139 | };
|
140 | R3Visitor.prototype.visitText = function (text) {
|
141 | // Text has no template nodes or expression nodes.
|
142 | };
|
143 | R3Visitor.prototype.visitBoundText = function (text) {
|
144 | var visitor = new ExpressionVisitor(this.position);
|
145 | visitor.visit(text.value, this.path);
|
146 | };
|
147 | R3Visitor.prototype.visitIcu = function (icu) {
|
148 | var e_1, _a, e_2, _b;
|
149 | try {
|
150 | for (var _c = tslib_1.__values(Object.values(icu.vars)), _d = _c.next(); !_d.done; _d = _c.next()) {
|
151 | var boundText = _d.value;
|
152 | this.visit(boundText);
|
153 | }
|
154 | }
|
155 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
156 | finally {
|
157 | try {
|
158 | if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
|
159 | }
|
160 | finally { if (e_1) throw e_1.error; }
|
161 | }
|
162 | try {
|
163 | for (var _e = tslib_1.__values(Object.values(icu.placeholders)), _f = _e.next(); !_f.done; _f = _e.next()) {
|
164 | var boundTextOrText = _f.value;
|
165 | this.visit(boundTextOrText);
|
166 | }
|
167 | }
|
168 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
169 | finally {
|
170 | try {
|
171 | if (_f && !_f.done && (_b = _e.return)) _b.call(_e);
|
172 | }
|
173 | finally { if (e_2) throw e_2.error; }
|
174 | }
|
175 | };
|
176 | R3Visitor.prototype.visitAll = function (nodes) {
|
177 | var e_3, _a;
|
178 | try {
|
179 | for (var nodes_1 = tslib_1.__values(nodes), nodes_1_1 = nodes_1.next(); !nodes_1_1.done; nodes_1_1 = nodes_1.next()) {
|
180 | var node = nodes_1_1.value;
|
181 | this.visit(node);
|
182 | }
|
183 | }
|
184 | catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
185 | finally {
|
186 | try {
|
187 | if (nodes_1_1 && !nodes_1_1.done && (_a = nodes_1.return)) _a.call(nodes_1);
|
188 | }
|
189 | finally { if (e_3) throw e_3.error; }
|
190 | }
|
191 | };
|
192 | return R3Visitor;
|
193 | }());
|
194 | var ExpressionVisitor = /** @class */ (function (_super) {
|
195 | tslib_1.__extends(ExpressionVisitor, _super);
|
196 | // Position must be absolute in the source file.
|
197 | function ExpressionVisitor(position) {
|
198 | var _this = _super.call(this) || this;
|
199 | _this.position = position;
|
200 | return _this;
|
201 | }
|
202 | ExpressionVisitor.prototype.visit = function (node, path) {
|
203 | if (node instanceof e.ASTWithSource) {
|
204 | // In order to reduce noise, do not include `ASTWithSource` in the path.
|
205 | // For the purpose of source spans, there is no difference between
|
206 | // `ASTWithSource` and and underlying node that it wraps.
|
207 | node = node.ast;
|
208 | }
|
209 | // The third condition is to account for the implicit receiver, which should
|
210 | // not be visited.
|
211 | if (isWithin(this.position, node.sourceSpan) && !(node instanceof e.ImplicitReceiver)) {
|
212 | path.push(node);
|
213 | node.visit(this, path);
|
214 | }
|
215 | };
|
216 | return ExpressionVisitor;
|
217 | }(e.RecursiveAstVisitor));
|
218 | function getSpanIncludingEndTag(ast) {
|
219 | var result = {
|
220 | start: ast.sourceSpan.start.offset,
|
221 | end: ast.sourceSpan.end.offset,
|
222 | };
|
223 | // For Element and Template node, sourceSpan.end is the end of the opening
|
224 | // tag. For the purpose of language service, we need to actually recognize
|
225 | // the end of the closing tag. Otherwise, for situation like
|
226 | // <my-component></my-comp¦onent> where the cursor is in the closing tag
|
227 | // we will not be able to return any information.
|
228 | if ((ast instanceof t.Element || ast instanceof t.Template) && ast.endSourceSpan) {
|
229 | result.end = ast.endSourceSpan.end.offset;
|
230 | }
|
231 | return result;
|
232 | }
|
233 | function isWithin(position, span) {
|
234 | var start, end;
|
235 | if (span instanceof compiler_1.ParseSourceSpan) {
|
236 | start = span.start.offset;
|
237 | end = span.end.offset;
|
238 | }
|
239 | else {
|
240 | start = span.start;
|
241 | end = span.end;
|
242 | }
|
243 | // Note both start and end are inclusive because we want to match conditions
|
244 | // like ¦start and end¦ where ¦ is the cursor.
|
245 | return start <= position && position <= end;
|
246 | }
|
247 | });
|
248 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"hybrid_visitor.js","sourceRoot":"","sources":["../../../../../../packages/language-service/ivy/hybrid_visitor.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 {AbsoluteSourceSpan, ParseSourceSpan} from '@angular/compiler';\nimport * as e from '@angular/compiler/src/expression_parser/ast';  // e for expression AST\nimport * as t from '@angular/compiler/src/render3/r3_ast';         // t for template AST\n\nimport {isTemplateNode, isTemplateNodeWithKeyAndValue} from './utils';\n\n/**\n * Return the path to the template AST node or expression AST node that most accurately\n * represents the node at the specified cursor `position`.\n *\n * @param ast AST tree\n * @param position cursor position\n */\nexport function getPathToNodeAtPosition(ast: t.Node[], position: number): Array<t.Node|e.AST>|\n    undefined {\n  const visitor = new R3Visitor(position);\n  visitor.visitAll(ast);\n  const candidate = visitor.path[visitor.path.length - 1];\n  if (!candidate) {\n    return;\n  }\n  if (isTemplateNodeWithKeyAndValue(candidate)) {\n    const {keySpan, valueSpan} = candidate;\n    const isWithinKeyValue =\n        isWithin(position, keySpan) || (valueSpan && isWithin(position, valueSpan));\n    if (!isWithinKeyValue) {\n      // If cursor is within source span but not within key span or value span,\n      // do not return the node.\n      return;\n    }\n  }\n  return visitor.path;\n}\n\n/**\n * Return the template AST node or expression AST node that most accurately\n * represents the node at the specified cursor `position`.\n *\n * @param ast AST tree\n * @param position cursor position\n */\nexport function findNodeAtPosition(ast: t.Node[], position: number): t.Node|e.AST|undefined {\n  const path = getPathToNodeAtPosition(ast, position);\n  if (!path) {\n    return;\n  }\n  return path[path.length - 1];\n}\n\nclass R3Visitor implements t.Visitor {\n  // We need to keep a path instead of the last node because we might need more\n  // context for the last node, for example what is the parent node?\n  readonly path: Array<t.Node|e.AST> = [];\n\n  // Position must be absolute in the source file.\n  constructor(private readonly position: number) {}\n\n  visit(node: t.Node) {\n    const {start, end} = getSpanIncludingEndTag(node);\n    if (isWithin(this.position, {start, end})) {\n      const length = end - start;\n      const last: t.Node|e.AST|undefined = this.path[this.path.length - 1];\n      if (last) {\n        const {start, end} = isTemplateNode(last) ? getSpanIncludingEndTag(last) : last.sourceSpan;\n        const lastLength = end - start;\n        if (length > lastLength) {\n          // The current node has a span that is larger than the last node found\n          // so we do not descend into it. This typically means we have found\n          // a candidate in one of the root nodes so we do not need to visit\n          // other root nodes.\n          return;\n        }\n      }\n      this.path.push(node);\n      node.visit(this);\n    }\n  }\n\n  visitElement(element: t.Element) {\n    this.visitAll(element.attributes);\n    this.visitAll(element.inputs);\n    this.visitAll(element.outputs);\n    this.visitAll(element.references);\n    this.visitAll(element.children);\n  }\n\n  visitTemplate(template: t.Template) {\n    this.visitAll(template.attributes);\n    this.visitAll(template.inputs);\n    this.visitAll(template.outputs);\n    this.visitAll(template.templateAttrs);\n    this.visitAll(template.references);\n    this.visitAll(template.variables);\n    this.visitAll(template.children);\n  }\n\n  visitContent(content: t.Content) {\n    t.visitAll(this, content.attributes);\n  }\n\n  visitVariable(variable: t.Variable) {\n    // Variable has no template nodes or expression nodes.\n  }\n\n  visitReference(reference: t.Reference) {\n    // Reference has no template nodes or expression nodes.\n  }\n\n  visitTextAttribute(attribute: t.TextAttribute) {\n    // Text attribute has no template nodes or expression nodes.\n  }\n\n  visitBoundAttribute(attribute: t.BoundAttribute) {\n    const visitor = new ExpressionVisitor(this.position);\n    visitor.visit(attribute.value, this.path);\n  }\n\n  visitBoundEvent(event: t.BoundEvent) {\n    const isTwoWayBinding =\n        this.path.some(n => n instanceof t.BoundAttribute && event.name === n.name + 'Change');\n    if (isTwoWayBinding) {\n      // For two-way binding aka banana-in-a-box, there are two matches:\n      // BoundAttribute and BoundEvent. Both have the same spans. We choose to\n      // return BoundAttribute because it matches the identifier name verbatim.\n      // TODO: For operations like go to definition, ideally we want to return\n      // both.\n      this.path.pop();  // remove bound event from the AST path\n      return;\n    }\n    const visitor = new ExpressionVisitor(this.position);\n    visitor.visit(event.handler, this.path);\n  }\n\n  visitText(text: t.Text) {\n    // Text has no template nodes or expression nodes.\n  }\n\n  visitBoundText(text: t.BoundText) {\n    const visitor = new ExpressionVisitor(this.position);\n    visitor.visit(text.value, this.path);\n  }\n\n  visitIcu(icu: t.Icu) {\n    for (const boundText of Object.values(icu.vars)) {\n      this.visit(boundText);\n    }\n    for (const boundTextOrText of Object.values(icu.placeholders)) {\n      this.visit(boundTextOrText);\n    }\n  }\n\n  visitAll(nodes: t.Node[]) {\n    for (const node of nodes) {\n      this.visit(node);\n    }\n  }\n}\n\nclass ExpressionVisitor extends e.RecursiveAstVisitor {\n  // Position must be absolute in the source file.\n  constructor(private readonly position: number) {\n    super();\n  }\n\n  visit(node: e.AST, path: Array<t.Node|e.AST>) {\n    if (node instanceof e.ASTWithSource) {\n      // In order to reduce noise, do not include `ASTWithSource` in the path.\n      // For the purpose of source spans, there is no difference between\n      // `ASTWithSource` and and underlying node that it wraps.\n      node = node.ast;\n    }\n    // The third condition is to account for the implicit receiver, which should\n    // not be visited.\n    if (isWithin(this.position, node.sourceSpan) && !(node instanceof e.ImplicitReceiver)) {\n      path.push(node);\n      node.visit(this, path);\n    }\n  }\n}\n\nfunction getSpanIncludingEndTag(ast: t.Node) {\n  const result = {\n    start: ast.sourceSpan.start.offset,\n    end: ast.sourceSpan.end.offset,\n  };\n  // For Element and Template node, sourceSpan.end is the end of the opening\n  // tag. For the purpose of language service, we need to actually recognize\n  // the end of the closing tag. Otherwise, for situation like\n  // <my-component></my-comp¦onent> where the cursor is in the closing tag\n  // we will not be able to return any information.\n  if ((ast instanceof t.Element || ast instanceof t.Template) && ast.endSourceSpan) {\n    result.end = ast.endSourceSpan.end.offset;\n  }\n  return result;\n}\n\nfunction isWithin(position: number, span: AbsoluteSourceSpan|ParseSourceSpan): boolean {\n  let start: number, end: number;\n  if (span instanceof ParseSourceSpan) {\n    start = span.start.offset;\n    end = span.end.offset;\n  } else {\n    start = span.start;\n    end = span.end;\n  }\n  // Note both start and end are inclusive because we want to match conditions\n  // like ¦start and end¦ where ¦ is the cursor.\n  return start <= position && position <= end;\n}\n"]} |
\ | No newline at end of file |