UNPKG

89.2 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/completions", ["require", "exports", "tslib", "@angular/compiler", "@angular/compiler/src/chars", "@angular/language-service/src/binding_utils", "@angular/language-service/src/expression_diagnostics", "@angular/language-service/src/expressions", "@angular/language-service/src/html_info", "@angular/language-service/src/template", "@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.getTemplateCompletions = void 0;
20 var tslib_1 = require("tslib");
21 var compiler_1 = require("@angular/compiler");
22 var chars_1 = require("@angular/compiler/src/chars");
23 var binding_utils_1 = require("@angular/language-service/src/binding_utils");
24 var expression_diagnostics_1 = require("@angular/language-service/src/expression_diagnostics");
25 var expressions_1 = require("@angular/language-service/src/expressions");
26 var html_info_1 = require("@angular/language-service/src/html_info");
27 var template_1 = require("@angular/language-service/src/template");
28 var ng = require("@angular/language-service/src/types");
29 var utils_1 = require("@angular/language-service/src/utils");
30 var HIDDEN_HTML_ELEMENTS = new Set(['html', 'script', 'noscript', 'base', 'body', 'title', 'head', 'link']);
31 var HTML_ELEMENTS = html_info_1.elementNames().filter(function (name) { return !HIDDEN_HTML_ELEMENTS.has(name); }).map(function (name) {
32 return {
33 name: name,
34 kind: ng.CompletionKind.HTML_ELEMENT,
35 sortText: name,
36 };
37 });
38 var ANGULAR_ELEMENTS = [
39 {
40 name: 'ng-container',
41 kind: ng.CompletionKind.ANGULAR_ELEMENT,
42 sortText: 'ng-container',
43 },
44 {
45 name: 'ng-content',
46 kind: ng.CompletionKind.ANGULAR_ELEMENT,
47 sortText: 'ng-content',
48 },
49 {
50 name: 'ng-template',
51 kind: ng.CompletionKind.ANGULAR_ELEMENT,
52 sortText: 'ng-template',
53 },
54 ];
55 function isIdentifierPart(code) {
56 // Identifiers consist of alphanumeric characters, '_', or '$'.
57 return chars_1.isAsciiLetter(code) || chars_1.isDigit(code) || code == chars_1.$$ || code == chars_1.$_;
58 }
59 /**
60 * Gets the span of word in a template that surrounds `position`. If there is no word around
61 * `position`, nothing is returned.
62 */
63 function getBoundedWordSpan(templateInfo, position, ast) {
64 var template = templateInfo.template;
65 var templateSrc = template.source;
66 if (!templateSrc)
67 return;
68 if (ast instanceof compiler_1.Element) {
69 // The HTML tag may include `-` (e.g. `app-root`),
70 // so use the HtmlAst to get the span before ayazhafiz refactor the code.
71 return {
72 start: templateInfo.template.span.start + ast.startSourceSpan.start.offset + 1,
73 length: ast.name.length
74 };
75 }
76 // TODO(ayazhafiz): A solution based on word expansion will always be expensive compared to one
77 // based on ASTs. Whatever penalty we incur is probably manageable for small-length (i.e. the
78 // majority of) identifiers, but the current solution involes a number of branchings and we can't
79 // control potentially very long identifiers. Consider moving to an AST-based solution once
80 // existing difficulties with AST spans are more clearly resolved (see #31898 for discussion of
81 // known problems, and #33091 for how they affect text replacement).
82 //
83 // `templatePosition` represents the right-bound location of a cursor in the template.
84 // key.ent|ry
85 // ^---- cursor, at position `r` is at.
86 // A cursor is not itself a character in the template; it has a left (lower) and right (upper)
87 // index bound that hugs the cursor itself.
88 var templatePosition = position - template.span.start;
89 // To perform word expansion, we want to determine the left and right indices that hug the cursor.
90 // There are three cases here.
91 var left, right;
92 if (templatePosition === 0) {
93 // 1. Case like
94 // |rest of template
95 // the cursor is at the start of the template, hugged only by the right side (0-index).
96 left = right = 0;
97 }
98 else if (templatePosition === templateSrc.length) {
99 // 2. Case like
100 // rest of template|
101 // the cursor is at the end of the template, hugged only by the left side (last-index).
102 left = right = templateSrc.length - 1;
103 }
104 else {
105 // 3. Case like
106 // wo|rd
107 // there is a clear left and right index.
108 left = templatePosition - 1;
109 right = templatePosition;
110 }
111 if (!isIdentifierPart(templateSrc.charCodeAt(left)) &&
112 !isIdentifierPart(templateSrc.charCodeAt(right))) {
113 // Case like
114 // .|.
115 // left ---^ ^--- right
116 // There is no word here.
117 return;
118 }
119 // Expand on the left and right side until a word boundary is hit. Back up one expansion on both
120 // side to stay inside the word.
121 while (left >= 0 && isIdentifierPart(templateSrc.charCodeAt(left)))
122 --left;
123 ++left;
124 while (right < templateSrc.length && isIdentifierPart(templateSrc.charCodeAt(right)))
125 ++right;
126 --right;
127 var absoluteStartPosition = position - (templatePosition - left);
128 var length = right - left + 1;
129 return { start: absoluteStartPosition, length: length };
130 }
131 function getTemplateCompletions(templateInfo, position) {
132 var htmlAst = templateInfo.htmlAst, template = templateInfo.template;
133 // Calculate the position relative to the start of the template. This is needed
134 // because spans in HTML AST are relative. Inline template has non-zero start position.
135 var templatePosition = position - template.span.start;
136 var htmlPath = utils_1.getPathToNodeAtPosition(htmlAst, templatePosition);
137 var mostSpecific = htmlPath.tail;
138 var visitor = new HtmlVisitor(templateInfo, htmlPath);
139 var results = mostSpecific ?
140 mostSpecific.visit(visitor, null /* context */) :
141 elementCompletions(templateInfo);
142 var replacementSpan = getBoundedWordSpan(templateInfo, position, mostSpecific);
143 return results.map(function (entry) {
144 return tslib_1.__assign(tslib_1.__assign({}, entry), { replacementSpan: replacementSpan });
145 });
146 }
147 exports.getTemplateCompletions = getTemplateCompletions;
148 var HtmlVisitor = /** @class */ (function () {
149 function HtmlVisitor(templateInfo, htmlPath) {
150 this.templateInfo = templateInfo;
151 this.htmlPath = htmlPath;
152 this.relativePosition = htmlPath.position;
153 }
154 // Note that every visitor method must explicitly specify return type because
155 // Visitor returns `any` for all methods.
156 HtmlVisitor.prototype.visitElement = function (ast) {
157 var startTagSpan = utils_1.spanOf(ast.sourceSpan);
158 var tagLen = ast.name.length;
159 // + 1 for the opening angle bracket
160 if (this.relativePosition <= startTagSpan.start + tagLen + 1) {
161 // If we are in the tag then return the element completions.
162 return elementCompletions(this.templateInfo);
163 }
164 if (this.relativePosition < startTagSpan.end) {
165 // We are in the attribute section of the element (but not in an attribute).
166 // Return the attribute completions.
167 return attributeCompletionsForElement(this.templateInfo, ast.name);
168 }
169 return [];
170 };
171 HtmlVisitor.prototype.visitAttribute = function (ast) {
172 // An attribute consists of two parts, LHS="RHS".
173 // Determine if completions are requested for LHS or RHS
174 if (ast.valueSpan && utils_1.inSpan(this.relativePosition, utils_1.spanOf(ast.valueSpan))) {
175 // RHS completion
176 return attributeValueCompletions(this.templateInfo, this.htmlPath);
177 }
178 // LHS completion
179 return attributeCompletions(this.templateInfo, this.htmlPath);
180 };
181 HtmlVisitor.prototype.visitText = function () {
182 var _this = this;
183 var _a;
184 var templatePath = utils_1.findTemplateAstAt(this.templateInfo.templateAst, this.relativePosition);
185 if (templatePath.tail instanceof compiler_1.BoundTextAst) {
186 // If we know that this is an interpolation then do not try other scenarios.
187 var visitor = new ExpressionVisitor(this.templateInfo, this.relativePosition, function () {
188 return expression_diagnostics_1.getExpressionScope(utils_1.diagnosticInfoFromTemplateInfo(_this.templateInfo), templatePath);
189 });
190 (_a = templatePath.tail) === null || _a === void 0 ? void 0 : _a.visit(visitor, null);
191 return visitor.results;
192 }
193 // TODO(kyliau): Not sure if this check is really needed since we don't have
194 // any test cases for it.
195 var element = this.htmlPath.first(compiler_1.Element);
196 if (element &&
197 compiler_1.getHtmlTagDefinition(element.name).getContentType() !== compiler_1.TagContentType.PARSABLE_DATA) {
198 return [];
199 }
200 // This is to account for cases like <h1> <a> text | </h1> where the
201 // closest element has no closing tag and thus is considered plain text.
202 var results = voidElementAttributeCompletions(this.templateInfo, this.htmlPath);
203 if (results.length) {
204 return results;
205 }
206 return elementCompletions(this.templateInfo);
207 };
208 HtmlVisitor.prototype.visitComment = function () {
209 return [];
210 };
211 HtmlVisitor.prototype.visitExpansion = function () {
212 return [];
213 };
214 HtmlVisitor.prototype.visitExpansionCase = function () {
215 return [];
216 };
217 return HtmlVisitor;
218 }());
219 function attributeCompletions(info, path) {
220 var attr = path.tail;
221 var elem = path.parentOf(attr);
222 if (!(attr instanceof compiler_1.Attribute) || !(elem instanceof compiler_1.Element)) {
223 return [];
224 }
225 // TODO: Consider parsing the attrinute name to a proper AST instead of
226 // matching using regex. This is because the regexp would incorrectly identify
227 // bind parts for cases like [()|]
228 // ^ cursor is here
229 var binding = binding_utils_1.getBindingDescriptor(attr.name);
230 if (!binding) {
231 // This is a normal HTML attribute, not an Angular attribute.
232 return attributeCompletionsForElement(info, elem.name);
233 }
234 var results = [];
235 var ngAttrs = angularAttributes(info, elem.name);
236 switch (binding.kind) {
237 case binding_utils_1.ATTR.KW_MICROSYNTAX:
238 // template reference attribute: *attrName
239 results.push.apply(results, tslib_1.__spread(ngAttrs.templateRefs));
240 break;
241 case binding_utils_1.ATTR.KW_BIND:
242 case binding_utils_1.ATTR.IDENT_PROPERTY:
243 // property binding via bind- or []
244 results.push.apply(results, tslib_1.__spread(html_info_1.propertyNames(elem.name), ngAttrs.inputs));
245 break;
246 case binding_utils_1.ATTR.KW_ON:
247 case binding_utils_1.ATTR.IDENT_EVENT:
248 // event binding via on- or ()
249 results.push.apply(results, tslib_1.__spread(html_info_1.eventNames(elem.name), ngAttrs.outputs));
250 break;
251 case binding_utils_1.ATTR.KW_BINDON:
252 case binding_utils_1.ATTR.IDENT_BANANA_BOX:
253 // banana-in-a-box binding via bindon- or [()]
254 results.push.apply(results, tslib_1.__spread(ngAttrs.bananas));
255 break;
256 }
257 return results.map(function (name) {
258 return {
259 name: name,
260 kind: ng.CompletionKind.ATTRIBUTE,
261 sortText: name,
262 };
263 });
264 }
265 function attributeCompletionsForElement(info, elementName) {
266 var e_1, _a, e_2, _b;
267 var results = [];
268 if (info.template instanceof template_1.InlineTemplate) {
269 try {
270 // Provide HTML attributes completion only for inline templates
271 for (var _c = tslib_1.__values(html_info_1.attributeNames(elementName)), _d = _c.next(); !_d.done; _d = _c.next()) {
272 var name_1 = _d.value;
273 results.push({
274 name: name_1,
275 kind: ng.CompletionKind.HTML_ATTRIBUTE,
276 sortText: name_1,
277 });
278 }
279 }
280 catch (e_1_1) { e_1 = { error: e_1_1 }; }
281 finally {
282 try {
283 if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
284 }
285 finally { if (e_1) throw e_1.error; }
286 }
287 }
288 // Add Angular attributes
289 var ngAttrs = angularAttributes(info, elementName);
290 try {
291 for (var _e = tslib_1.__values(ngAttrs.others), _f = _e.next(); !_f.done; _f = _e.next()) {
292 var name_2 = _f.value;
293 results.push({
294 name: name_2,
295 kind: ng.CompletionKind.ATTRIBUTE,
296 sortText: name_2,
297 });
298 }
299 }
300 catch (e_2_1) { e_2 = { error: e_2_1 }; }
301 finally {
302 try {
303 if (_f && !_f.done && (_b = _e.return)) _b.call(_e);
304 }
305 finally { if (e_2) throw e_2.error; }
306 }
307 return results;
308 }
309 /**
310 * Provide completions to the RHS of an attribute, which is of the form
311 * LHS="RHS". The template path is computed from the specified `info` whereas
312 * the context is determined from the specified `htmlPath`.
313 * @param info Object that contains the template AST
314 * @param htmlPath Path to the HTML node
315 */
316 function attributeValueCompletions(info, htmlPath) {
317 // Find the corresponding Template AST path.
318 var templatePath = utils_1.findTemplateAstAt(info.templateAst, htmlPath.position);
319 var visitor = new ExpressionVisitor(info, htmlPath.position, function () {
320 var dinfo = utils_1.diagnosticInfoFromTemplateInfo(info);
321 return expression_diagnostics_1.getExpressionScope(dinfo, templatePath);
322 });
323 if (templatePath.tail instanceof compiler_1.AttrAst ||
324 templatePath.tail instanceof compiler_1.BoundElementPropertyAst ||
325 templatePath.tail instanceof compiler_1.BoundEventAst) {
326 templatePath.tail.visit(visitor, null);
327 return visitor.results;
328 }
329 // In order to provide accurate attribute value completion, we need to know
330 // what the LHS is, and construct the proper AST if it is missing.
331 var htmlAttr = htmlPath.tail;
332 var binding = binding_utils_1.getBindingDescriptor(htmlAttr.name);
333 if (binding && binding.kind === binding_utils_1.ATTR.KW_REF) {
334 var refAst = void 0;
335 var elemAst = void 0;
336 if (templatePath.tail instanceof compiler_1.ReferenceAst) {
337 refAst = templatePath.tail;
338 var parent_1 = templatePath.parentOf(refAst);
339 if (parent_1 instanceof compiler_1.ElementAst) {
340 elemAst = parent_1;
341 }
342 }
343 else if (templatePath.tail instanceof compiler_1.ElementAst) {
344 refAst = new compiler_1.ReferenceAst(htmlAttr.name, null, htmlAttr.value, htmlAttr.valueSpan);
345 elemAst = templatePath.tail;
346 }
347 if (refAst && elemAst) {
348 refAst.visit(visitor, elemAst);
349 }
350 }
351 else {
352 // HtmlAst contains the `Attribute` node, however the corresponding `AttrAst`
353 // node is missing from the TemplateAst.
354 var attrAst = new compiler_1.AttrAst(htmlAttr.name, htmlAttr.value, htmlAttr.valueSpan);
355 attrAst.visit(visitor, null);
356 }
357 return visitor.results;
358 }
359 function elementCompletions(info) {
360 var e_3, _a;
361 var results = tslib_1.__spread(ANGULAR_ELEMENTS);
362 if (info.template instanceof template_1.InlineTemplate) {
363 // Provide HTML elements completion only for inline templates
364 results.push.apply(results, tslib_1.__spread(HTML_ELEMENTS));
365 }
366 // Collect the elements referenced by the selectors
367 var components = new Set();
368 try {
369 for (var _b = tslib_1.__values(utils_1.getSelectors(info).selectors), _c = _b.next(); !_c.done; _c = _b.next()) {
370 var selector = _c.value;
371 var name_3 = selector.element;
372 if (name_3 && !components.has(name_3)) {
373 components.add(name_3);
374 results.push({
375 name: name_3,
376 kind: ng.CompletionKind.COMPONENT,
377 sortText: name_3,
378 });
379 }
380 }
381 }
382 catch (e_3_1) { e_3 = { error: e_3_1 }; }
383 finally {
384 try {
385 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
386 }
387 finally { if (e_3) throw e_3.error; }
388 }
389 return results;
390 }
391 // There is a special case of HTML where text that contains a unclosed tag is treated as
392 // text. For exaple '<h1> Some <a text </h1>' produces a text nodes inside of the H1
393 // element "Some <a text". We, however, want to treat this as if the user was requesting
394 // the attributes of an "a" element, not requesting completion in the a text element. This
395 // code checks for this case and returns element completions if it is detected or undefined
396 // if it is not.
397 function voidElementAttributeCompletions(info, path) {
398 var tail = path.tail;
399 if (tail instanceof compiler_1.Text) {
400 var match = tail.value.match(/<(\w(\w|\d|-)*:)?(\w(\w|\d|-)*)\s/);
401 // The position must be after the match, otherwise we are still in a place where elements
402 // are expected (such as `<|a` or `<a|`; we only want attributes for `<a |` or after).
403 if (match &&
404 path.position >= (match.index || 0) + match[0].length + tail.sourceSpan.start.offset) {
405 return attributeCompletionsForElement(info, match[3]);
406 }
407 }
408 return [];
409 }
410 var ExpressionVisitor = /** @class */ (function (_super) {
411 tslib_1.__extends(ExpressionVisitor, _super);
412 function ExpressionVisitor(info, position, getExpressionScope) {
413 var _this = _super.call(this) || this;
414 _this.info = info;
415 _this.position = position;
416 _this.getExpressionScope = getExpressionScope;
417 _this.completions = new Map();
418 return _this;
419 }
420 Object.defineProperty(ExpressionVisitor.prototype, "results", {
421 get: function () {
422 return Array.from(this.completions.values());
423 },
424 enumerable: false,
425 configurable: true
426 });
427 ExpressionVisitor.prototype.visitDirectiveProperty = function (ast) {
428 this.processExpressionCompletions(ast.value);
429 };
430 ExpressionVisitor.prototype.visitElementProperty = function (ast) {
431 this.processExpressionCompletions(ast.value);
432 };
433 ExpressionVisitor.prototype.visitEvent = function (ast) {
434 this.processExpressionCompletions(ast.handler);
435 };
436 ExpressionVisitor.prototype.visitElement = function () {
437 // no-op for now
438 };
439 ExpressionVisitor.prototype.visitAttr = function (ast) {
440 var binding = binding_utils_1.getBindingDescriptor(ast.name);
441 if (binding && binding.kind === binding_utils_1.ATTR.KW_MICROSYNTAX) {
442 // This a template binding given by micro syntax expression.
443 // First, verify the attribute consists of some binding we can give completions for.
444 // The sourceSpan of AttrAst points to the RHS of the attribute
445 var templateKey = binding.name;
446 var templateValue = ast.sourceSpan.toString();
447 var templateUrl = ast.sourceSpan.start.file.url;
448 // TODO(kyliau): We are unable to determine the absolute offset of the key
449 // but it is okay here, because we are only looking at the RHS of the attr
450 var absKeyOffset = 0;
451 var absValueOffset = ast.sourceSpan.start.offset;
452 var templateBindings = this.info.expressionParser.parseTemplateBindings(templateKey, templateValue, templateUrl, absKeyOffset, absValueOffset).templateBindings;
453 // Find the nearest template binding to the position.
454 var lastBindingEnd = templateBindings.length > 0 &&
455 templateBindings[templateBindings.length - 1].sourceSpan.end;
456 var normalizedPositionToBinding_1 = lastBindingEnd && this.position > lastBindingEnd ? lastBindingEnd : this.position;
457 var templateBinding = templateBindings.find(function (b) { return utils_1.inSpan(normalizedPositionToBinding_1, b.sourceSpan); });
458 if (!templateBinding) {
459 return;
460 }
461 this.microSyntaxInAttributeValue(ast, templateBinding);
462 }
463 else {
464 var expressionAst = this.info.expressionParser.parseBinding(ast.value, ast.sourceSpan.toString(), ast.sourceSpan.start.offset);
465 this.processExpressionCompletions(expressionAst);
466 }
467 };
468 ExpressionVisitor.prototype.visitReference = function (_ast, context) {
469 var _this = this;
470 context.directives.forEach(function (dir) {
471 var exportAs = dir.directive.exportAs;
472 if (exportAs) {
473 _this.completions.set(exportAs, { name: exportAs, kind: ng.CompletionKind.REFERENCE, sortText: exportAs });
474 }
475 });
476 };
477 ExpressionVisitor.prototype.visitBoundText = function (ast) {
478 if (utils_1.inSpan(this.position, ast.value.sourceSpan)) {
479 var completions = expressions_1.getExpressionCompletions(this.getExpressionScope(), ast.value, this.position, this.info.template);
480 if (completions) {
481 this.addSymbolsToCompletions(completions);
482 }
483 }
484 };
485 ExpressionVisitor.prototype.processExpressionCompletions = function (value) {
486 var symbols = expressions_1.getExpressionCompletions(this.getExpressionScope(), value, this.position, this.info.template);
487 if (symbols) {
488 this.addSymbolsToCompletions(symbols);
489 }
490 };
491 ExpressionVisitor.prototype.addSymbolsToCompletions = function (symbols) {
492 var e_4, _a;
493 try {
494 for (var symbols_1 = tslib_1.__values(symbols), symbols_1_1 = symbols_1.next(); !symbols_1_1.done; symbols_1_1 = symbols_1.next()) {
495 var s = symbols_1_1.value;
496 if (s.name.startsWith('__') || !s.public || this.completions.has(s.name)) {
497 continue;
498 }
499 // The pipe method should not include parentheses.
500 // e.g. {{ value_expression | slice : start [ : end ] }}
501 var shouldInsertParentheses = s.callable && s.kind !== ng.CompletionKind.PIPE;
502 this.completions.set(s.name, {
503 name: s.name,
504 kind: s.kind,
505 sortText: s.name,
506 insertText: shouldInsertParentheses ? s.name + "()" : s.name,
507 });
508 }
509 }
510 catch (e_4_1) { e_4 = { error: e_4_1 }; }
511 finally {
512 try {
513 if (symbols_1_1 && !symbols_1_1.done && (_a = symbols_1.return)) _a.call(symbols_1);
514 }
515 finally { if (e_4) throw e_4.error; }
516 }
517 };
518 /**
519 * This method handles the completions of attribute values for directives that
520 * support the microsyntax format. Examples are *ngFor and *ngIf.
521 * These directives allows declaration of "let" variables, adds context-specific
522 * symbols like $implicit, index, count, among other behaviors.
523 * For a complete description of such format, see
524 * https://angular.io/guide/structural-directives#the-asterisk--prefix
525 *
526 * @param attr descriptor for attribute name and value pair
527 * @param binding template binding for the expression in the attribute
528 */
529 ExpressionVisitor.prototype.microSyntaxInAttributeValue = function (attr, binding) {
530 var _a;
531 var key = attr.name.substring(1); // remove leading asterisk
532 // Find the selector - eg ngFor, ngIf, etc
533 var selectorInfo = utils_1.getSelectors(this.info);
534 var selector = selectorInfo.selectors.find(function (s) {
535 // attributes are listed in (attribute, value) pairs
536 for (var i = 0; i < s.attrs.length; i += 2) {
537 if (s.attrs[i] === key) {
538 return true;
539 }
540 }
541 });
542 if (!selector) {
543 return;
544 }
545 var valueRelativePosition = this.position - attr.sourceSpan.start.offset;
546 if (binding instanceof compiler_1.VariableBinding) {
547 // TODO(kyliau): With expression sourceSpan we shouldn't have to search
548 // the attribute value string anymore. Just check if position is in the
549 // expression source span.
550 var equalLocation = attr.value.indexOf('=');
551 if (equalLocation > 0 && valueRelativePosition > equalLocation) {
552 // We are after the '=' in a let clause. The valid values here are the members of the
553 // template reference's type parameter.
554 var directiveMetadata = selectorInfo.map.get(selector);
555 if (directiveMetadata) {
556 var contextTable = this.info.template.query.getTemplateContext(directiveMetadata.type.reference);
557 if (contextTable) {
558 // This adds symbols like $implicit, index, count, etc.
559 this.addSymbolsToCompletions(contextTable.values());
560 return;
561 }
562 }
563 }
564 }
565 else if (binding instanceof compiler_1.ExpressionBinding) {
566 if (utils_1.inSpan(this.position, (_a = binding.value) === null || _a === void 0 ? void 0 : _a.ast.sourceSpan)) {
567 this.processExpressionCompletions(binding.value.ast);
568 return;
569 }
570 else if (!binding.value && this.position > binding.key.span.end) {
571 // No expression is defined for the value of the key expression binding, but the cursor is
572 // in a location where the expression would be defined. This can happen in a case like
573 // let i of |
574 // ^-- cursor
575 // In this case, backfill the value to be an empty expression and retrieve completions.
576 this.processExpressionCompletions(new compiler_1.EmptyExpr(new compiler_1.ParseSpan(valueRelativePosition, valueRelativePosition), new compiler_1.AbsoluteSourceSpan(this.position, this.position)));
577 return;
578 }
579 }
580 };
581 return ExpressionVisitor;
582 }(compiler_1.NullTemplateVisitor));
583 /**
584 * Return all Angular-specific attributes for the element with `elementName`.
585 * @param info
586 * @param elementName
587 */
588 function angularAttributes(info, elementName) {
589 var e_5, _a, e_6, _b, e_7, _c, e_8, _d;
590 var _e = utils_1.getSelectors(info), selectors = _e.selectors, selectorMap = _e.map;
591 var templateRefs = new Set();
592 var inputs = new Set();
593 var outputs = new Set();
594 var bananas = new Set();
595 var others = new Set();
596 try {
597 for (var selectors_1 = tslib_1.__values(selectors), selectors_1_1 = selectors_1.next(); !selectors_1_1.done; selectors_1_1 = selectors_1.next()) {
598 var selector = selectors_1_1.value;
599 if (selector.element && selector.element !== elementName) {
600 continue;
601 }
602 var summary = selectorMap.get(selector);
603 var hasTemplateRef = utils_1.isStructuralDirective(summary.type);
604 // attributes are listed in (attribute, value) pairs
605 for (var i = 0; i < selector.attrs.length; i += 2) {
606 var attr = selector.attrs[i];
607 if (hasTemplateRef) {
608 templateRefs.add(attr);
609 }
610 else {
611 others.add(attr);
612 }
613 }
614 try {
615 for (var _f = (e_6 = void 0, tslib_1.__values(Object.values(summary.inputs))), _g = _f.next(); !_g.done; _g = _f.next()) {
616 var input = _g.value;
617 inputs.add(input);
618 }
619 }
620 catch (e_6_1) { e_6 = { error: e_6_1 }; }
621 finally {
622 try {
623 if (_g && !_g.done && (_b = _f.return)) _b.call(_f);
624 }
625 finally { if (e_6) throw e_6.error; }
626 }
627 try {
628 for (var _h = (e_7 = void 0, tslib_1.__values(Object.values(summary.outputs))), _j = _h.next(); !_j.done; _j = _h.next()) {
629 var output = _j.value;
630 outputs.add(output);
631 }
632 }
633 catch (e_7_1) { e_7 = { error: e_7_1 }; }
634 finally {
635 try {
636 if (_j && !_j.done && (_c = _h.return)) _c.call(_h);
637 }
638 finally { if (e_7) throw e_7.error; }
639 }
640 }
641 }
642 catch (e_5_1) { e_5 = { error: e_5_1 }; }
643 finally {
644 try {
645 if (selectors_1_1 && !selectors_1_1.done && (_a = selectors_1.return)) _a.call(selectors_1);
646 }
647 finally { if (e_5) throw e_5.error; }
648 }
649 try {
650 for (var inputs_1 = tslib_1.__values(inputs), inputs_1_1 = inputs_1.next(); !inputs_1_1.done; inputs_1_1 = inputs_1.next()) {
651 var name_4 = inputs_1_1.value;
652 // Add banana-in-a-box syntax
653 // https://angular.io/guide/template-syntax#two-way-binding-
654 if (outputs.has(name_4 + "Change")) {
655 bananas.add(name_4);
656 }
657 }
658 }
659 catch (e_8_1) { e_8 = { error: e_8_1 }; }
660 finally {
661 try {
662 if (inputs_1_1 && !inputs_1_1.done && (_d = inputs_1.return)) _d.call(inputs_1);
663 }
664 finally { if (e_8) throw e_8.error; }
665 }
666 return { templateRefs: templateRefs, inputs: inputs, outputs: outputs, bananas: bananas, others: others };
667 }
668});
669//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"completions.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/completions.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, AST, AstPath, AttrAst, Attribute, BoundDirectivePropertyAst, BoundElementPropertyAst, BoundEventAst, BoundTextAst, Element, ElementAst, EmptyExpr, ExpressionBinding, getHtmlTagDefinition, HtmlAstPath, Node as HtmlAst, NullTemplateVisitor, ParseSpan, ReferenceAst, TagContentType, TemplateBinding, Text, VariableBinding, Visitor} from '@angular/compiler';\nimport {$$, $_, isAsciiLetter, isDigit} from '@angular/compiler/src/chars';\n\nimport {ATTR, getBindingDescriptor} from './binding_utils';\nimport {getExpressionScope} from './expression_diagnostics';\nimport {getExpressionCompletions} from './expressions';\nimport {attributeNames, elementNames, eventNames, propertyNames} from './html_info';\nimport {InlineTemplate} from './template';\nimport * as ng from './types';\nimport {diagnosticInfoFromTemplateInfo, findTemplateAstAt, getPathToNodeAtPosition, getSelectors, inSpan, isStructuralDirective, spanOf} from './utils';\n\nconst HIDDEN_HTML_ELEMENTS: ReadonlySet<string> =\n    new Set(['html', 'script', 'noscript', 'base', 'body', 'title', 'head', 'link']);\nconst HTML_ELEMENTS: ReadonlyArray<ng.CompletionEntry> =\n    elementNames().filter(name => !HIDDEN_HTML_ELEMENTS.has(name)).map(name => {\n      return {\n        name,\n        kind: ng.CompletionKind.HTML_ELEMENT,\n        sortText: name,\n      };\n    });\nconst ANGULAR_ELEMENTS: ReadonlyArray<ng.CompletionEntry> = [\n  {\n    name: 'ng-container',\n    kind: ng.CompletionKind.ANGULAR_ELEMENT,\n    sortText: 'ng-container',\n  },\n  {\n    name: 'ng-content',\n    kind: ng.CompletionKind.ANGULAR_ELEMENT,\n    sortText: 'ng-content',\n  },\n  {\n    name: 'ng-template',\n    kind: ng.CompletionKind.ANGULAR_ELEMENT,\n    sortText: 'ng-template',\n  },\n];\n\nfunction isIdentifierPart(code: number) {\n  // Identifiers consist of alphanumeric characters, '_', or '$'.\n  return isAsciiLetter(code) || isDigit(code) || code == $$ || code == $_;\n}\n\n/**\n * Gets the span of word in a template that surrounds `position`. If there is no word around\n * `position`, nothing is returned.\n */\nfunction getBoundedWordSpan(\n    templateInfo: ng.AstResult, position: number, ast: HtmlAst|undefined): ts.TextSpan|undefined {\n  const {template} = templateInfo;\n  const templateSrc = template.source;\n\n  if (!templateSrc) return;\n\n  if (ast instanceof Element) {\n    // The HTML tag may include `-` (e.g. `app-root`),\n    // so use the HtmlAst to get the span before ayazhafiz refactor the code.\n    return {\n      start: templateInfo.template.span.start + ast.startSourceSpan.start.offset + 1,\n      length: ast.name.length\n    };\n  }\n\n  // TODO(ayazhafiz): A solution based on word expansion will always be expensive compared to one\n  // based on ASTs. Whatever penalty we incur is probably manageable for small-length (i.e. the\n  // majority of) identifiers, but the current solution involes a number of branchings and we can't\n  // control potentially very long identifiers. Consider moving to an AST-based solution once\n  // existing difficulties with AST spans are more clearly resolved (see #31898 for discussion of\n  // known problems, and #33091 for how they affect text replacement).\n  //\n  // `templatePosition` represents the right-bound location of a cursor in the template.\n  //    key.ent|ry\n  //           ^---- cursor, at position `r` is at.\n  // A cursor is not itself a character in the template; it has a left (lower) and right (upper)\n  // index bound that hugs the cursor itself.\n  let templatePosition = position - template.span.start;\n  // To perform word expansion, we want to determine the left and right indices that hug the cursor.\n  // There are three cases here.\n  let left, right;\n  if (templatePosition === 0) {\n    // 1. Case like\n    //      |rest of template\n    //    the cursor is at the start of the template, hugged only by the right side (0-index).\n    left = right = 0;\n  } else if (templatePosition === templateSrc.length) {\n    // 2. Case like\n    //      rest of template|\n    //    the cursor is at the end of the template, hugged only by the left side (last-index).\n    left = right = templateSrc.length - 1;\n  } else {\n    // 3. Case like\n    //      wo|rd\n    //    there is a clear left and right index.\n    left = templatePosition - 1;\n    right = templatePosition;\n  }\n\n  if (!isIdentifierPart(templateSrc.charCodeAt(left)) &&\n      !isIdentifierPart(templateSrc.charCodeAt(right))) {\n    // Case like\n    //         .|.\n    // left ---^ ^--- right\n    // There is no word here.\n    return;\n  }\n\n  // Expand on the left and right side until a word boundary is hit. Back up one expansion on both\n  // side to stay inside the word.\n  while (left >= 0 && isIdentifierPart(templateSrc.charCodeAt(left))) --left;\n  ++left;\n  while (right < templateSrc.length && isIdentifierPart(templateSrc.charCodeAt(right))) ++right;\n  --right;\n\n  const absoluteStartPosition = position - (templatePosition - left);\n  const length = right - left + 1;\n  return {start: absoluteStartPosition, length};\n}\n\nexport function getTemplateCompletions(\n    templateInfo: ng.AstResult, position: number): ng.CompletionEntry[] {\n  const {htmlAst, template} = templateInfo;\n  // Calculate the position relative to the start of the template. This is needed\n  // because spans in HTML AST are relative. Inline template has non-zero start position.\n  const templatePosition = position - template.span.start;\n  const htmlPath: HtmlAstPath = getPathToNodeAtPosition(htmlAst, templatePosition);\n  const mostSpecific = htmlPath.tail;\n  const visitor = new HtmlVisitor(templateInfo, htmlPath);\n  const results: ng.CompletionEntry[] = mostSpecific ?\n      mostSpecific.visit(visitor, null /* context */) :\n      elementCompletions(templateInfo);\n  const replacementSpan = getBoundedWordSpan(templateInfo, position, mostSpecific);\n  return results.map(entry => {\n    return {\n      ...entry,\n      replacementSpan,\n    };\n  });\n}\n\nclass HtmlVisitor implements Visitor {\n  /**\n   * Position relative to the start of the template.\n   */\n  private readonly relativePosition: number;\n  constructor(private readonly templateInfo: ng.AstResult, private readonly htmlPath: HtmlAstPath) {\n    this.relativePosition = htmlPath.position;\n  }\n  // Note that every visitor method must explicitly specify return type because\n  // Visitor returns `any` for all methods.\n  visitElement(ast: Element): ng.CompletionEntry[] {\n    const startTagSpan = spanOf(ast.sourceSpan);\n    const tagLen = ast.name.length;\n    // + 1 for the opening angle bracket\n    if (this.relativePosition <= startTagSpan.start + tagLen + 1) {\n      // If we are in the tag then return the element completions.\n      return elementCompletions(this.templateInfo);\n    }\n    if (this.relativePosition < startTagSpan.end) {\n      // We are in the attribute section of the element (but not in an attribute).\n      // Return the attribute completions.\n      return attributeCompletionsForElement(this.templateInfo, ast.name);\n    }\n    return [];\n  }\n  visitAttribute(ast: Attribute): ng.CompletionEntry[] {\n    // An attribute consists of two parts, LHS=\"RHS\".\n    // Determine if completions are requested for LHS or RHS\n    if (ast.valueSpan && inSpan(this.relativePosition, spanOf(ast.valueSpan))) {\n      // RHS completion\n      return attributeValueCompletions(this.templateInfo, this.htmlPath);\n    }\n    // LHS completion\n    return attributeCompletions(this.templateInfo, this.htmlPath);\n  }\n  visitText(): ng.CompletionEntry[] {\n    const templatePath = findTemplateAstAt(this.templateInfo.templateAst, this.relativePosition);\n    if (templatePath.tail instanceof BoundTextAst) {\n      // If we know that this is an interpolation then do not try other scenarios.\n      const visitor = new ExpressionVisitor(\n          this.templateInfo, this.relativePosition,\n          () =>\n              getExpressionScope(diagnosticInfoFromTemplateInfo(this.templateInfo), templatePath));\n      templatePath.tail?.visit(visitor, null);\n      return visitor.results;\n    }\n    // TODO(kyliau): Not sure if this check is really needed since we don't have\n    // any test cases for it.\n    const element = this.htmlPath.first(Element);\n    if (element &&\n        getHtmlTagDefinition(element.name).getContentType() !== TagContentType.PARSABLE_DATA) {\n      return [];\n    }\n    // This is to account for cases like <h1> <a> text | </h1> where the\n    // closest element has no closing tag and thus is considered plain text.\n    const results = voidElementAttributeCompletions(this.templateInfo, this.htmlPath);\n    if (results.length) {\n      return results;\n    }\n    return elementCompletions(this.templateInfo);\n  }\n  visitComment(): ng.CompletionEntry[] {\n    return [];\n  }\n  visitExpansion(): ng.CompletionEntry[] {\n    return [];\n  }\n  visitExpansionCase(): ng.CompletionEntry[] {\n    return [];\n  }\n}\n\nfunction attributeCompletions(info: ng.AstResult, path: AstPath<HtmlAst>): ng.CompletionEntry[] {\n  const attr = path.tail;\n  const elem = path.parentOf(attr);\n  if (!(attr instanceof Attribute) || !(elem instanceof Element)) {\n    return [];\n  }\n\n  // TODO: Consider parsing the attrinute name to a proper AST instead of\n  // matching using regex. This is because the regexp would incorrectly identify\n  // bind parts for cases like [()|]\n  //                              ^ cursor is here\n  const binding = getBindingDescriptor(attr.name);\n  if (!binding) {\n    // This is a normal HTML attribute, not an Angular attribute.\n    return attributeCompletionsForElement(info, elem.name);\n  }\n\n  const results: string[] = [];\n  const ngAttrs = angularAttributes(info, elem.name);\n  switch (binding.kind) {\n    case ATTR.KW_MICROSYNTAX:\n      // template reference attribute: *attrName\n      results.push(...ngAttrs.templateRefs);\n      break;\n\n    case ATTR.KW_BIND:\n    case ATTR.IDENT_PROPERTY:\n      // property binding via bind- or []\n      results.push(...propertyNames(elem.name), ...ngAttrs.inputs);\n      break;\n\n    case ATTR.KW_ON:\n    case ATTR.IDENT_EVENT:\n      // event binding via on- or ()\n      results.push(...eventNames(elem.name), ...ngAttrs.outputs);\n      break;\n\n    case ATTR.KW_BINDON:\n    case ATTR.IDENT_BANANA_BOX:\n      // banana-in-a-box binding via bindon- or [()]\n      results.push(...ngAttrs.bananas);\n      break;\n  }\n\n  return results.map(name => {\n    return {\n      name,\n      kind: ng.CompletionKind.ATTRIBUTE,\n      sortText: name,\n    };\n  });\n}\n\nfunction attributeCompletionsForElement(\n    info: ng.AstResult, elementName: string): ng.CompletionEntry[] {\n  const results: ng.CompletionEntry[] = [];\n\n  if (info.template instanceof InlineTemplate) {\n    // Provide HTML attributes completion only for inline templates\n    for (const name of attributeNames(elementName)) {\n      results.push({\n        name,\n        kind: ng.CompletionKind.HTML_ATTRIBUTE,\n        sortText: name,\n      });\n    }\n  }\n\n  // Add Angular attributes\n  const ngAttrs = angularAttributes(info, elementName);\n  for (const name of ngAttrs.others) {\n    results.push({\n      name,\n      kind: ng.CompletionKind.ATTRIBUTE,\n      sortText: name,\n    });\n  }\n\n  return results;\n}\n\n/**\n * Provide completions to the RHS of an attribute, which is of the form\n * LHS=\"RHS\". The template path is computed from the specified `info` whereas\n * the context is determined from the specified `htmlPath`.\n * @param info Object that contains the template AST\n * @param htmlPath Path to the HTML node\n */\nfunction attributeValueCompletions(\n    info: ng.AstResult, htmlPath: HtmlAstPath): ng.CompletionEntry[] {\n  // Find the corresponding Template AST path.\n  const templatePath = findTemplateAstAt(info.templateAst, htmlPath.position);\n  const visitor = new ExpressionVisitor(info, htmlPath.position, () => {\n    const dinfo = diagnosticInfoFromTemplateInfo(info);\n    return getExpressionScope(dinfo, templatePath);\n  });\n  if (templatePath.tail instanceof AttrAst ||\n      templatePath.tail instanceof BoundElementPropertyAst ||\n      templatePath.tail instanceof BoundEventAst) {\n    templatePath.tail.visit(visitor, null);\n    return visitor.results;\n  }\n  // In order to provide accurate attribute value completion, we need to know\n  // what the LHS is, and construct the proper AST if it is missing.\n  const htmlAttr = htmlPath.tail as Attribute;\n  const binding = getBindingDescriptor(htmlAttr.name);\n  if (binding && binding.kind === ATTR.KW_REF) {\n    let refAst: ReferenceAst|undefined;\n    let elemAst: ElementAst|undefined;\n    if (templatePath.tail instanceof ReferenceAst) {\n      refAst = templatePath.tail;\n      const parent = templatePath.parentOf(refAst);\n      if (parent instanceof ElementAst) {\n        elemAst = parent;\n      }\n    } else if (templatePath.tail instanceof ElementAst) {\n      refAst = new ReferenceAst(htmlAttr.name, null!, htmlAttr.value, htmlAttr.valueSpan!);\n      elemAst = templatePath.tail;\n    }\n    if (refAst && elemAst) {\n      refAst.visit(visitor, elemAst);\n    }\n  } else {\n    // HtmlAst contains the `Attribute` node, however the corresponding `AttrAst`\n    // node is missing from the TemplateAst.\n    const attrAst = new AttrAst(htmlAttr.name, htmlAttr.value, htmlAttr.valueSpan!);\n    attrAst.visit(visitor, null);\n  }\n  return visitor.results;\n}\n\nfunction elementCompletions(info: ng.AstResult): ng.CompletionEntry[] {\n  const results: ng.CompletionEntry[] = [...ANGULAR_ELEMENTS];\n\n  if (info.template instanceof InlineTemplate) {\n    // Provide HTML elements completion only for inline templates\n    results.push(...HTML_ELEMENTS);\n  }\n\n  // Collect the elements referenced by the selectors\n  const components = new Set<string>();\n  for (const selector of getSelectors(info).selectors) {\n    const name = selector.element;\n    if (name && !components.has(name)) {\n      components.add(name);\n      results.push({\n        name,\n        kind: ng.CompletionKind.COMPONENT,\n        sortText: name,\n      });\n    }\n  }\n\n  return results;\n}\n\n// There is a special case of HTML where text that contains a unclosed tag is treated as\n// text. For exaple '<h1> Some <a text </h1>' produces a text nodes inside of the H1\n// element \"Some <a text\". We, however, want to treat this as if the user was requesting\n// the attributes of an \"a\" element, not requesting completion in the a text element. This\n// code checks for this case and returns element completions if it is detected or undefined\n// if it is not.\nfunction voidElementAttributeCompletions(\n    info: ng.AstResult, path: AstPath<HtmlAst>): ng.CompletionEntry[] {\n  const tail = path.tail;\n  if (tail instanceof Text) {\n    const match = tail.value.match(/<(\\w(\\w|\\d|-)*:)?(\\w(\\w|\\d|-)*)\\s/);\n    // The position must be after the match, otherwise we are still in a place where elements\n    // are expected (such as `<|a` or `<a|`; we only want attributes for `<a |` or after).\n    if (match &&\n        path.position >= (match.index || 0) + match[0].length + tail.sourceSpan.start.offset) {\n      return attributeCompletionsForElement(info, match[3]);\n    }\n  }\n  return [];\n}\n\nclass ExpressionVisitor extends NullTemplateVisitor {\n  private readonly completions = new Map<string, ng.CompletionEntry>();\n\n  constructor(\n      private readonly info: ng.AstResult, private readonly position: number,\n      private readonly getExpressionScope: () => ng.SymbolTable) {\n    super();\n  }\n\n  get results(): ng.CompletionEntry[] {\n    return Array.from(this.completions.values());\n  }\n\n  visitDirectiveProperty(ast: BoundDirectivePropertyAst): void {\n    this.processExpressionCompletions(ast.value);\n  }\n\n  visitElementProperty(ast: BoundElementPropertyAst): void {\n    this.processExpressionCompletions(ast.value);\n  }\n\n  visitEvent(ast: BoundEventAst): void {\n    this.processExpressionCompletions(ast.handler);\n  }\n\n  visitElement(): void {\n    // no-op for now\n  }\n\n  visitAttr(ast: AttrAst) {\n    const binding = getBindingDescriptor(ast.name);\n    if (binding && binding.kind === ATTR.KW_MICROSYNTAX) {\n      // This a template binding given by micro syntax expression.\n      // First, verify the attribute consists of some binding we can give completions for.\n      // The sourceSpan of AttrAst points to the RHS of the attribute\n      const templateKey = binding.name;\n      const templateValue = ast.sourceSpan.toString();\n      const templateUrl = ast.sourceSpan.start.file.url;\n      // TODO(kyliau): We are unable to determine the absolute offset of the key\n      // but it is okay here, because we are only looking at the RHS of the attr\n      const absKeyOffset = 0;\n      const absValueOffset = ast.sourceSpan.start.offset;\n      const {templateBindings} = this.info.expressionParser.parseTemplateBindings(\n          templateKey, templateValue, templateUrl, absKeyOffset, absValueOffset);\n      // Find the nearest template binding to the position.\n      const lastBindingEnd = templateBindings.length > 0 &&\n          templateBindings[templateBindings.length - 1].sourceSpan.end;\n      const normalizedPositionToBinding =\n          lastBindingEnd && this.position > lastBindingEnd ? lastBindingEnd : this.position;\n      const templateBinding =\n          templateBindings.find(b => inSpan(normalizedPositionToBinding, b.sourceSpan));\n\n      if (!templateBinding) {\n        return;\n      }\n\n      this.microSyntaxInAttributeValue(ast, templateBinding);\n    } else {\n      const expressionAst = this.info.expressionParser.parseBinding(\n          ast.value, ast.sourceSpan.toString(), ast.sourceSpan.start.offset);\n      this.processExpressionCompletions(expressionAst);\n    }\n  }\n\n  visitReference(_ast: ReferenceAst, context: ElementAst) {\n    context.directives.forEach(dir => {\n      const {exportAs} = dir.directive;\n      if (exportAs) {\n        this.completions.set(\n            exportAs, {name: exportAs, kind: ng.CompletionKind.REFERENCE, sortText: exportAs});\n      }\n    });\n  }\n\n  visitBoundText(ast: BoundTextAst) {\n    if (inSpan(this.position, ast.value.sourceSpan)) {\n      const completions = getExpressionCompletions(\n          this.getExpressionScope(), ast.value, this.position, this.info.template);\n      if (completions) {\n        this.addSymbolsToCompletions(completions);\n      }\n    }\n  }\n\n  private processExpressionCompletions(value: AST) {\n    const symbols = getExpressionCompletions(\n        this.getExpressionScope(), value, this.position, this.info.template);\n    if (symbols) {\n      this.addSymbolsToCompletions(symbols);\n    }\n  }\n\n  private addSymbolsToCompletions(symbols: ng.Symbol[]) {\n    for (const s of symbols) {\n      if (s.name.startsWith('__') || !s.public || this.completions.has(s.name)) {\n        continue;\n      }\n\n      // The pipe method should not include parentheses.\n      // e.g. {{ value_expression | slice : start [ : end ] }}\n      const shouldInsertParentheses = s.callable && s.kind !== ng.CompletionKind.PIPE;\n      this.completions.set(s.name, {\n        name: s.name,\n        kind: s.kind as ng.CompletionKind,\n        sortText: s.name,\n        insertText: shouldInsertParentheses ? `${s.name}()` : s.name,\n      });\n    }\n  }\n\n  /**\n   * This method handles the completions of attribute values for directives that\n   * support the microsyntax format. Examples are *ngFor and *ngIf.\n   * These directives allows declaration of \"let\" variables, adds context-specific\n   * symbols like $implicit, index, count, among other behaviors.\n   * For a complete description of such format, see\n   * https://angular.io/guide/structural-directives#the-asterisk--prefix\n   *\n   * @param attr descriptor for attribute name and value pair\n   * @param binding template binding for the expression in the attribute\n   */\n  private microSyntaxInAttributeValue(attr: AttrAst, binding: TemplateBinding) {\n    const key = attr.name.substring(1);  // remove leading asterisk\n\n    // Find the selector - eg ngFor, ngIf, etc\n    const selectorInfo = getSelectors(this.info);\n    const selector = selectorInfo.selectors.find(s => {\n      // attributes are listed in (attribute, value) pairs\n      for (let i = 0; i < s.attrs.length; i += 2) {\n        if (s.attrs[i] === key) {\n          return true;\n        }\n      }\n    });\n\n    if (!selector) {\n      return;\n    }\n\n    const valueRelativePosition = this.position - attr.sourceSpan.start.offset;\n\n    if (binding instanceof VariableBinding) {\n      // TODO(kyliau): With expression sourceSpan we shouldn't have to search\n      // the attribute value string anymore. Just check if position is in the\n      // expression source span.\n      const equalLocation = attr.value.indexOf('=');\n      if (equalLocation > 0 && valueRelativePosition > equalLocation) {\n        // We are after the '=' in a let clause. The valid values here are the members of the\n        // template reference's type parameter.\n        const directiveMetadata = selectorInfo.map.get(selector);\n        if (directiveMetadata) {\n          const contextTable =\n              this.info.template.query.getTemplateContext(directiveMetadata.type.reference);\n          if (contextTable) {\n            // This adds symbols like $implicit, index, count, etc.\n            this.addSymbolsToCompletions(contextTable.values());\n            return;\n          }\n        }\n      }\n    } else if (binding instanceof ExpressionBinding) {\n      if (inSpan(this.position, binding.value?.ast.sourceSpan)) {\n        this.processExpressionCompletions(binding.value!.ast);\n        return;\n      } else if (!binding.value && this.position > binding.key.span.end) {\n        // No expression is defined for the value of the key expression binding, but the cursor is\n        // in a location where the expression would be defined. This can happen in a case like\n        //   let i of |\n        //            ^-- cursor\n        // In this case, backfill the value to be an empty expression and retrieve completions.\n        this.processExpressionCompletions(new EmptyExpr(\n            new ParseSpan(valueRelativePosition, valueRelativePosition),\n            new AbsoluteSourceSpan(this.position, this.position)));\n        return;\n      }\n    }\n  }\n}\n\ninterface AngularAttributes {\n  /**\n   * Attributes that support the * syntax. See https://angular.io/api/core/TemplateRef\n   */\n  templateRefs: Set<string>;\n  /**\n   * Attributes with the @Input annotation.\n   */\n  inputs: Set<string>;\n  /**\n   * Attributes with the @Output annotation.\n   */\n  outputs: Set<string>;\n  /**\n   * Attributes that support the [()] or bindon- syntax.\n   */\n  bananas: Set<string>;\n  /**\n   * General attributes that match the specified element.\n   */\n  others: Set<string>;\n}\n\n/**\n * Return all Angular-specific attributes for the element with `elementName`.\n * @param info\n * @param elementName\n */\nfunction angularAttributes(info: ng.AstResult, elementName: string): AngularAttributes {\n  const {selectors, map: selectorMap} = getSelectors(info);\n  const templateRefs = new Set<string>();\n  const inputs = new Set<string>();\n  const outputs = new Set<string>();\n  const bananas = new Set<string>();\n  const others = new Set<string>();\n  for (const selector of selectors) {\n    if (selector.element && selector.element !== elementName) {\n      continue;\n    }\n    const summary = selectorMap.get(selector)!;\n    const hasTemplateRef = isStructuralDirective(summary.type);\n    // attributes are listed in (attribute, value) pairs\n    for (let i = 0; i < selector.attrs.length; i += 2) {\n      const attr = selector.attrs[i];\n      if (hasTemplateRef) {\n        templateRefs.add(attr);\n      } else {\n        others.add(attr);\n      }\n    }\n    for (const input of Object.values(summary.inputs)) {\n      inputs.add(input);\n    }\n    for (const output of Object.values(summary.outputs)) {\n      outputs.add(output);\n    }\n  }\n  for (const name of inputs) {\n    // Add banana-in-a-box syntax\n    // https://angular.io/guide/template-syntax#two-way-binding-\n    if (outputs.has(`${name}Change`)) {\n      bananas.add(name);\n    }\n  }\n  return {templateRefs, inputs, outputs, bananas, others};\n}\n"]}
\No newline at end of file