UNPKG

28.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/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 "use strict";
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