@angular/language-service 9.0.0-rc.6 → 9.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -50,6 +50,32 @@
50
50
  sortText: 'ng-template',
51
51
  },
52
52
  ];
53
+ // This is adapted from packages/compiler/src/render3/r3_template_transform.ts
54
+ // to allow empty binding names.
55
+ var BIND_NAME_REGEXP = /^(?:(?:(?:(bind-)|(let-)|(ref-|#)|(on-)|(bindon-)|(@))(.*))|\[\(([^\)]*)\)\]|\[([^\]]*)\]|\(([^\)]*)\))$/;
56
+ var ATTR;
57
+ (function (ATTR) {
58
+ // Group 1 = "bind-"
59
+ ATTR[ATTR["KW_BIND_IDX"] = 1] = "KW_BIND_IDX";
60
+ // Group 2 = "let-"
61
+ ATTR[ATTR["KW_LET_IDX"] = 2] = "KW_LET_IDX";
62
+ // Group 3 = "ref-/#"
63
+ ATTR[ATTR["KW_REF_IDX"] = 3] = "KW_REF_IDX";
64
+ // Group 4 = "on-"
65
+ ATTR[ATTR["KW_ON_IDX"] = 4] = "KW_ON_IDX";
66
+ // Group 5 = "bindon-"
67
+ ATTR[ATTR["KW_BINDON_IDX"] = 5] = "KW_BINDON_IDX";
68
+ // Group 6 = "@"
69
+ ATTR[ATTR["KW_AT_IDX"] = 6] = "KW_AT_IDX";
70
+ // Group 7 = the identifier after "bind-", "let-", "ref-/#", "on-", "bindon-" or "@"
71
+ ATTR[ATTR["IDENT_KW_IDX"] = 7] = "IDENT_KW_IDX";
72
+ // Group 8 = identifier inside [()]
73
+ ATTR[ATTR["IDENT_BANANA_BOX_IDX"] = 8] = "IDENT_BANANA_BOX_IDX";
74
+ // Group 9 = identifier inside []
75
+ ATTR[ATTR["IDENT_PROPERTY_IDX"] = 9] = "IDENT_PROPERTY_IDX";
76
+ // Group 10 = identifier inside ()
77
+ ATTR[ATTR["IDENT_EVENT_IDX"] = 10] = "IDENT_EVENT_IDX";
78
+ })(ATTR || (ATTR = {}));
53
79
  function isIdentifierPart(code) {
54
80
  // Identifiers consist of alphanumeric characters, '_', or '$'.
55
81
  return chars_1.isAsciiLetter(code) || chars_1.isDigit(code) || code == chars_1.$$ || code == chars_1.$_;
@@ -123,7 +149,7 @@
123
149
  var htmlAst = templateInfo.htmlAst, template = templateInfo.template;
124
150
  // The templateNode starts at the delimiter character so we add 1 to skip it.
125
151
  var templatePosition = position - template.span.start;
126
- var path = compiler_1.findNode(htmlAst, templatePosition);
152
+ var path = utils_1.getPathToNodeAtPosition(htmlAst, templatePosition);
127
153
  var mostSpecific = path.tail;
128
154
  if (path.empty || !mostSpecific) {
129
155
  result = elementCompletions(templateInfo);
@@ -142,16 +168,19 @@
142
168
  else if (templatePosition < startTagSpan.end) {
143
169
  // We are in the attribute section of the element (but not in an attribute).
144
170
  // Return the attribute completions.
145
- result = attributeCompletions(templateInfo, path);
171
+ result = attributeCompletionsForElement(templateInfo, ast.name);
146
172
  }
147
173
  },
148
174
  visitAttribute: function (ast) {
149
- if (!ast.valueSpan || !utils_1.inSpan(templatePosition, utils_1.spanOf(ast.valueSpan))) {
150
- // We are in the name of an attribute. Show attribute completions.
151
- result = attributeCompletions(templateInfo, path);
175
+ // An attribute consists of two parts, LHS="RHS".
176
+ // Determine if completions are requested for LHS or RHS
177
+ if (ast.valueSpan && utils_1.inSpan(templatePosition, utils_1.spanOf(ast.valueSpan))) {
178
+ // RHS completion
179
+ result = attributeValueCompletions(templateInfo, path);
152
180
  }
153
- else if (ast.valueSpan && utils_1.inSpan(templatePosition, utils_1.spanOf(ast.valueSpan))) {
154
- result = attributeValueCompletions(templateInfo, templatePosition, ast);
181
+ else {
182
+ // LHS completion
183
+ result = attributeCompletions(templateInfo, path);
155
184
  }
156
185
  },
157
186
  visitText: function (ast) {
@@ -181,9 +210,9 @@
181
210
  }
182
211
  }
183
212
  },
184
- visitComment: function (ast) { },
185
- visitExpansion: function (ast) { },
186
- visitExpansionCase: function (ast) { }
213
+ visitComment: function () { },
214
+ visitExpansion: function () { },
215
+ visitExpansionCase: function () { }
187
216
  }, null);
188
217
  }
189
218
  var replacementSpan = getBoundedWordSpan(templateInfo, position);
@@ -193,20 +222,58 @@
193
222
  }
194
223
  exports.getTemplateCompletions = getTemplateCompletions;
195
224
  function attributeCompletions(info, path) {
196
- var item = path.tail instanceof compiler_1.Element ? path.tail : path.parentOf(path.tail);
197
- if (item instanceof compiler_1.Element) {
198
- return attributeCompletionsForElement(info, item.name);
225
+ var attr = path.tail;
226
+ var elem = path.parentOf(attr);
227
+ if (!(attr instanceof compiler_1.Attribute) || !(elem instanceof compiler_1.Element)) {
228
+ return [];
199
229
  }
200
- return [];
230
+ // TODO: Consider parsing the attrinute name to a proper AST instead of
231
+ // matching using regex. This is because the regexp would incorrectly identify
232
+ // bind parts for cases like [()|]
233
+ // ^ cursor is here
234
+ var bindParts = attr.name.match(BIND_NAME_REGEXP);
235
+ // TemplateRef starts with '*'. See https://angular.io/api/core/TemplateRef
236
+ var isTemplateRef = attr.name.startsWith('*');
237
+ var isBinding = bindParts !== null || isTemplateRef;
238
+ if (!isBinding) {
239
+ return attributeCompletionsForElement(info, elem.name);
240
+ }
241
+ var results = [];
242
+ var ngAttrs = angularAttributes(info, elem.name);
243
+ if (!bindParts) {
244
+ // If bindParts is null then this must be a TemplateRef.
245
+ results.push.apply(results, tslib_1.__spread(ngAttrs.templateRefs));
246
+ }
247
+ else if (bindParts[ATTR.KW_BIND_IDX] !== undefined ||
248
+ bindParts[ATTR.IDENT_PROPERTY_IDX] !== undefined) {
249
+ // property binding via bind- or []
250
+ results.push.apply(results, tslib_1.__spread(html_info_1.propertyNames(elem.name), ngAttrs.inputs));
251
+ }
252
+ else if (bindParts[ATTR.KW_ON_IDX] !== undefined || bindParts[ATTR.IDENT_EVENT_IDX] !== undefined) {
253
+ // event binding via on- or ()
254
+ results.push.apply(results, tslib_1.__spread(html_info_1.eventNames(elem.name), ngAttrs.outputs));
255
+ }
256
+ else if (bindParts[ATTR.KW_BINDON_IDX] !== undefined ||
257
+ bindParts[ATTR.IDENT_BANANA_BOX_IDX] !== undefined) {
258
+ // banana-in-a-box binding via bindon- or [()]
259
+ results.push.apply(results, tslib_1.__spread(ngAttrs.bananas));
260
+ }
261
+ return results.map(function (name) {
262
+ return {
263
+ name: name,
264
+ kind: ng.CompletionKind.ATTRIBUTE,
265
+ sortText: name,
266
+ };
267
+ });
201
268
  }
202
269
  function attributeCompletionsForElement(info, elementName) {
203
- var e_1, _a, e_2, _b, e_3, _c;
270
+ var e_1, _a, e_2, _b;
204
271
  var results = [];
205
272
  if (info.template instanceof template_1.InlineTemplate) {
206
273
  try {
207
274
  // Provide HTML attributes completion only for inline templates
208
- for (var _d = tslib_1.__values(html_info_1.attributeNames(elementName)), _e = _d.next(); !_e.done; _e = _d.next()) {
209
- var name_1 = _e.value;
275
+ for (var _c = tslib_1.__values(html_info_1.attributeNames(elementName)), _d = _c.next(); !_d.done; _d = _c.next()) {
276
+ var name_1 = _d.value;
210
277
  results.push({
211
278
  name: name_1,
212
279
  kind: ng.CompletionKind.HTML_ATTRIBUTE,
@@ -217,17 +284,18 @@
217
284
  catch (e_1_1) { e_1 = { error: e_1_1 }; }
218
285
  finally {
219
286
  try {
220
- if (_e && !_e.done && (_a = _d.return)) _a.call(_d);
287
+ if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
221
288
  }
222
289
  finally { if (e_1) throw e_1.error; }
223
290
  }
224
291
  }
292
+ // Add Angular attributes
293
+ var ngAttrs = angularAttributes(info, elementName);
225
294
  try {
226
- // Add html properties
227
- for (var _f = tslib_1.__values(html_info_1.propertyNames(elementName)), _g = _f.next(); !_g.done; _g = _f.next()) {
228
- var name_2 = _g.value;
295
+ for (var _e = tslib_1.__values(ngAttrs.others), _f = _e.next(); !_f.done; _f = _e.next()) {
296
+ var name_2 = _f.value;
229
297
  results.push({
230
- name: "[" + name_2 + "]",
298
+ name: name_2,
231
299
  kind: ng.CompletionKind.ATTRIBUTE,
232
300
  sortText: name_2,
233
301
  });
@@ -236,55 +304,64 @@
236
304
  catch (e_2_1) { e_2 = { error: e_2_1 }; }
237
305
  finally {
238
306
  try {
239
- if (_g && !_g.done && (_b = _f.return)) _b.call(_f);
307
+ if (_f && !_f.done && (_b = _e.return)) _b.call(_e);
240
308
  }
241
309
  finally { if (e_2) throw e_2.error; }
242
310
  }
243
- try {
244
- // Add html events
245
- for (var _h = tslib_1.__values(html_info_1.eventNames(elementName)), _j = _h.next(); !_j.done; _j = _h.next()) {
246
- var name_3 = _j.value;
247
- results.push({
248
- name: "(" + name_3 + ")",
249
- kind: ng.CompletionKind.ATTRIBUTE,
250
- sortText: name_3,
251
- });
252
- }
253
- }
254
- catch (e_3_1) { e_3 = { error: e_3_1 }; }
255
- finally {
256
- try {
257
- if (_j && !_j.done && (_c = _h.return)) _c.call(_h);
258
- }
259
- finally { if (e_3) throw e_3.error; }
260
- }
261
- // Add Angular attributes
262
- results.push.apply(results, tslib_1.__spread(angularAttributes(info, elementName)));
263
311
  return results;
264
312
  }
265
- function attributeValueCompletions(info, position, attr) {
266
- var path = utils_1.findTemplateAstAt(info.templateAst, position);
267
- if (!path.tail) {
268
- return [];
269
- }
270
- var dinfo = utils_1.diagnosticInfoFromTemplateInfo(info);
271
- var visitor = new ExpressionVisitor(info, position, function () { return expression_diagnostics_1.getExpressionScope(dinfo, path, false); }, attr);
272
- path.tail.visit(visitor, null);
273
- var results = visitor.results;
274
- if (results.length) {
275
- return results;
313
+ /**
314
+ * Provide completions to the RHS of an attribute, which is of the form
315
+ * LHS="RHS". The template path is computed from the specified `info` whereas
316
+ * the context is determined from the specified `htmlPath`.
317
+ * @param info Object that contains the template AST
318
+ * @param htmlPath Path to the HTML node
319
+ */
320
+ function attributeValueCompletions(info, htmlPath) {
321
+ // Find the corresponding Template AST path.
322
+ var templatePath = utils_1.findTemplateAstAt(info.templateAst, htmlPath.position);
323
+ var visitor = new ExpressionVisitor(info, htmlPath.position, function () {
324
+ var dinfo = utils_1.diagnosticInfoFromTemplateInfo(info);
325
+ return expression_diagnostics_1.getExpressionScope(dinfo, templatePath);
326
+ });
327
+ if (templatePath.tail instanceof compiler_1.AttrAst ||
328
+ templatePath.tail instanceof compiler_1.BoundElementPropertyAst ||
329
+ templatePath.tail instanceof compiler_1.BoundEventAst) {
330
+ templatePath.tail.visit(visitor, null);
331
+ return visitor.results;
332
+ }
333
+ // In order to provide accurate attribute value completion, we need to know
334
+ // what the LHS is, and construct the proper AST if it is missing.
335
+ var htmlAttr = htmlPath.tail;
336
+ var bindParts = htmlAttr.name.match(BIND_NAME_REGEXP);
337
+ if (bindParts && bindParts[ATTR.KW_REF_IDX] !== undefined) {
338
+ var refAst = void 0;
339
+ var elemAst = void 0;
340
+ if (templatePath.tail instanceof compiler_1.ReferenceAst) {
341
+ refAst = templatePath.tail;
342
+ var parent_1 = templatePath.parentOf(refAst);
343
+ if (parent_1 instanceof compiler_1.ElementAst) {
344
+ elemAst = parent_1;
345
+ }
346
+ }
347
+ else if (templatePath.tail instanceof compiler_1.ElementAst) {
348
+ refAst = new compiler_1.ReferenceAst(htmlAttr.name, null, htmlAttr.value, htmlAttr.valueSpan);
349
+ elemAst = templatePath.tail;
350
+ }
351
+ if (refAst && elemAst) {
352
+ refAst.visit(visitor, elemAst);
353
+ }
276
354
  }
277
- // Try allowing widening the path
278
- var widerPath = utils_1.findTemplateAstAt(info.templateAst, position, /* allowWidening */ true);
279
- if (widerPath.tail) {
280
- var widerVisitor = new ExpressionVisitor(info, position, function () { return expression_diagnostics_1.getExpressionScope(dinfo, widerPath, false); }, attr);
281
- widerPath.tail.visit(widerVisitor, null);
282
- return widerVisitor.results;
355
+ else {
356
+ // HtmlAst contains the `Attribute` node, however the corresponding `AttrAst`
357
+ // node is missing from the TemplateAst.
358
+ var attrAst = new compiler_1.AttrAst(htmlAttr.name, htmlAttr.value, htmlAttr.valueSpan);
359
+ attrAst.visit(visitor, null);
283
360
  }
284
- return results;
361
+ return visitor.results;
285
362
  }
286
363
  function elementCompletions(info) {
287
- var e_4, _a;
364
+ var e_3, _a;
288
365
  var results = tslib_1.__spread(ANGULAR_ELEMENTS);
289
366
  if (info.template instanceof template_1.InlineTemplate) {
290
367
  // Provide HTML elements completion only for inline templates
@@ -295,23 +372,23 @@
295
372
  try {
296
373
  for (var _b = tslib_1.__values(utils_1.getSelectors(info).selectors), _c = _b.next(); !_c.done; _c = _b.next()) {
297
374
  var selector = _c.value;
298
- var name_4 = selector.element;
299
- if (name_4 && !components.has(name_4)) {
300
- components.add(name_4);
375
+ var name_3 = selector.element;
376
+ if (name_3 && !components.has(name_3)) {
377
+ components.add(name_3);
301
378
  results.push({
302
- name: name_4,
379
+ name: name_3,
303
380
  kind: ng.CompletionKind.COMPONENT,
304
- sortText: name_4,
381
+ sortText: name_3,
305
382
  });
306
383
  }
307
384
  }
308
385
  }
309
- catch (e_4_1) { e_4 = { error: e_4_1 }; }
386
+ catch (e_3_1) { e_3 = { error: e_3_1 }; }
310
387
  finally {
311
388
  try {
312
389
  if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
313
390
  }
314
- finally { if (e_4) throw e_4.error; }
391
+ finally { if (e_3) throw e_3.error; }
315
392
  }
316
393
  return results;
317
394
  }
@@ -341,7 +418,7 @@
341
418
  if (!templatePath.tail) {
342
419
  return [];
343
420
  }
344
- var visitor = new ExpressionVisitor(info, position, function () { return expression_diagnostics_1.getExpressionScope(utils_1.diagnosticInfoFromTemplateInfo(info), templatePath, false); });
421
+ var visitor = new ExpressionVisitor(info, position, function () { return expression_diagnostics_1.getExpressionScope(utils_1.diagnosticInfoFromTemplateInfo(info), templatePath); });
345
422
  templatePath.tail.visit(visitor, null);
346
423
  return visitor.results;
347
424
  }
@@ -366,12 +443,11 @@
366
443
  }
367
444
  var ExpressionVisitor = /** @class */ (function (_super) {
368
445
  tslib_1.__extends(ExpressionVisitor, _super);
369
- function ExpressionVisitor(info, position, getExpressionScope, attr) {
446
+ function ExpressionVisitor(info, position, getExpressionScope) {
370
447
  var _this = _super.call(this) || this;
371
448
  _this.info = info;
372
449
  _this.position = position;
373
450
  _this.getExpressionScope = getExpressionScope;
374
- _this.attr = attr;
375
451
  _this.completions = new Map();
376
452
  return _this;
377
453
  }
@@ -381,39 +457,43 @@
381
457
  configurable: true
382
458
  });
383
459
  ExpressionVisitor.prototype.visitDirectiveProperty = function (ast) {
384
- this.addAttributeValuesToCompletions(ast.value);
460
+ this.processExpressionCompletions(ast.value);
385
461
  };
386
462
  ExpressionVisitor.prototype.visitElementProperty = function (ast) {
387
- this.addAttributeValuesToCompletions(ast.value);
463
+ this.processExpressionCompletions(ast.value);
388
464
  };
389
- ExpressionVisitor.prototype.visitEvent = function (ast) { this.addAttributeValuesToCompletions(ast.handler); };
390
- ExpressionVisitor.prototype.visitElement = function (ast) {
391
- if (!this.attr || !this.attr.valueSpan) {
392
- return;
393
- }
394
- // The attribute value is a template expression but the expression AST
395
- // was not produced when the TemplateAst was produced so do that here.
396
- var templateBindings = this.info.expressionParser.parseTemplateBindings(this.attr.name, this.attr.value, this.attr.sourceSpan.toString(), this.attr.sourceSpan.start.offset).templateBindings;
397
- // Find where the cursor is relative to the start of the attribute value.
398
- var valueRelativePosition = this.position - this.attr.valueSpan.start.offset;
399
- // Find the template binding that contains the position
400
- var binding = templateBindings.find(function (b) { return utils_1.inSpan(valueRelativePosition, b.span); });
401
- if (!binding) {
402
- return;
403
- }
404
- if (this.attr.name.startsWith('*')) {
405
- this.microSyntaxInAttributeValue(this.attr, binding);
465
+ ExpressionVisitor.prototype.visitEvent = function (ast) { this.processExpressionCompletions(ast.handler); };
466
+ ExpressionVisitor.prototype.visitElement = function () {
467
+ // no-op for now
468
+ };
469
+ ExpressionVisitor.prototype.visitAttr = function (ast) {
470
+ if (ast.name.startsWith('*')) {
471
+ // This a template binding given by micro syntax expression.
472
+ // First, verify the attribute consists of some binding we can give completions for.
473
+ var templateBindings = this.info.expressionParser.parseTemplateBindings(ast.name, ast.value, ast.sourceSpan.toString(), ast.sourceSpan.start.offset).templateBindings;
474
+ // Find where the cursor is relative to the start of the attribute value.
475
+ var valueRelativePosition_1 = this.position - ast.sourceSpan.start.offset;
476
+ // Find the template binding that contains the position.
477
+ var binding = templateBindings.find(function (b) { return utils_1.inSpan(valueRelativePosition_1, b.span); });
478
+ if (!binding) {
479
+ return;
480
+ }
481
+ this.microSyntaxInAttributeValue(ast, binding);
406
482
  }
407
- else if (valueRelativePosition >= 0) {
408
- // If the position is in the expression or after the key or there is no key,
409
- // return the expression completions
410
- var span = new compiler_1.ParseSpan(0, this.attr.value.length);
411
- var offset = ast.sourceSpan.start.offset;
412
- var receiver = new compiler_1.ImplicitReceiver(span, span.toAbsolute(offset));
413
- var expressionAst = new compiler_1.PropertyRead(span, span.toAbsolute(offset), receiver, '');
414
- this.addAttributeValuesToCompletions(expressionAst, valueRelativePosition);
483
+ else {
484
+ var expressionAst = this.info.expressionParser.parseBinding(ast.value, ast.sourceSpan.toString(), ast.sourceSpan.start.offset);
485
+ this.processExpressionCompletions(expressionAst);
415
486
  }
416
487
  };
488
+ ExpressionVisitor.prototype.visitReference = function (_ast, context) {
489
+ var _this = this;
490
+ context.directives.forEach(function (dir) {
491
+ var exportAs = dir.directive.exportAs;
492
+ if (exportAs) {
493
+ _this.completions.set(exportAs, { name: exportAs, kind: ng.CompletionKind.REFERENCE, sortText: exportAs });
494
+ }
495
+ });
496
+ };
417
497
  ExpressionVisitor.prototype.visitBoundText = function (ast) {
418
498
  if (utils_1.inSpan(this.position, ast.value.sourceSpan)) {
419
499
  var completions = expressions_1.getExpressionCompletions(this.getExpressionScope(), ast.value, this.position, this.info.template.query);
@@ -422,63 +502,39 @@
422
502
  }
423
503
  }
424
504
  };
425
- ExpressionVisitor.prototype.addAttributeValuesToCompletions = function (value, position) {
426
- var symbols = expressions_1.getExpressionCompletions(this.getExpressionScope(), value, position === undefined ? this.attributeValuePosition : position, this.info.template.query);
505
+ ExpressionVisitor.prototype.processExpressionCompletions = function (value) {
506
+ var symbols = expressions_1.getExpressionCompletions(this.getExpressionScope(), value, this.position, this.info.template.query);
427
507
  if (symbols) {
428
508
  this.addSymbolsToCompletions(symbols);
429
509
  }
430
510
  };
431
- ExpressionVisitor.prototype.addKeysToCompletions = function (selector, key) {
432
- if (key !== 'ngFor') {
433
- return;
434
- }
435
- this.completions.set('let', {
436
- name: 'let',
437
- kind: ng.CompletionKind.KEY,
438
- sortText: 'let',
439
- });
440
- if (selector.attrs.some(function (attr) { return attr === 'ngForOf'; })) {
441
- this.completions.set('of', {
442
- name: 'of',
443
- kind: ng.CompletionKind.KEY,
444
- sortText: 'of',
445
- });
446
- }
447
- };
448
511
  ExpressionVisitor.prototype.addSymbolsToCompletions = function (symbols) {
449
- var e_5, _a;
512
+ var e_4, _a;
450
513
  try {
451
514
  for (var symbols_1 = tslib_1.__values(symbols), symbols_1_1 = symbols_1.next(); !symbols_1_1.done; symbols_1_1 = symbols_1.next()) {
452
515
  var s = symbols_1_1.value;
453
516
  if (s.name.startsWith('__') || !s.public || this.completions.has(s.name)) {
454
517
  continue;
455
518
  }
519
+ // The pipe method should not include parentheses.
520
+ // e.g. {{ value_expression | slice : start [ : end ] }}
521
+ var shouldInsertParentheses = s.callable && s.kind !== ng.CompletionKind.PIPE;
456
522
  this.completions.set(s.name, {
457
523
  name: s.name,
458
524
  kind: s.kind,
459
525
  sortText: s.name,
460
- insertText: s.callable ? s.name + "()" : s.name,
526
+ insertText: shouldInsertParentheses ? s.name + "()" : s.name,
461
527
  });
462
528
  }
463
529
  }
464
- catch (e_5_1) { e_5 = { error: e_5_1 }; }
530
+ catch (e_4_1) { e_4 = { error: e_4_1 }; }
465
531
  finally {
466
532
  try {
467
533
  if (symbols_1_1 && !symbols_1_1.done && (_a = symbols_1.return)) _a.call(symbols_1);
468
534
  }
469
- finally { if (e_5) throw e_5.error; }
535
+ finally { if (e_4) throw e_4.error; }
470
536
  }
471
537
  };
472
- Object.defineProperty(ExpressionVisitor.prototype, "attributeValuePosition", {
473
- get: function () {
474
- if (this.attr && this.attr.valueSpan) {
475
- return this.position;
476
- }
477
- return 0;
478
- },
479
- enumerable: true,
480
- configurable: true
481
- });
482
538
  /**
483
539
  * This method handles the completions of attribute values for directives that
484
540
  * support the microsyntax format. Examples are *ngFor and *ngIf.
@@ -505,10 +561,10 @@
505
561
  if (!selector) {
506
562
  return;
507
563
  }
508
- var valueRelativePosition = this.position - attr.valueSpan.start.offset;
564
+ var valueRelativePosition = this.position - attr.sourceSpan.start.offset;
509
565
  if (binding.keyIsVar) {
510
566
  var equalLocation = attr.value.indexOf('=');
511
- if (equalLocation >= 0 && valueRelativePosition >= equalLocation) {
567
+ if (equalLocation > 0 && valueRelativePosition > equalLocation) {
512
568
  // We are after the '=' in a let clause. The valid values here are the members of the
513
569
  // template reference's type parameter.
514
570
  var directiveMetadata = selectorInfo.map.get(selector);
@@ -523,22 +579,36 @@
523
579
  }
524
580
  }
525
581
  if (binding.expression && utils_1.inSpan(valueRelativePosition, binding.expression.ast.span)) {
526
- this.addAttributeValuesToCompletions(binding.expression.ast, this.position);
582
+ this.processExpressionCompletions(binding.expression.ast);
527
583
  return;
528
584
  }
529
- this.addKeysToCompletions(selector, key);
585
+ // If the expression is incomplete, for example *ngFor="let x of |"
586
+ // binding.expression is null. We could still try to provide suggestions
587
+ // by looking for symbols that are in scope.
588
+ var KW_OF = ' of ';
589
+ var ofLocation = attr.value.indexOf(KW_OF);
590
+ if (ofLocation > 0 && valueRelativePosition >= ofLocation + KW_OF.length) {
591
+ var expressionAst = this.info.expressionParser.parseBinding(attr.value, attr.sourceSpan.toString(), attr.sourceSpan.start.offset);
592
+ this.processExpressionCompletions(expressionAst);
593
+ }
530
594
  };
531
595
  return ExpressionVisitor;
532
596
  }(compiler_1.NullTemplateVisitor));
533
597
  function getSourceText(template, span) {
534
598
  return template.source.substring(span.start, span.end);
535
599
  }
600
+ /**
601
+ * Return all Angular-specific attributes for the element with `elementName`.
602
+ * @param info
603
+ * @param elementName
604
+ */
536
605
  function angularAttributes(info, elementName) {
537
- var e_6, _a, e_7, _b, e_8, _c, e_9, _d, e_10, _e, e_11, _f, e_12, _g, e_13, _h;
538
- var _j = utils_1.getSelectors(info), selectors = _j.selectors, selectorMap = _j.map;
606
+ var e_5, _a, e_6, _b, e_7, _c, e_8, _d;
607
+ var _e = utils_1.getSelectors(info), selectors = _e.selectors, selectorMap = _e.map;
539
608
  var templateRefs = new Set();
540
609
  var inputs = new Set();
541
610
  var outputs = new Set();
611
+ var bananas = new Set();
542
612
  var others = new Set();
543
613
  try {
544
614
  for (var selectors_1 = tslib_1.__values(selectors), selectors_1_1 = selectors_1.next(); !selectors_1_1.done; selectors_1_1 = selectors_1.next()) {
@@ -547,140 +617,70 @@
547
617
  continue;
548
618
  }
549
619
  var summary = selectorMap.get(selector);
550
- try {
551
- for (var _k = (e_7 = void 0, tslib_1.__values(selector.attrs)), _l = _k.next(); !_l.done; _l = _k.next()) {
552
- var attr = _l.value;
553
- if (attr) {
554
- if (utils_1.hasTemplateReference(summary.type)) {
555
- templateRefs.add(attr);
556
- }
557
- else {
558
- others.add(attr);
559
- }
560
- }
620
+ var isTemplateRef = utils_1.hasTemplateReference(summary.type);
621
+ // attributes are listed in (attribute, value) pairs
622
+ for (var i = 0; i < selector.attrs.length; i += 2) {
623
+ var attr = selector.attrs[i];
624
+ if (isTemplateRef) {
625
+ templateRefs.add(attr);
561
626
  }
562
- }
563
- catch (e_7_1) { e_7 = { error: e_7_1 }; }
564
- finally {
565
- try {
566
- if (_l && !_l.done && (_b = _k.return)) _b.call(_k);
627
+ else {
628
+ others.add(attr);
567
629
  }
568
- finally { if (e_7) throw e_7.error; }
569
630
  }
570
631
  try {
571
- for (var _m = (e_8 = void 0, tslib_1.__values(Object.values(summary.inputs))), _o = _m.next(); !_o.done; _o = _m.next()) {
572
- var input = _o.value;
632
+ for (var _f = (e_6 = void 0, tslib_1.__values(Object.values(summary.inputs))), _g = _f.next(); !_g.done; _g = _f.next()) {
633
+ var input = _g.value;
573
634
  inputs.add(input);
574
635
  }
575
636
  }
576
- catch (e_8_1) { e_8 = { error: e_8_1 }; }
637
+ catch (e_6_1) { e_6 = { error: e_6_1 }; }
577
638
  finally {
578
639
  try {
579
- if (_o && !_o.done && (_c = _m.return)) _c.call(_m);
640
+ if (_g && !_g.done && (_b = _f.return)) _b.call(_f);
580
641
  }
581
- finally { if (e_8) throw e_8.error; }
642
+ finally { if (e_6) throw e_6.error; }
582
643
  }
583
644
  try {
584
- for (var _p = (e_9 = void 0, tslib_1.__values(Object.values(summary.outputs))), _q = _p.next(); !_q.done; _q = _p.next()) {
585
- var output = _q.value;
645
+ for (var _h = (e_7 = void 0, tslib_1.__values(Object.values(summary.outputs))), _j = _h.next(); !_j.done; _j = _h.next()) {
646
+ var output = _j.value;
586
647
  outputs.add(output);
587
648
  }
588
649
  }
589
- catch (e_9_1) { e_9 = { error: e_9_1 }; }
650
+ catch (e_7_1) { e_7 = { error: e_7_1 }; }
590
651
  finally {
591
652
  try {
592
- if (_q && !_q.done && (_d = _p.return)) _d.call(_p);
653
+ if (_j && !_j.done && (_c = _h.return)) _c.call(_h);
593
654
  }
594
- finally { if (e_9) throw e_9.error; }
655
+ finally { if (e_7) throw e_7.error; }
595
656
  }
596
657
  }
597
658
  }
598
- catch (e_6_1) { e_6 = { error: e_6_1 }; }
659
+ catch (e_5_1) { e_5 = { error: e_5_1 }; }
599
660
  finally {
600
661
  try {
601
662
  if (selectors_1_1 && !selectors_1_1.done && (_a = selectors_1.return)) _a.call(selectors_1);
602
663
  }
603
- finally { if (e_6) throw e_6.error; }
604
- }
605
- var results = [];
606
- try {
607
- for (var templateRefs_1 = tslib_1.__values(templateRefs), templateRefs_1_1 = templateRefs_1.next(); !templateRefs_1_1.done; templateRefs_1_1 = templateRefs_1.next()) {
608
- var name_5 = templateRefs_1_1.value;
609
- results.push({
610
- name: "*" + name_5,
611
- kind: ng.CompletionKind.ATTRIBUTE,
612
- sortText: name_5,
613
- });
614
- }
615
- }
616
- catch (e_10_1) { e_10 = { error: e_10_1 }; }
617
- finally {
618
- try {
619
- if (templateRefs_1_1 && !templateRefs_1_1.done && (_e = templateRefs_1.return)) _e.call(templateRefs_1);
620
- }
621
- finally { if (e_10) throw e_10.error; }
664
+ finally { if (e_5) throw e_5.error; }
622
665
  }
623
666
  try {
624
667
  for (var inputs_1 = tslib_1.__values(inputs), inputs_1_1 = inputs_1.next(); !inputs_1_1.done; inputs_1_1 = inputs_1.next()) {
625
- var name_6 = inputs_1_1.value;
626
- results.push({
627
- name: "[" + name_6 + "]",
628
- kind: ng.CompletionKind.ATTRIBUTE,
629
- sortText: name_6,
630
- });
668
+ var name_4 = inputs_1_1.value;
631
669
  // Add banana-in-a-box syntax
632
670
  // https://angular.io/guide/template-syntax#two-way-binding-
633
- if (outputs.has(name_6 + "Change")) {
634
- results.push({
635
- name: "[(" + name_6 + ")]",
636
- kind: ng.CompletionKind.ATTRIBUTE,
637
- sortText: name_6,
638
- });
671
+ if (outputs.has(name_4 + "Change")) {
672
+ bananas.add(name_4);
639
673
  }
640
674
  }
641
675
  }
642
- catch (e_11_1) { e_11 = { error: e_11_1 }; }
676
+ catch (e_8_1) { e_8 = { error: e_8_1 }; }
643
677
  finally {
644
678
  try {
645
- if (inputs_1_1 && !inputs_1_1.done && (_f = inputs_1.return)) _f.call(inputs_1);
646
- }
647
- finally { if (e_11) throw e_11.error; }
648
- }
649
- try {
650
- for (var outputs_1 = tslib_1.__values(outputs), outputs_1_1 = outputs_1.next(); !outputs_1_1.done; outputs_1_1 = outputs_1.next()) {
651
- var name_7 = outputs_1_1.value;
652
- results.push({
653
- name: "(" + name_7 + ")",
654
- kind: ng.CompletionKind.ATTRIBUTE,
655
- sortText: name_7,
656
- });
679
+ if (inputs_1_1 && !inputs_1_1.done && (_d = inputs_1.return)) _d.call(inputs_1);
657
680
  }
681
+ finally { if (e_8) throw e_8.error; }
658
682
  }
659
- catch (e_12_1) { e_12 = { error: e_12_1 }; }
660
- finally {
661
- try {
662
- if (outputs_1_1 && !outputs_1_1.done && (_g = outputs_1.return)) _g.call(outputs_1);
663
- }
664
- finally { if (e_12) throw e_12.error; }
665
- }
666
- try {
667
- for (var others_1 = tslib_1.__values(others), others_1_1 = others_1.next(); !others_1_1.done; others_1_1 = others_1.next()) {
668
- var name_8 = others_1_1.value;
669
- results.push({
670
- name: name_8,
671
- kind: ng.CompletionKind.ATTRIBUTE,
672
- sortText: name_8,
673
- });
674
- }
675
- }
676
- catch (e_13_1) { e_13 = { error: e_13_1 }; }
677
- finally {
678
- try {
679
- if (others_1_1 && !others_1_1.done && (_h = others_1.return)) _h.call(others_1);
680
- }
681
- finally { if (e_13) throw e_13.error; }
682
- }
683
- return results;
683
+ return { templateRefs: templateRefs, inputs: inputs, outputs: outputs, bananas: bananas, others: others };
684
684
  }
685
685
  });
686
- //# 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 Inc. 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, Attribute, BoundDirectivePropertyAst, BoundElementPropertyAst, BoundEventAst, BoundTextAst, CssSelector, Element, ElementAst, ImplicitReceiver, NAMED_ENTITIES, Node as HtmlAst, NullTemplateVisitor, ParseSpan, PropertyRead, TagContentType, TemplateBinding, Text, findNode, getHtmlTagDefinition} from '@angular/compiler';\nimport {$$, $_, isAsciiLetter, isDigit} from '@angular/compiler/src/chars';\n\nimport {AstResult} from './common';\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, getSelectors, hasTemplateReference, inSpan, 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(templateInfo: AstResult, position: number): ts.TextSpan|undefined {\n  const {template} = templateInfo;\n  const templateSrc = template.source;\n\n  if (!templateSrc) return;\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: AstResult, position: number): ng.CompletionEntry[] {\n  let result: ng.CompletionEntry[] = [];\n  const {htmlAst, template} = templateInfo;\n  // The templateNode starts at the delimiter character so we add 1 to skip it.\n  const templatePosition = position - template.span.start;\n  const path = findNode(htmlAst, templatePosition);\n  const mostSpecific = path.tail;\n  if (path.empty || !mostSpecific) {\n    result = elementCompletions(templateInfo);\n  } else {\n    const astPosition = templatePosition - mostSpecific.sourceSpan.start.offset;\n    mostSpecific.visit(\n        {\n          visitElement(ast) {\n            const startTagSpan = spanOf(ast.sourceSpan);\n            const tagLen = ast.name.length;\n            // + 1 for the opening angle bracket\n            if (templatePosition <= startTagSpan.start + tagLen + 1) {\n              // If we are in the tag then return the element completions.\n              result = elementCompletions(templateInfo);\n            } else if (templatePosition < startTagSpan.end) {\n              // We are in the attribute section of the element (but not in an attribute).\n              // Return the attribute completions.\n              result = attributeCompletions(templateInfo, path);\n            }\n          },\n          visitAttribute(ast) {\n            if (!ast.valueSpan || !inSpan(templatePosition, spanOf(ast.valueSpan))) {\n              // We are in the name of an attribute. Show attribute completions.\n              result = attributeCompletions(templateInfo, path);\n            } else if (ast.valueSpan && inSpan(templatePosition, spanOf(ast.valueSpan))) {\n              result = attributeValueCompletions(templateInfo, templatePosition, ast);\n            }\n          },\n          visitText(ast) {\n            // Check if we are in a entity.\n            result = entityCompletions(getSourceText(template, spanOf(ast)), astPosition);\n            if (result.length) return result;\n            result = interpolationCompletions(templateInfo, templatePosition);\n            if (result.length) return result;\n            const element = path.first(Element);\n            if (element) {\n              const definition = getHtmlTagDefinition(element.name);\n              if (definition.contentType === TagContentType.PARSABLE_DATA) {\n                result = voidElementAttributeCompletions(templateInfo, path);\n                if (!result.length) {\n                  // If the element can hold content, show element completions.\n                  result = elementCompletions(templateInfo);\n                }\n              }\n            } else {\n              // If no element container, implies parsable data so show elements.\n              result = voidElementAttributeCompletions(templateInfo, path);\n              if (!result.length) {\n                result = elementCompletions(templateInfo);\n              }\n            }\n          },\n          visitComment(ast) {},\n          visitExpansion(ast) {},\n          visitExpansionCase(ast) {}\n        },\n        null);\n  }\n\n  const replacementSpan = getBoundedWordSpan(templateInfo, position);\n  return result.map(entry => {\n    return {\n        ...entry, replacementSpan,\n    };\n  });\n}\n\nfunction attributeCompletions(info: AstResult, path: AstPath<HtmlAst>): ng.CompletionEntry[] {\n  const item = path.tail instanceof Element ? path.tail : path.parentOf(path.tail);\n  if (item instanceof Element) {\n    return attributeCompletionsForElement(info, item.name);\n  }\n  return [];\n}\n\nfunction attributeCompletionsForElement(\n    info: 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 html properties\n  for (const name of propertyNames(elementName)) {\n    results.push({\n      name: `[${name}]`,\n      kind: ng.CompletionKind.ATTRIBUTE,\n      sortText: name,\n    });\n  }\n\n  // Add html events\n  for (const name of eventNames(elementName)) {\n    results.push({\n      name: `(${name})`,\n      kind: ng.CompletionKind.ATTRIBUTE,\n      sortText: name,\n    });\n  }\n\n  // Add Angular attributes\n  results.push(...angularAttributes(info, elementName));\n\n  return results;\n}\n\nfunction attributeValueCompletions(\n    info: AstResult, position: number, attr: Attribute): ng.CompletionEntry[] {\n  const path = findTemplateAstAt(info.templateAst, position);\n  if (!path.tail) {\n    return [];\n  }\n  const dinfo = diagnosticInfoFromTemplateInfo(info);\n  const visitor =\n      new ExpressionVisitor(info, position, () => getExpressionScope(dinfo, path, false), attr);\n  path.tail.visit(visitor, null);\n  const {results} = visitor;\n  if (results.length) {\n    return results;\n  }\n  // Try allowing widening the path\n  const widerPath = findTemplateAstAt(info.templateAst, position, /* allowWidening */ true);\n  if (widerPath.tail) {\n    const widerVisitor = new ExpressionVisitor(\n        info, position, () => getExpressionScope(dinfo, widerPath, false), attr);\n    widerPath.tail.visit(widerVisitor, null);\n    return widerVisitor.results;\n  }\n  return results;\n}\n\nfunction elementCompletions(info: 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\nfunction entityCompletions(value: string, position: number): ng.CompletionEntry[] {\n  // Look for entity completions\n  const re = /&[A-Za-z]*;?(?!\\d)/g;\n  let found: RegExpExecArray|null;\n  let result: ng.CompletionEntry[] = [];\n  while (found = re.exec(value)) {\n    let len = found[0].length;\n    if (position >= found.index && position < (found.index + len)) {\n      result = Object.keys(NAMED_ENTITIES).map(name => {\n        return {\n          name: `&${name};`,\n          kind: ng.CompletionKind.ENTITY,\n          sortText: name,\n        };\n      });\n      break;\n    }\n  }\n  return result;\n}\n\nfunction interpolationCompletions(info: AstResult, position: number): ng.CompletionEntry[] {\n  // Look for an interpolation in at the position.\n  const templatePath = findTemplateAstAt(info.templateAst, position);\n  if (!templatePath.tail) {\n    return [];\n  }\n  const visitor = new ExpressionVisitor(\n      info, position,\n      () => getExpressionScope(diagnosticInfoFromTemplateInfo(info), templatePath, false));\n  templatePath.tail.visit(visitor, null);\n  return visitor.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: 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: AstResult, private readonly position: number,\n      private readonly getExpressionScope: () => ng.SymbolTable,\n      private readonly attr?: Attribute) {\n    super();\n  }\n\n  get results(): ng.CompletionEntry[] { return Array.from(this.completions.values()); }\n\n  visitDirectiveProperty(ast: BoundDirectivePropertyAst): void {\n    this.addAttributeValuesToCompletions(ast.value);\n  }\n\n  visitElementProperty(ast: BoundElementPropertyAst): void {\n    this.addAttributeValuesToCompletions(ast.value);\n  }\n\n  visitEvent(ast: BoundEventAst): void { this.addAttributeValuesToCompletions(ast.handler); }\n\n  visitElement(ast: ElementAst): void {\n    if (!this.attr || !this.attr.valueSpan) {\n      return;\n    }\n\n    // The attribute value is a template expression but the expression AST\n    // was not produced when the TemplateAst was produced so do that here.\n    const {templateBindings} = this.info.expressionParser.parseTemplateBindings(\n        this.attr.name, this.attr.value, this.attr.sourceSpan.toString(),\n        this.attr.sourceSpan.start.offset);\n\n    // Find where the cursor is relative to the start of the attribute value.\n    const valueRelativePosition = this.position - this.attr.valueSpan.start.offset;\n    // Find the template binding that contains the position\n    const binding = templateBindings.find(b => inSpan(valueRelativePosition, b.span));\n\n    if (!binding) {\n      return;\n    }\n\n    if (this.attr.name.startsWith('*')) {\n      this.microSyntaxInAttributeValue(this.attr, binding);\n    } else if (valueRelativePosition >= 0) {\n      // If the position is in the expression or after the key or there is no key,\n      // return the expression completions\n      const span = new ParseSpan(0, this.attr.value.length);\n      const offset = ast.sourceSpan.start.offset;\n      const receiver = new ImplicitReceiver(span, span.toAbsolute(offset));\n      const expressionAst = new PropertyRead(span, span.toAbsolute(offset), receiver, '');\n      this.addAttributeValuesToCompletions(expressionAst, valueRelativePosition);\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.query);\n      if (completions) {\n        this.addSymbolsToCompletions(completions);\n      }\n    }\n  }\n\n  private addAttributeValuesToCompletions(value: AST, position?: number) {\n    const symbols = getExpressionCompletions(\n        this.getExpressionScope(), value,\n        position === undefined ? this.attributeValuePosition : position, this.info.template.query);\n    if (symbols) {\n      this.addSymbolsToCompletions(symbols);\n    }\n  }\n\n  private addKeysToCompletions(selector: CssSelector, key: string) {\n    if (key !== 'ngFor') {\n      return;\n    }\n    this.completions.set('let', {\n      name: 'let',\n      kind: ng.CompletionKind.KEY,\n      sortText: 'let',\n    });\n    if (selector.attrs.some(attr => attr === 'ngForOf')) {\n      this.completions.set('of', {\n        name: 'of',\n        kind: ng.CompletionKind.KEY,\n        sortText: 'of',\n      });\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      this.completions.set(s.name, {\n        name: s.name,\n        kind: s.kind as ng.CompletionKind,\n        sortText: s.name,\n        insertText: s.callable ? `${s.name}()` : s.name,\n      });\n    }\n  }\n\n  private get attributeValuePosition() {\n    if (this.attr && this.attr.valueSpan) {\n      return this.position;\n    }\n    return 0;\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: Attribute, 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.valueSpan !.start.offset;\n\n    if (binding.keyIsVar) {\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    }\n\n    if (binding.expression && inSpan(valueRelativePosition, binding.expression.ast.span)) {\n      this.addAttributeValuesToCompletions(binding.expression.ast, this.position);\n      return;\n    }\n\n    this.addKeysToCompletions(selector, key);\n  }\n}\n\nfunction getSourceText(template: ng.TemplateSource, span: ng.Span): string {\n  return template.source.substring(span.start, span.end);\n}\n\nfunction angularAttributes(info: AstResult, elementName: string): ng.CompletionEntry[] {\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 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    for (const attr of selector.attrs) {\n      if (attr) {\n        if (hasTemplateReference(summary.type)) {\n          templateRefs.add(attr);\n        } else {\n          others.add(attr);\n        }\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\n  const results: ng.CompletionEntry[] = [];\n  for (const name of templateRefs) {\n    results.push({\n      name: `*${name}`,\n      kind: ng.CompletionKind.ATTRIBUTE,\n      sortText: name,\n    });\n  }\n  for (const name of inputs) {\n    results.push({\n      name: `[${name}]`,\n      kind: ng.CompletionKind.ATTRIBUTE,\n      sortText: name,\n    });\n    // Add banana-in-a-box syntax\n    // https://angular.io/guide/template-syntax#two-way-binding-\n    if (outputs.has(`${name}Change`)) {\n      results.push({\n        name: `[(${name})]`,\n        kind: ng.CompletionKind.ATTRIBUTE,\n        sortText: name,\n      });\n    }\n  }\n  for (const name of outputs) {\n    results.push({\n      name: `(${name})`,\n      kind: ng.CompletionKind.ATTRIBUTE,\n      sortText: name,\n    });\n  }\n  for (const name of others) {\n    results.push({\n      name,\n      kind: ng.CompletionKind.ATTRIBUTE,\n      sortText: name,\n    });\n  }\n  return results;\n}\n"]}
686
+ //# 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 Inc. 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, AttrAst, Attribute, BoundDirectivePropertyAst, BoundElementPropertyAst, BoundEventAst, BoundTextAst, Element, ElementAst, HtmlAstPath, NAMED_ENTITIES, Node as HtmlAst, NullTemplateVisitor, ReferenceAst, TagContentType, TemplateBinding, Text, getHtmlTagDefinition} from '@angular/compiler';\nimport {$$, $_, isAsciiLetter, isDigit} from '@angular/compiler/src/chars';\n\nimport {AstResult} from './common';\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, hasTemplateReference, inSpan, 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\n// This is adapted from packages/compiler/src/render3/r3_template_transform.ts\n// to allow empty binding names.\nconst BIND_NAME_REGEXP =\n    /^(?:(?:(?:(bind-)|(let-)|(ref-|#)|(on-)|(bindon-)|(@))(.*))|\\[\\(([^\\)]*)\\)\\]|\\[([^\\]]*)\\]|\\(([^\\)]*)\\))$/;\nenum ATTR {\n  // Group 1 = \"bind-\"\n  KW_BIND_IDX = 1,\n  // Group 2 = \"let-\"\n  KW_LET_IDX = 2,\n  // Group 3 = \"ref-/#\"\n  KW_REF_IDX = 3,\n  // Group 4 = \"on-\"\n  KW_ON_IDX = 4,\n  // Group 5 = \"bindon-\"\n  KW_BINDON_IDX = 5,\n  // Group 6 = \"@\"\n  KW_AT_IDX = 6,\n  // Group 7 = the identifier after \"bind-\", \"let-\", \"ref-/#\", \"on-\", \"bindon-\" or \"@\"\n  IDENT_KW_IDX = 7,\n  // Group 8 = identifier inside [()]\n  IDENT_BANANA_BOX_IDX = 8,\n  // Group 9 = identifier inside []\n  IDENT_PROPERTY_IDX = 9,\n  // Group 10 = identifier inside ()\n  IDENT_EVENT_IDX = 10,\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(templateInfo: AstResult, position: number): ts.TextSpan|undefined {\n  const {template} = templateInfo;\n  const templateSrc = template.source;\n\n  if (!templateSrc) return;\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: AstResult, position: number): ng.CompletionEntry[] {\n  let result: ng.CompletionEntry[] = [];\n  const {htmlAst, template} = templateInfo;\n  // The templateNode starts at the delimiter character so we add 1 to skip it.\n  const templatePosition = position - template.span.start;\n  const path = getPathToNodeAtPosition(htmlAst, templatePosition);\n  const mostSpecific = path.tail;\n  if (path.empty || !mostSpecific) {\n    result = elementCompletions(templateInfo);\n  } else {\n    const astPosition = templatePosition - mostSpecific.sourceSpan.start.offset;\n    mostSpecific.visit(\n        {\n          visitElement(ast) {\n            const startTagSpan = spanOf(ast.sourceSpan);\n            const tagLen = ast.name.length;\n            // + 1 for the opening angle bracket\n            if (templatePosition <= startTagSpan.start + tagLen + 1) {\n              // If we are in the tag then return the element completions.\n              result = elementCompletions(templateInfo);\n            } else if (templatePosition < startTagSpan.end) {\n              // We are in the attribute section of the element (but not in an attribute).\n              // Return the attribute completions.\n              result = attributeCompletionsForElement(templateInfo, ast.name);\n            }\n          },\n          visitAttribute(ast: Attribute) {\n            // An attribute consists of two parts, LHS=\"RHS\".\n            // Determine if completions are requested for LHS or RHS\n            if (ast.valueSpan && inSpan(templatePosition, spanOf(ast.valueSpan))) {\n              // RHS completion\n              result = attributeValueCompletions(templateInfo, path);\n            } else {\n              // LHS completion\n              result = attributeCompletions(templateInfo, path);\n            }\n          },\n          visitText(ast) {\n            // Check if we are in a entity.\n            result = entityCompletions(getSourceText(template, spanOf(ast)), astPosition);\n            if (result.length) return result;\n            result = interpolationCompletions(templateInfo, templatePosition);\n            if (result.length) return result;\n            const element = path.first(Element);\n            if (element) {\n              const definition = getHtmlTagDefinition(element.name);\n              if (definition.contentType === TagContentType.PARSABLE_DATA) {\n                result = voidElementAttributeCompletions(templateInfo, path);\n                if (!result.length) {\n                  // If the element can hold content, show element completions.\n                  result = elementCompletions(templateInfo);\n                }\n              }\n            } else {\n              // If no element container, implies parsable data so show elements.\n              result = voidElementAttributeCompletions(templateInfo, path);\n              if (!result.length) {\n                result = elementCompletions(templateInfo);\n              }\n            }\n          },\n          visitComment() {},\n          visitExpansion() {},\n          visitExpansionCase() {}\n        },\n        null);\n  }\n\n  const replacementSpan = getBoundedWordSpan(templateInfo, position);\n  return result.map(entry => {\n    return {\n        ...entry, replacementSpan,\n    };\n  });\n}\n\nfunction attributeCompletions(info: 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 bindParts = attr.name.match(BIND_NAME_REGEXP);\n  // TemplateRef starts with '*'. See https://angular.io/api/core/TemplateRef\n  const isTemplateRef = attr.name.startsWith('*');\n  const isBinding = bindParts !== null || isTemplateRef;\n\n  if (!isBinding) {\n    return attributeCompletionsForElement(info, elem.name);\n  }\n\n  const results: string[] = [];\n  const ngAttrs = angularAttributes(info, elem.name);\n  if (!bindParts) {\n    // If bindParts is null then this must be a TemplateRef.\n    results.push(...ngAttrs.templateRefs);\n  } else if (\n      bindParts[ATTR.KW_BIND_IDX] !== undefined ||\n      bindParts[ATTR.IDENT_PROPERTY_IDX] !== undefined) {\n    // property binding via bind- or []\n    results.push(...propertyNames(elem.name), ...ngAttrs.inputs);\n  } else if (\n      bindParts[ATTR.KW_ON_IDX] !== undefined || bindParts[ATTR.IDENT_EVENT_IDX] !== undefined) {\n    // event binding via on- or ()\n    results.push(...eventNames(elem.name), ...ngAttrs.outputs);\n  } else if (\n      bindParts[ATTR.KW_BINDON_IDX] !== undefined ||\n      bindParts[ATTR.IDENT_BANANA_BOX_IDX] !== undefined) {\n    // banana-in-a-box binding via bindon- or [()]\n    results.push(...ngAttrs.bananas);\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: 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(info: 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 bindParts = htmlAttr.name.match(BIND_NAME_REGEXP);\n  if (bindParts && bindParts[ATTR.KW_REF_IDX] !== undefined) {\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: 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\nfunction entityCompletions(value: string, position: number): ng.CompletionEntry[] {\n  // Look for entity completions\n  const re = /&[A-Za-z]*;?(?!\\d)/g;\n  let found: RegExpExecArray|null;\n  let result: ng.CompletionEntry[] = [];\n  while (found = re.exec(value)) {\n    let len = found[0].length;\n    if (position >= found.index && position < (found.index + len)) {\n      result = Object.keys(NAMED_ENTITIES).map(name => {\n        return {\n          name: `&${name};`,\n          kind: ng.CompletionKind.ENTITY,\n          sortText: name,\n        };\n      });\n      break;\n    }\n  }\n  return result;\n}\n\nfunction interpolationCompletions(info: AstResult, position: number): ng.CompletionEntry[] {\n  // Look for an interpolation in at the position.\n  const templatePath = findTemplateAstAt(info.templateAst, position);\n  if (!templatePath.tail) {\n    return [];\n  }\n  const visitor = new ExpressionVisitor(\n      info, position, () => getExpressionScope(diagnosticInfoFromTemplateInfo(info), templatePath));\n  templatePath.tail.visit(visitor, null);\n  return visitor.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: 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: AstResult, private readonly position: number,\n      private readonly getExpressionScope: () => ng.SymbolTable) {\n    super();\n  }\n\n  get results(): ng.CompletionEntry[] { return Array.from(this.completions.values()); }\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 { this.processExpressionCompletions(ast.handler); }\n\n  visitElement(): void {\n    // no-op for now\n  }\n\n  visitAttr(ast: AttrAst) {\n    if (ast.name.startsWith('*')) {\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      const {templateBindings} = this.info.expressionParser.parseTemplateBindings(\n          ast.name, ast.value, ast.sourceSpan.toString(), ast.sourceSpan.start.offset);\n      // Find where the cursor is relative to the start of the attribute value.\n      const valueRelativePosition = this.position - ast.sourceSpan.start.offset;\n      // Find the template binding that contains the position.\n      const binding = templateBindings.find(b => inSpan(valueRelativePosition, b.span));\n\n      if (!binding) {\n        return;\n      }\n\n      this.microSyntaxInAttributeValue(ast, binding);\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.query);\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.query);\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.keyIsVar) {\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    }\n\n    if (binding.expression && inSpan(valueRelativePosition, binding.expression.ast.span)) {\n      this.processExpressionCompletions(binding.expression.ast);\n      return;\n    }\n\n    // If the expression is incomplete, for example *ngFor=\"let x of |\"\n    // binding.expression is null. We could still try to provide suggestions\n    // by looking for symbols that are in scope.\n    const KW_OF = ' of ';\n    const ofLocation = attr.value.indexOf(KW_OF);\n    if (ofLocation > 0 && valueRelativePosition >= ofLocation + KW_OF.length) {\n      const expressionAst = this.info.expressionParser.parseBinding(\n          attr.value, attr.sourceSpan.toString(), attr.sourceSpan.start.offset);\n      this.processExpressionCompletions(expressionAst);\n    }\n  }\n}\n\nfunction getSourceText(template: ng.TemplateSource, span: ng.Span): string {\n  return template.source.substring(span.start, span.end);\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: 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 isTemplateRef = hasTemplateReference(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 (isTemplateRef) {\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"]}