@angular/language-service 9.0.0-rc.7 → 9.0.1
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.
- package/bundles/language-service.umd.js +2086 -2500
- package/package.json +1 -1
- package/src/completions.js +100 -48
- package/src/definitions.js +61 -24
- package/src/diagnostics.d.ts +0 -7
- package/src/diagnostics.js +1 -37
- package/src/expression_diagnostics.d.ts +8 -4
- package/src/expression_diagnostics.js +64 -52
- package/src/expression_type.d.ts +6 -13
- package/src/expression_type.js +74 -84
- package/src/expressions.js +8 -2
- package/src/global_symbols.js +6 -1
- package/src/hover.d.ts +7 -6
- package/src/hover.js +71 -76
- package/src/html_info.d.ts +1 -1
- package/src/language_service.d.ts +12 -2
- package/src/language_service.js +16 -20
- package/src/locate_symbol.d.ts +9 -8
- package/src/locate_symbol.js +245 -182
- package/src/symbols.d.ts +9 -0
- package/src/symbols.js +1 -1
- package/src/template.d.ts +6 -6
- package/src/template.js +4 -2
- package/src/ts_plugin.js +6 -6
- package/src/types.d.ts +6 -40
- package/src/types.js +1 -1
- package/src/typescript_host.d.ts +5 -14
- package/src/typescript_host.js +7 -19
- package/src/typescript_symbols.d.ts +1 -1
- package/src/typescript_symbols.js +85 -63
- package/src/utils.d.ts +22 -8
- package/src/utils.js +63 -8
- package/src/version.js +1 -1
package/package.json
CHANGED
package/src/completions.js
CHANGED
|
@@ -172,12 +172,15 @@
|
|
|
172
172
|
}
|
|
173
173
|
},
|
|
174
174
|
visitAttribute: function (ast) {
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
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);
|
|
178
180
|
}
|
|
179
|
-
else
|
|
180
|
-
|
|
181
|
+
else {
|
|
182
|
+
// LHS completion
|
|
183
|
+
result = attributeCompletions(templateInfo, path);
|
|
181
184
|
}
|
|
182
185
|
},
|
|
183
186
|
visitText: function (ast) {
|
|
@@ -207,9 +210,9 @@
|
|
|
207
210
|
}
|
|
208
211
|
}
|
|
209
212
|
},
|
|
210
|
-
visitComment: function (
|
|
211
|
-
visitExpansion: function (
|
|
212
|
-
visitExpansionCase: function (
|
|
213
|
+
visitComment: function () { },
|
|
214
|
+
visitExpansion: function () { },
|
|
215
|
+
visitExpansionCase: function () { }
|
|
213
216
|
}, null);
|
|
214
217
|
}
|
|
215
218
|
var replacementSpan = getBoundedWordSpan(templateInfo, position);
|
|
@@ -307,21 +310,54 @@
|
|
|
307
310
|
}
|
|
308
311
|
return results;
|
|
309
312
|
}
|
|
310
|
-
|
|
311
|
-
|
|
312
|
-
|
|
313
|
-
|
|
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
|
+
}
|
|
314
354
|
}
|
|
315
|
-
|
|
316
|
-
|
|
317
|
-
|
|
318
|
-
|
|
319
|
-
|
|
320
|
-
path.push(new compiler_1.AttrAst(attr.name, attr.value, attr.valueSpan));
|
|
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);
|
|
321
360
|
}
|
|
322
|
-
var dinfo = utils_1.diagnosticInfoFromTemplateInfo(info);
|
|
323
|
-
var visitor = new ExpressionVisitor(info, position, function () { return expression_diagnostics_1.getExpressionScope(dinfo, path, false); });
|
|
324
|
-
path.tail.visit(visitor, null);
|
|
325
361
|
return visitor.results;
|
|
326
362
|
}
|
|
327
363
|
function elementCompletions(info) {
|
|
@@ -382,7 +418,7 @@
|
|
|
382
418
|
if (!templatePath.tail) {
|
|
383
419
|
return [];
|
|
384
420
|
}
|
|
385
|
-
var visitor = new ExpressionVisitor(info, position, function () { return expression_diagnostics_1.getExpressionScope(utils_1.diagnosticInfoFromTemplateInfo(info), templatePath
|
|
421
|
+
var visitor = new ExpressionVisitor(info, position, function () { return expression_diagnostics_1.getExpressionScope(utils_1.diagnosticInfoFromTemplateInfo(info), templatePath); });
|
|
386
422
|
templatePath.tail.visit(visitor, null);
|
|
387
423
|
return visitor.results;
|
|
388
424
|
}
|
|
@@ -421,39 +457,43 @@
|
|
|
421
457
|
configurable: true
|
|
422
458
|
});
|
|
423
459
|
ExpressionVisitor.prototype.visitDirectiveProperty = function (ast) {
|
|
424
|
-
this.
|
|
460
|
+
this.processExpressionCompletions(ast.value);
|
|
425
461
|
};
|
|
426
462
|
ExpressionVisitor.prototype.visitElementProperty = function (ast) {
|
|
427
|
-
this.
|
|
463
|
+
this.processExpressionCompletions(ast.value);
|
|
428
464
|
};
|
|
429
|
-
ExpressionVisitor.prototype.visitEvent = function (ast) { this.
|
|
430
|
-
ExpressionVisitor.prototype.visitElement = function (
|
|
465
|
+
ExpressionVisitor.prototype.visitEvent = function (ast) { this.processExpressionCompletions(ast.handler); };
|
|
466
|
+
ExpressionVisitor.prototype.visitElement = function () {
|
|
431
467
|
// no-op for now
|
|
432
468
|
};
|
|
433
469
|
ExpressionVisitor.prototype.visitAttr = function (ast) {
|
|
434
|
-
// The attribute value is a template expression but the expression AST
|
|
435
|
-
// was not produced when the TemplateAst was produced so do that here.
|
|
436
|
-
var templateBindings = this.info.expressionParser.parseTemplateBindings(ast.name, ast.value, ast.sourceSpan.toString(), ast.sourceSpan.start.offset).templateBindings;
|
|
437
|
-
// Find where the cursor is relative to the start of the attribute value.
|
|
438
|
-
var valueRelativePosition = this.position - ast.sourceSpan.start.offset;
|
|
439
|
-
// Find the template binding that contains the position
|
|
440
|
-
var binding = templateBindings.find(function (b) { return utils_1.inSpan(valueRelativePosition, b.span); });
|
|
441
|
-
if (!binding) {
|
|
442
|
-
return;
|
|
443
|
-
}
|
|
444
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
|
+
}
|
|
445
481
|
this.microSyntaxInAttributeValue(ast, binding);
|
|
446
482
|
}
|
|
447
483
|
else {
|
|
448
|
-
|
|
449
|
-
|
|
450
|
-
var span = new compiler_1.ParseSpan(0, ast.value.length);
|
|
451
|
-
var offset = ast.sourceSpan.start.offset;
|
|
452
|
-
var receiver = new compiler_1.ImplicitReceiver(span, span.toAbsolute(offset));
|
|
453
|
-
var expressionAst = new compiler_1.PropertyRead(span, span.toAbsolute(offset), receiver, '');
|
|
454
|
-
this.addAttributeValuesToCompletions(expressionAst);
|
|
484
|
+
var expressionAst = this.info.expressionParser.parseBinding(ast.value, ast.sourceSpan.toString(), ast.sourceSpan.start.offset);
|
|
485
|
+
this.processExpressionCompletions(expressionAst);
|
|
455
486
|
}
|
|
456
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
|
+
};
|
|
457
497
|
ExpressionVisitor.prototype.visitBoundText = function (ast) {
|
|
458
498
|
if (utils_1.inSpan(this.position, ast.value.sourceSpan)) {
|
|
459
499
|
var completions = expressions_1.getExpressionCompletions(this.getExpressionScope(), ast.value, this.position, this.info.template.query);
|
|
@@ -462,7 +502,7 @@
|
|
|
462
502
|
}
|
|
463
503
|
}
|
|
464
504
|
};
|
|
465
|
-
ExpressionVisitor.prototype.
|
|
505
|
+
ExpressionVisitor.prototype.processExpressionCompletions = function (value) {
|
|
466
506
|
var symbols = expressions_1.getExpressionCompletions(this.getExpressionScope(), value, this.position, this.info.template.query);
|
|
467
507
|
if (symbols) {
|
|
468
508
|
this.addSymbolsToCompletions(symbols);
|
|
@@ -476,11 +516,14 @@
|
|
|
476
516
|
if (s.name.startsWith('__') || !s.public || this.completions.has(s.name)) {
|
|
477
517
|
continue;
|
|
478
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;
|
|
479
522
|
this.completions.set(s.name, {
|
|
480
523
|
name: s.name,
|
|
481
524
|
kind: s.kind,
|
|
482
525
|
sortText: s.name,
|
|
483
|
-
insertText:
|
|
526
|
+
insertText: shouldInsertParentheses ? s.name + "()" : s.name,
|
|
484
527
|
});
|
|
485
528
|
}
|
|
486
529
|
}
|
|
@@ -521,7 +564,7 @@
|
|
|
521
564
|
var valueRelativePosition = this.position - attr.sourceSpan.start.offset;
|
|
522
565
|
if (binding.keyIsVar) {
|
|
523
566
|
var equalLocation = attr.value.indexOf('=');
|
|
524
|
-
if (equalLocation
|
|
567
|
+
if (equalLocation > 0 && valueRelativePosition > equalLocation) {
|
|
525
568
|
// We are after the '=' in a let clause. The valid values here are the members of the
|
|
526
569
|
// template reference's type parameter.
|
|
527
570
|
var directiveMetadata = selectorInfo.map.get(selector);
|
|
@@ -536,9 +579,18 @@
|
|
|
536
579
|
}
|
|
537
580
|
}
|
|
538
581
|
if (binding.expression && utils_1.inSpan(valueRelativePosition, binding.expression.ast.span)) {
|
|
539
|
-
this.
|
|
582
|
+
this.processExpressionCompletions(binding.expression.ast);
|
|
540
583
|
return;
|
|
541
584
|
}
|
|
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
|
+
}
|
|
542
594
|
};
|
|
543
595
|
return ExpressionVisitor;
|
|
544
596
|
}(compiler_1.NullTemplateVisitor));
|
|
@@ -631,4 +683,4 @@
|
|
|
631
683
|
return { templateRefs: templateRefs, inputs: inputs, outputs: outputs, bananas: bananas, others: others };
|
|
632
684
|
}
|
|
633
685
|
});
|
|
634
|
-
//# 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, ImplicitReceiver, NAMED_ENTITIES, Node as HtmlAst, NullTemplateVisitor, ParseSpan, PropertyRead, 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) {\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 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\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  // HtmlAst contains the `Attribute` node, however the corresponding `AttrAst`\n  // node is missing from the TemplateAst. In this case, we have to manually\n  // append the `AttrAst` node to the path.\n  if (!(path.tail instanceof AttrAst)) {\n    // The sourceSpan of an AttrAst is the valueSpan of the HTML Attribute.\n    path.push(new AttrAst(attr.name, attr.value, attr.valueSpan !));\n  }\n  const dinfo = diagnosticInfoFromTemplateInfo(info);\n  const visitor =\n      new ExpressionVisitor(info, position, () => getExpressionScope(dinfo, path, false));\n  path.tail.visit(visitor, null);\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,\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    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    // no-op for now\n  }\n\n  visitAttr(ast: AttrAst) {\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        ast.name, ast.value, ast.sourceSpan.toString(), ast.sourceSpan.start.offset);\n\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    if (ast.name.startsWith('*')) {\n      this.microSyntaxInAttributeValue(ast, binding);\n    } else {\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, ast.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);\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) {\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      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  /**\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.addAttributeValuesToCompletions(binding.expression.ast);\n      return;\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"]}
|
|
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"]}
|