UNPKG

89.2 kBJavaScriptView Raw
1/**
2 * @license
3 * Copyright Google LLC All Rights Reserved.
4 *
5 * Use of this source code is governed by an MIT-style license that can be
6 * found in the LICENSE file at https://angular.io/license
7 */
8(function (factory) {
9 if (typeof module === "object" && typeof module.exports === "object") {
10 var v = factory(require, exports);
11 if (v !== undefined) module.exports = v;
12 }
13 else if (typeof define === "function" && define.amd) {
14 define("@angular/language-service/ivy/completions", ["require", "exports", "tslib", "@angular/compiler", "@angular/compiler-cli/src/ngtsc/typecheck/api", "@angular/compiler/src/render3/r3_ast", "typescript", "@angular/language-service/ivy/attribute_completions", "@angular/language-service/ivy/display_parts", "@angular/language-service/ivy/utils"], factory);
15 }
16})(function (require, exports) {
17 "use strict";
18 Object.defineProperty(exports, "__esModule", { value: true });
19 exports.CompletionBuilder = exports.CompletionNodeContext = void 0;
20 var tslib_1 = require("tslib");
21 var compiler_1 = require("@angular/compiler");
22 var api_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/api");
23 var r3_ast_1 = require("@angular/compiler/src/render3/r3_ast");
24 var ts = require("typescript");
25 var attribute_completions_1 = require("@angular/language-service/ivy/attribute_completions");
26 var display_parts_1 = require("@angular/language-service/ivy/display_parts");
27 var utils_1 = require("@angular/language-service/ivy/utils");
28 var CompletionNodeContext;
29 (function (CompletionNodeContext) {
30 CompletionNodeContext[CompletionNodeContext["None"] = 0] = "None";
31 CompletionNodeContext[CompletionNodeContext["ElementTag"] = 1] = "ElementTag";
32 CompletionNodeContext[CompletionNodeContext["ElementAttributeKey"] = 2] = "ElementAttributeKey";
33 CompletionNodeContext[CompletionNodeContext["ElementAttributeValue"] = 3] = "ElementAttributeValue";
34 CompletionNodeContext[CompletionNodeContext["EventValue"] = 4] = "EventValue";
35 CompletionNodeContext[CompletionNodeContext["TwoWayBinding"] = 5] = "TwoWayBinding";
36 })(CompletionNodeContext = exports.CompletionNodeContext || (exports.CompletionNodeContext = {}));
37 /**
38 * Performs autocompletion operations on a given node in the template.
39 *
40 * This class acts as a closure around all of the context required to perform the 3 autocompletion
41 * operations (completions, get details, and get symbol) at a specific node.
42 *
43 * The generic `N` type for the template node is narrowed internally for certain operations, as the
44 * compiler operations required to implement completion may be different for different node types.
45 *
46 * @param N type of the template node in question, narrowed accordingly.
47 */
48 var CompletionBuilder = /** @class */ (function () {
49 function CompletionBuilder(tsLS, compiler, component, node, nodeContext, nodeParent, template) {
50 this.tsLS = tsLS;
51 this.compiler = compiler;
52 this.component = component;
53 this.node = node;
54 this.nodeContext = nodeContext;
55 this.nodeParent = nodeParent;
56 this.template = template;
57 this.typeChecker = this.compiler.getNextProgram().getTypeChecker();
58 this.templateTypeChecker = this.compiler.getTemplateTypeChecker();
59 }
60 /**
61 * Analogue for `ts.LanguageService.getCompletionsAtPosition`.
62 */
63 CompletionBuilder.prototype.getCompletionsAtPosition = function (options) {
64 if (this.isPropertyExpressionCompletion()) {
65 return this.getPropertyExpressionCompletion(options);
66 }
67 else if (this.isElementTagCompletion()) {
68 return this.getElementTagCompletion();
69 }
70 else if (this.isElementAttributeCompletion()) {
71 return this.getElementAttributeCompletions();
72 }
73 else if (this.isPipeCompletion()) {
74 return this.getPipeCompletions();
75 }
76 else {
77 return undefined;
78 }
79 };
80 /**
81 * Analogue for `ts.LanguageService.getCompletionEntryDetails`.
82 */
83 CompletionBuilder.prototype.getCompletionEntryDetails = function (entryName, formatOptions, preferences) {
84 if (this.isPropertyExpressionCompletion()) {
85 return this.getPropertyExpressionCompletionDetails(entryName, formatOptions, preferences);
86 }
87 else if (this.isElementTagCompletion()) {
88 return this.getElementTagCompletionDetails(entryName);
89 }
90 else if (this.isElementAttributeCompletion()) {
91 return this.getElementAttributeCompletionDetails(entryName);
92 }
93 };
94 /**
95 * Analogue for `ts.LanguageService.getCompletionEntrySymbol`.
96 */
97 CompletionBuilder.prototype.getCompletionEntrySymbol = function (name) {
98 if (this.isPropertyExpressionCompletion()) {
99 return this.getPropertyExpressionCompletionSymbol(name);
100 }
101 else if (this.isElementTagCompletion()) {
102 return this.getElementTagCompletionSymbol(name);
103 }
104 else if (this.isElementAttributeCompletion()) {
105 return this.getElementAttributeCompletionSymbol(name);
106 }
107 else {
108 return undefined;
109 }
110 };
111 /**
112 * Determine if the current node is the completion of a property expression, and narrow the type
113 * of `this.node` if so.
114 *
115 * This narrowing gives access to additional methods related to completion of property
116 * expressions.
117 */
118 CompletionBuilder.prototype.isPropertyExpressionCompletion = function () {
119 return this.node instanceof compiler_1.PropertyRead || this.node instanceof compiler_1.MethodCall ||
120 this.node instanceof compiler_1.SafePropertyRead || this.node instanceof compiler_1.SafeMethodCall ||
121 this.node instanceof compiler_1.PropertyWrite || this.node instanceof compiler_1.EmptyExpr ||
122 // BoundEvent nodes only count as property completions if in an EventValue context.
123 (this.node instanceof r3_ast_1.BoundEvent && this.nodeContext === CompletionNodeContext.EventValue);
124 };
125 /**
126 * Get completions for property expressions.
127 */
128 CompletionBuilder.prototype.getPropertyExpressionCompletion = function (options) {
129 var e_1, _a;
130 if (this.node instanceof compiler_1.EmptyExpr || this.node instanceof r3_ast_1.BoundEvent ||
131 this.node.receiver instanceof compiler_1.ImplicitReceiver) {
132 return this.getGlobalPropertyExpressionCompletion(options);
133 }
134 else {
135 var location_1 = this.compiler.getTemplateTypeChecker().getExpressionCompletionLocation(this.node, this.component);
136 if (location_1 === null) {
137 return undefined;
138 }
139 var tsResults = this.tsLS.getCompletionsAtPosition(location_1.shimPath, location_1.positionInShimFile, options);
140 if (tsResults === undefined) {
141 return undefined;
142 }
143 var replacementSpan = makeReplacementSpanFromAst(this.node);
144 var ngResults = [];
145 try {
146 for (var _b = tslib_1.__values(tsResults.entries), _c = _b.next(); !_c.done; _c = _b.next()) {
147 var result = _c.value;
148 ngResults.push(tslib_1.__assign(tslib_1.__assign({}, result), { replacementSpan: replacementSpan }));
149 }
150 }
151 catch (e_1_1) { e_1 = { error: e_1_1 }; }
152 finally {
153 try {
154 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
155 }
156 finally { if (e_1) throw e_1.error; }
157 }
158 return tslib_1.__assign(tslib_1.__assign({}, tsResults), { entries: ngResults });
159 }
160 };
161 /**
162 * Get the details of a specific completion for a property expression.
163 */
164 CompletionBuilder.prototype.getPropertyExpressionCompletionDetails = function (entryName, formatOptions, preferences) {
165 var details = undefined;
166 if (this.node instanceof compiler_1.EmptyExpr || this.node instanceof r3_ast_1.BoundEvent ||
167 this.node.receiver instanceof compiler_1.ImplicitReceiver) {
168 details =
169 this.getGlobalPropertyExpressionCompletionDetails(entryName, formatOptions, preferences);
170 }
171 else {
172 var location_2 = this.compiler.getTemplateTypeChecker().getExpressionCompletionLocation(this.node, this.component);
173 if (location_2 === null) {
174 return undefined;
175 }
176 details = this.tsLS.getCompletionEntryDetails(location_2.shimPath, location_2.positionInShimFile, entryName, formatOptions,
177 /* source */ undefined, preferences);
178 }
179 if (details !== undefined) {
180 details.displayParts = utils_1.filterAliasImports(details.displayParts);
181 }
182 return details;
183 };
184 /**
185 * Get the `ts.Symbol` for a specific completion for a property expression.
186 */
187 CompletionBuilder.prototype.getPropertyExpressionCompletionSymbol = function (name) {
188 if (this.node instanceof compiler_1.EmptyExpr || this.node instanceof compiler_1.LiteralPrimitive ||
189 this.node instanceof r3_ast_1.BoundEvent || this.node.receiver instanceof compiler_1.ImplicitReceiver) {
190 return this.getGlobalPropertyExpressionCompletionSymbol(name);
191 }
192 else {
193 var location_3 = this.compiler.getTemplateTypeChecker().getExpressionCompletionLocation(this.node, this.component);
194 if (location_3 === null) {
195 return undefined;
196 }
197 return this.tsLS.getCompletionEntrySymbol(location_3.shimPath, location_3.positionInShimFile, name, /* source */ undefined);
198 }
199 };
200 /**
201 * Get completions for a property expression in a global context (e.g. `{{y|}}`).
202 */
203 CompletionBuilder.prototype.getGlobalPropertyExpressionCompletion = function (options) {
204 var e_2, _a, e_3, _b;
205 var completions = this.templateTypeChecker.getGlobalCompletions(this.template, this.component);
206 if (completions === null) {
207 return undefined;
208 }
209 var componentContext = completions.componentContext, templateContext = completions.templateContext;
210 var replacementSpan = makeReplacementSpanFromAst(this.node);
211 // Merge TS completion results with results from the template scope.
212 var entries = [];
213 var tsLsCompletions = this.tsLS.getCompletionsAtPosition(componentContext.shimPath, componentContext.positionInShimFile, options);
214 if (tsLsCompletions !== undefined) {
215 try {
216 for (var _c = tslib_1.__values(tsLsCompletions.entries), _d = _c.next(); !_d.done; _d = _c.next()) {
217 var tsCompletion = _d.value;
218 // Skip completions that are shadowed by a template entity definition.
219 if (templateContext.has(tsCompletion.name)) {
220 continue;
221 }
222 entries.push(tslib_1.__assign(tslib_1.__assign({}, tsCompletion), {
223 // Substitute the TS completion's `replacementSpan` (which uses offsets within the TCB)
224 // with the `replacementSpan` within the template source.
225 replacementSpan: replacementSpan }));
226 }
227 }
228 catch (e_2_1) { e_2 = { error: e_2_1 }; }
229 finally {
230 try {
231 if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
232 }
233 finally { if (e_2) throw e_2.error; }
234 }
235 }
236 try {
237 for (var templateContext_1 = tslib_1.__values(templateContext), templateContext_1_1 = templateContext_1.next(); !templateContext_1_1.done; templateContext_1_1 = templateContext_1.next()) {
238 var _e = tslib_1.__read(templateContext_1_1.value, 2), name_1 = _e[0], entity = _e[1];
239 entries.push({
240 name: name_1,
241 sortText: name_1,
242 replacementSpan: replacementSpan,
243 kindModifiers: ts.ScriptElementKindModifier.none,
244 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(entity.kind === api_1.CompletionKind.Reference ? display_parts_1.DisplayInfoKind.REFERENCE :
245 display_parts_1.DisplayInfoKind.VARIABLE),
246 });
247 }
248 }
249 catch (e_3_1) { e_3 = { error: e_3_1 }; }
250 finally {
251 try {
252 if (templateContext_1_1 && !templateContext_1_1.done && (_b = templateContext_1.return)) _b.call(templateContext_1);
253 }
254 finally { if (e_3) throw e_3.error; }
255 }
256 return {
257 entries: entries,
258 // Although this completion is "global" in the sense of an Angular expression (there is no
259 // explicit receiver), it is not "global" in a TypeScript sense since Angular expressions have
260 // the component as an implicit receiver.
261 isGlobalCompletion: false,
262 isMemberCompletion: true,
263 isNewIdentifierLocation: false,
264 };
265 };
266 /**
267 * Get the details of a specific completion for a property expression in a global context (e.g.
268 * `{{y|}}`).
269 */
270 CompletionBuilder.prototype.getGlobalPropertyExpressionCompletionDetails = function (entryName, formatOptions, preferences) {
271 var completions = this.templateTypeChecker.getGlobalCompletions(this.template, this.component);
272 if (completions === null) {
273 return undefined;
274 }
275 var componentContext = completions.componentContext, templateContext = completions.templateContext;
276 if (templateContext.has(entryName)) {
277 var entry = templateContext.get(entryName);
278 // Entries that reference a symbol in the template context refer either to local references or
279 // variables.
280 var symbol = this.templateTypeChecker.getSymbolOfNode(entry.node, this.component);
281 if (symbol === null) {
282 return undefined;
283 }
284 var _a = display_parts_1.getSymbolDisplayInfo(this.tsLS, this.typeChecker, symbol), kind = _a.kind, displayParts = _a.displayParts, documentation = _a.documentation;
285 return {
286 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(kind),
287 name: entryName,
288 kindModifiers: ts.ScriptElementKindModifier.none,
289 displayParts: displayParts,
290 documentation: documentation,
291 };
292 }
293 else {
294 return this.tsLS.getCompletionEntryDetails(componentContext.shimPath, componentContext.positionInShimFile, entryName, formatOptions,
295 /* source */ undefined, preferences);
296 }
297 };
298 /**
299 * Get the `ts.Symbol` of a specific completion for a property expression in a global context
300 * (e.g.
301 * `{{y|}}`).
302 */
303 CompletionBuilder.prototype.getGlobalPropertyExpressionCompletionSymbol = function (entryName) {
304 var completions = this.templateTypeChecker.getGlobalCompletions(this.template, this.component);
305 if (completions === null) {
306 return undefined;
307 }
308 var componentContext = completions.componentContext, templateContext = completions.templateContext;
309 if (templateContext.has(entryName)) {
310 var node = templateContext.get(entryName).node;
311 var symbol = this.templateTypeChecker.getSymbolOfNode(node, this.component);
312 if (symbol === null || symbol.tsSymbol === null) {
313 return undefined;
314 }
315 return symbol.tsSymbol;
316 }
317 else {
318 return this.tsLS.getCompletionEntrySymbol(componentContext.shimPath, componentContext.positionInShimFile, entryName,
319 /* source */ undefined);
320 }
321 };
322 CompletionBuilder.prototype.isElementTagCompletion = function () {
323 return this.node instanceof compiler_1.TmplAstElement &&
324 this.nodeContext === CompletionNodeContext.ElementTag;
325 };
326 CompletionBuilder.prototype.getElementTagCompletion = function () {
327 var templateTypeChecker = this.compiler.getTemplateTypeChecker();
328 // The replacementSpan is the tag name.
329 var replacementSpan = {
330 start: this.node.sourceSpan.start.offset + 1,
331 length: this.node.name.length,
332 };
333 var entries = Array.from(templateTypeChecker.getPotentialElementTags(this.component))
334 .map(function (_a) {
335 var _b = tslib_1.__read(_a, 2), tag = _b[0], directive = _b[1];
336 return ({
337 kind: tagCompletionKind(directive),
338 name: tag,
339 sortText: tag,
340 replacementSpan: replacementSpan,
341 });
342 });
343 return {
344 entries: entries,
345 isGlobalCompletion: false,
346 isMemberCompletion: false,
347 isNewIdentifierLocation: false,
348 };
349 };
350 CompletionBuilder.prototype.getElementTagCompletionDetails = function (entryName) {
351 var templateTypeChecker = this.compiler.getTemplateTypeChecker();
352 var tagMap = templateTypeChecker.getPotentialElementTags(this.component);
353 if (!tagMap.has(entryName)) {
354 return undefined;
355 }
356 var directive = tagMap.get(entryName);
357 var displayParts;
358 var documentation = undefined;
359 if (directive === null) {
360 displayParts = [];
361 }
362 else {
363 var displayInfo = display_parts_1.getDirectiveDisplayInfo(this.tsLS, directive);
364 displayParts = displayInfo.displayParts;
365 documentation = displayInfo.documentation;
366 }
367 return {
368 kind: tagCompletionKind(directive),
369 name: entryName,
370 kindModifiers: ts.ScriptElementKindModifier.none,
371 displayParts: displayParts,
372 documentation: documentation,
373 };
374 };
375 CompletionBuilder.prototype.getElementTagCompletionSymbol = function (entryName) {
376 var templateTypeChecker = this.compiler.getTemplateTypeChecker();
377 var tagMap = templateTypeChecker.getPotentialElementTags(this.component);
378 if (!tagMap.has(entryName)) {
379 return undefined;
380 }
381 var directive = tagMap.get(entryName);
382 return directive === null || directive === void 0 ? void 0 : directive.tsSymbol;
383 };
384 CompletionBuilder.prototype.isElementAttributeCompletion = function () {
385 return (this.nodeContext === CompletionNodeContext.ElementAttributeKey ||
386 this.nodeContext === CompletionNodeContext.TwoWayBinding) &&
387 (this.node instanceof compiler_1.TmplAstElement || this.node instanceof compiler_1.TmplAstBoundAttribute ||
388 this.node instanceof compiler_1.TmplAstTextAttribute || this.node instanceof compiler_1.TmplAstBoundEvent);
389 };
390 CompletionBuilder.prototype.getElementAttributeCompletions = function () {
391 var e_4, _a;
392 var element;
393 if (this.node instanceof compiler_1.TmplAstElement) {
394 element = this.node;
395 }
396 else if (this.nodeParent instanceof compiler_1.TmplAstElement || this.nodeParent instanceof compiler_1.TmplAstTemplate) {
397 element = this.nodeParent;
398 }
399 else {
400 // Nothing to do without an element to process.
401 return undefined;
402 }
403 var replacementSpan = undefined;
404 if ((this.node instanceof compiler_1.TmplAstBoundAttribute || this.node instanceof compiler_1.TmplAstBoundEvent ||
405 this.node instanceof compiler_1.TmplAstTextAttribute) &&
406 this.node.keySpan !== undefined) {
407 replacementSpan = makeReplacementSpanFromParseSourceSpan(this.node.keySpan);
408 }
409 var attrTable = attribute_completions_1.buildAttributeCompletionTable(this.component, element, this.compiler.getTemplateTypeChecker());
410 var entries = [];
411 try {
412 for (var _b = tslib_1.__values(attrTable.values()), _c = _b.next(); !_c.done; _c = _b.next()) {
413 var completion = _c.value;
414 // First, filter out completions that don't make sense for the current node. For example, if
415 // the user is completing on a property binding `[foo|]`, don't offer output event
416 // completions.
417 switch (completion.kind) {
418 case attribute_completions_1.AttributeCompletionKind.DomAttribute:
419 case attribute_completions_1.AttributeCompletionKind.DomProperty:
420 if (this.node instanceof compiler_1.TmplAstBoundEvent) {
421 continue;
422 }
423 break;
424 case attribute_completions_1.AttributeCompletionKind.DirectiveInput:
425 if (this.node instanceof compiler_1.TmplAstBoundEvent) {
426 continue;
427 }
428 if (!completion.twoWayBindingSupported &&
429 this.nodeContext === CompletionNodeContext.TwoWayBinding) {
430 continue;
431 }
432 break;
433 case attribute_completions_1.AttributeCompletionKind.DirectiveOutput:
434 if (this.node instanceof compiler_1.TmplAstBoundAttribute) {
435 continue;
436 }
437 break;
438 case attribute_completions_1.AttributeCompletionKind.DirectiveAttribute:
439 if (this.node instanceof compiler_1.TmplAstBoundAttribute ||
440 this.node instanceof compiler_1.TmplAstBoundEvent) {
441 continue;
442 }
443 break;
444 }
445 // Is the completion in an attribute context (instead of a property context)?
446 var isAttributeContext = (this.node instanceof compiler_1.TmplAstElement || this.node instanceof compiler_1.TmplAstTextAttribute);
447 // Is the completion for an element (not an <ng-template>)?
448 var isElementContext = this.node instanceof compiler_1.TmplAstElement || this.nodeParent instanceof compiler_1.TmplAstElement;
449 attribute_completions_1.addAttributeCompletionEntries(entries, completion, isAttributeContext, isElementContext, replacementSpan);
450 }
451 }
452 catch (e_4_1) { e_4 = { error: e_4_1 }; }
453 finally {
454 try {
455 if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
456 }
457 finally { if (e_4) throw e_4.error; }
458 }
459 return {
460 entries: entries,
461 isGlobalCompletion: false,
462 isMemberCompletion: false,
463 isNewIdentifierLocation: true,
464 };
465 };
466 CompletionBuilder.prototype.getElementAttributeCompletionDetails = function (entryName) {
467 // `entryName` here may be `foo` or `[foo]`, depending on which suggested completion the user
468 // chose. Strip off any binding syntax to get the real attribute name.
469 var _a = stripBindingSugar(entryName), name = _a.name, kind = _a.kind;
470 var element;
471 if (this.node instanceof compiler_1.TmplAstElement || this.node instanceof compiler_1.TmplAstTemplate) {
472 element = this.node;
473 }
474 else if (this.nodeParent instanceof compiler_1.TmplAstElement || this.nodeParent instanceof compiler_1.TmplAstTemplate) {
475 element = this.nodeParent;
476 }
477 else {
478 // Nothing to do without an element to process.
479 return undefined;
480 }
481 var attrTable = attribute_completions_1.buildAttributeCompletionTable(this.component, element, this.compiler.getTemplateTypeChecker());
482 if (!attrTable.has(name)) {
483 return undefined;
484 }
485 var completion = attrTable.get(name);
486 var displayParts;
487 var documentation = undefined;
488 var info;
489 switch (completion.kind) {
490 case attribute_completions_1.AttributeCompletionKind.DomAttribute:
491 case attribute_completions_1.AttributeCompletionKind.DomProperty:
492 // TODO(alxhub): ideally we would show the same documentation as quick info here. However,
493 // since these bindings don't exist in the TCB, there is no straightforward way to retrieve
494 // a `ts.Symbol` for the field in the TS DOM definition.
495 displayParts = [];
496 break;
497 case attribute_completions_1.AttributeCompletionKind.DirectiveAttribute:
498 info = display_parts_1.getDirectiveDisplayInfo(this.tsLS, completion.directive);
499 displayParts = info.displayParts;
500 documentation = info.documentation;
501 break;
502 case attribute_completions_1.AttributeCompletionKind.DirectiveInput:
503 case attribute_completions_1.AttributeCompletionKind.DirectiveOutput:
504 var propertySymbol = attribute_completions_1.getAttributeCompletionSymbol(completion, this.typeChecker);
505 if (propertySymbol === null) {
506 return undefined;
507 }
508 info = display_parts_1.getTsSymbolDisplayInfo(this.tsLS, this.typeChecker, propertySymbol, completion.kind === attribute_completions_1.AttributeCompletionKind.DirectiveInput ? display_parts_1.DisplayInfoKind.PROPERTY :
509 display_parts_1.DisplayInfoKind.EVENT, completion.directive.tsSymbol.name);
510 if (info === null) {
511 return undefined;
512 }
513 displayParts = info.displayParts;
514 documentation = info.documentation;
515 }
516 return {
517 name: entryName,
518 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(kind),
519 kindModifiers: ts.ScriptElementKindModifier.none,
520 displayParts: [],
521 documentation: documentation,
522 };
523 };
524 CompletionBuilder.prototype.getElementAttributeCompletionSymbol = function (attribute) {
525 var _a;
526 var name = stripBindingSugar(attribute).name;
527 var element;
528 if (this.node instanceof compiler_1.TmplAstElement || this.node instanceof compiler_1.TmplAstTemplate) {
529 element = this.node;
530 }
531 else if (this.nodeParent instanceof compiler_1.TmplAstElement || this.nodeParent instanceof compiler_1.TmplAstTemplate) {
532 element = this.nodeParent;
533 }
534 else {
535 // Nothing to do without an element to process.
536 return undefined;
537 }
538 var attrTable = attribute_completions_1.buildAttributeCompletionTable(this.component, element, this.compiler.getTemplateTypeChecker());
539 if (!attrTable.has(name)) {
540 return undefined;
541 }
542 var completion = attrTable.get(name);
543 return (_a = attribute_completions_1.getAttributeCompletionSymbol(completion, this.typeChecker)) !== null && _a !== void 0 ? _a : undefined;
544 };
545 CompletionBuilder.prototype.isPipeCompletion = function () {
546 return this.node instanceof compiler_1.BindingPipe;
547 };
548 CompletionBuilder.prototype.getPipeCompletions = function () {
549 var pipes = this.templateTypeChecker.getPipesInScope(this.component);
550 if (pipes === null) {
551 return undefined;
552 }
553 var replacementSpan = makeReplacementSpanFromAst(this.node);
554 var entries = pipes.map(function (pipe) { return ({
555 name: pipe.name,
556 sortText: pipe.name,
557 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.PIPE),
558 replacementSpan: replacementSpan,
559 }); });
560 return {
561 entries: entries,
562 isGlobalCompletion: false,
563 isMemberCompletion: false,
564 isNewIdentifierLocation: false,
565 };
566 };
567 return CompletionBuilder;
568 }());
569 exports.CompletionBuilder = CompletionBuilder;
570 function makeReplacementSpanFromParseSourceSpan(span) {
571 return {
572 start: span.start.offset,
573 length: span.end.offset - span.start.offset,
574 };
575 }
576 function makeReplacementSpanFromAst(node) {
577 if ((node instanceof compiler_1.EmptyExpr || node instanceof compiler_1.LiteralPrimitive ||
578 node instanceof r3_ast_1.BoundEvent)) {
579 // empty nodes do not replace any existing text
580 return undefined;
581 }
582 return {
583 start: node.nameSpan.start,
584 length: node.nameSpan.end - node.nameSpan.start,
585 };
586 }
587 function tagCompletionKind(directive) {
588 var kind;
589 if (directive === null) {
590 kind = display_parts_1.DisplayInfoKind.ELEMENT;
591 }
592 else if (directive.isComponent) {
593 kind = display_parts_1.DisplayInfoKind.COMPONENT;
594 }
595 else {
596 kind = display_parts_1.DisplayInfoKind.DIRECTIVE;
597 }
598 return display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(kind);
599 }
600 var BINDING_SUGAR = /[\[\(\)\]]/g;
601 function stripBindingSugar(binding) {
602 var name = binding.replace(BINDING_SUGAR, '');
603 if (binding.startsWith('[')) {
604 return { name: name, kind: display_parts_1.DisplayInfoKind.PROPERTY };
605 }
606 else if (binding.startsWith('(')) {
607 return { name: name, kind: display_parts_1.DisplayInfoKind.EVENT };
608 }
609 else {
610 return { name: name, kind: display_parts_1.DisplayInfoKind.ATTRIBUTE };
611 }
612 }
613});
614//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"completions.js","sourceRoot":"","sources":["../../../../../../packages/language-service/ivy/completions.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {AST, BindingPipe, EmptyExpr, ImplicitReceiver, LiteralPrimitive, MethodCall, ParseSourceSpan, PropertyRead, PropertyWrite, SafeMethodCall, SafePropertyRead, TmplAstBoundAttribute, TmplAstBoundEvent, TmplAstElement, TmplAstNode, TmplAstReference, TmplAstTemplate, TmplAstTextAttribute, TmplAstVariable} from '@angular/compiler';\nimport {NgCompiler} from '@angular/compiler-cli/src/ngtsc/core';\nimport {CompletionKind, DirectiveInScope, TemplateDeclarationSymbol} from '@angular/compiler-cli/src/ngtsc/typecheck/api';\nimport {BoundEvent} from '@angular/compiler/src/render3/r3_ast';\nimport * as ts from 'typescript';\n\nimport {addAttributeCompletionEntries, AttributeCompletionKind, buildAttributeCompletionTable, getAttributeCompletionSymbol} from './attribute_completions';\nimport {DisplayInfo, DisplayInfoKind, getDirectiveDisplayInfo, getSymbolDisplayInfo, getTsSymbolDisplayInfo, unsafeCastDisplayInfoKindToScriptElementKind} from './display_parts';\nimport {filterAliasImports} from './utils';\n\ntype PropertyExpressionCompletionBuilder =\n    CompletionBuilder<PropertyRead|PropertyWrite|MethodCall|EmptyExpr|SafePropertyRead|\n                      SafeMethodCall|TmplAstBoundEvent>;\n\ntype ElementAttributeCompletionBuilder =\n    CompletionBuilder<TmplAstElement|TmplAstBoundAttribute|TmplAstTextAttribute|TmplAstBoundEvent>;\n\ntype PipeCompletionBuilder = CompletionBuilder<BindingPipe>;\n\nexport enum CompletionNodeContext {\n  None,\n  ElementTag,\n  ElementAttributeKey,\n  ElementAttributeValue,\n  EventValue,\n  TwoWayBinding,\n}\n\n/**\n * Performs autocompletion operations on a given node in the template.\n *\n * This class acts as a closure around all of the context required to perform the 3 autocompletion\n * operations (completions, get details, and get symbol) at a specific node.\n *\n * The generic `N` type for the template node is narrowed internally for certain operations, as the\n * compiler operations required to implement completion may be different for different node types.\n *\n * @param N type of the template node in question, narrowed accordingly.\n */\nexport class CompletionBuilder<N extends TmplAstNode|AST> {\n  private readonly typeChecker = this.compiler.getNextProgram().getTypeChecker();\n  private readonly templateTypeChecker = this.compiler.getTemplateTypeChecker();\n\n  constructor(\n      private readonly tsLS: ts.LanguageService, private readonly compiler: NgCompiler,\n      private readonly component: ts.ClassDeclaration, private readonly node: N,\n      private readonly nodeContext: CompletionNodeContext,\n      private readonly nodeParent: TmplAstNode|AST|null,\n      private readonly template: TmplAstTemplate|null) {}\n\n  /**\n   * Analogue for `ts.LanguageService.getCompletionsAtPosition`.\n   */\n  getCompletionsAtPosition(options: ts.GetCompletionsAtPositionOptions|\n                           undefined): ts.WithMetadata<ts.CompletionInfo>|undefined {\n    if (this.isPropertyExpressionCompletion()) {\n      return this.getPropertyExpressionCompletion(options);\n    } else if (this.isElementTagCompletion()) {\n      return this.getElementTagCompletion();\n    } else if (this.isElementAttributeCompletion()) {\n      return this.getElementAttributeCompletions();\n    } else if (this.isPipeCompletion()) {\n      return this.getPipeCompletions();\n    } else {\n      return undefined;\n    }\n  }\n\n  /**\n   * Analogue for `ts.LanguageService.getCompletionEntryDetails`.\n   */\n  getCompletionEntryDetails(\n      entryName: string, formatOptions: ts.FormatCodeOptions|ts.FormatCodeSettings|undefined,\n      preferences: ts.UserPreferences|undefined): ts.CompletionEntryDetails|undefined {\n    if (this.isPropertyExpressionCompletion()) {\n      return this.getPropertyExpressionCompletionDetails(entryName, formatOptions, preferences);\n    } else if (this.isElementTagCompletion()) {\n      return this.getElementTagCompletionDetails(entryName);\n    } else if (this.isElementAttributeCompletion()) {\n      return this.getElementAttributeCompletionDetails(entryName);\n    }\n  }\n\n  /**\n   * Analogue for `ts.LanguageService.getCompletionEntrySymbol`.\n   */\n  getCompletionEntrySymbol(name: string): ts.Symbol|undefined {\n    if (this.isPropertyExpressionCompletion()) {\n      return this.getPropertyExpressionCompletionSymbol(name);\n    } else if (this.isElementTagCompletion()) {\n      return this.getElementTagCompletionSymbol(name);\n    } else if (this.isElementAttributeCompletion()) {\n      return this.getElementAttributeCompletionSymbol(name);\n    } else {\n      return undefined;\n    }\n  }\n\n  /**\n   * Determine if the current node is the completion of a property expression, and narrow the type\n   * of `this.node` if so.\n   *\n   * This narrowing gives access to additional methods related to completion of property\n   * expressions.\n   */\n  private isPropertyExpressionCompletion(this: CompletionBuilder<TmplAstNode|AST>):\n      this is PropertyExpressionCompletionBuilder {\n    return this.node instanceof PropertyRead || this.node instanceof MethodCall ||\n        this.node instanceof SafePropertyRead || this.node instanceof SafeMethodCall ||\n        this.node instanceof PropertyWrite || this.node instanceof EmptyExpr ||\n        // BoundEvent nodes only count as property completions if in an EventValue context.\n        (this.node instanceof BoundEvent && this.nodeContext === CompletionNodeContext.EventValue);\n  }\n\n  /**\n   * Get completions for property expressions.\n   */\n  private getPropertyExpressionCompletion(\n      this: PropertyExpressionCompletionBuilder,\n      options: ts.GetCompletionsAtPositionOptions|\n      undefined): ts.WithMetadata<ts.CompletionInfo>|undefined {\n    if (this.node instanceof EmptyExpr || this.node instanceof BoundEvent ||\n        this.node.receiver instanceof ImplicitReceiver) {\n      return this.getGlobalPropertyExpressionCompletion(options);\n    } else {\n      const location = this.compiler.getTemplateTypeChecker().getExpressionCompletionLocation(\n          this.node, this.component);\n      if (location === null) {\n        return undefined;\n      }\n      const tsResults = this.tsLS.getCompletionsAtPosition(\n          location.shimPath, location.positionInShimFile, options);\n      if (tsResults === undefined) {\n        return undefined;\n      }\n\n      const replacementSpan = makeReplacementSpanFromAst(this.node);\n\n      let ngResults: ts.CompletionEntry[] = [];\n      for (const result of tsResults.entries) {\n        ngResults.push({\n          ...result,\n          replacementSpan,\n        });\n      }\n      return {\n        ...tsResults,\n        entries: ngResults,\n      };\n    }\n  }\n\n  /**\n   * Get the details of a specific completion for a property expression.\n   */\n  private getPropertyExpressionCompletionDetails(\n      this: PropertyExpressionCompletionBuilder, entryName: string,\n      formatOptions: ts.FormatCodeOptions|ts.FormatCodeSettings|undefined,\n      preferences: ts.UserPreferences|undefined): ts.CompletionEntryDetails|undefined {\n    let details: ts.CompletionEntryDetails|undefined = undefined;\n    if (this.node instanceof EmptyExpr || this.node instanceof BoundEvent ||\n        this.node.receiver instanceof ImplicitReceiver) {\n      details =\n          this.getGlobalPropertyExpressionCompletionDetails(entryName, formatOptions, preferences);\n    } else {\n      const location = this.compiler.getTemplateTypeChecker().getExpressionCompletionLocation(\n          this.node, this.component);\n      if (location === null) {\n        return undefined;\n      }\n      details = this.tsLS.getCompletionEntryDetails(\n          location.shimPath, location.positionInShimFile, entryName, formatOptions,\n          /* source */ undefined, preferences);\n    }\n    if (details !== undefined) {\n      details.displayParts = filterAliasImports(details.displayParts);\n    }\n    return details;\n  }\n\n  /**\n   * Get the `ts.Symbol` for a specific completion for a property expression.\n   */\n  private getPropertyExpressionCompletionSymbol(\n      this: PropertyExpressionCompletionBuilder, name: string): ts.Symbol|undefined {\n    if (this.node instanceof EmptyExpr || this.node instanceof LiteralPrimitive ||\n        this.node instanceof BoundEvent || this.node.receiver instanceof ImplicitReceiver) {\n      return this.getGlobalPropertyExpressionCompletionSymbol(name);\n    } else {\n      const location = this.compiler.getTemplateTypeChecker().getExpressionCompletionLocation(\n          this.node, this.component);\n      if (location === null) {\n        return undefined;\n      }\n      return this.tsLS.getCompletionEntrySymbol(\n          location.shimPath, location.positionInShimFile, name, /* source */ undefined);\n    }\n  }\n\n  /**\n   * Get completions for a property expression in a global context (e.g. `{{y|}}`).\n   */\n  private getGlobalPropertyExpressionCompletion(\n      this: PropertyExpressionCompletionBuilder,\n      options: ts.GetCompletionsAtPositionOptions|\n      undefined): ts.WithMetadata<ts.CompletionInfo>|undefined {\n    const completions =\n        this.templateTypeChecker.getGlobalCompletions(this.template, this.component);\n    if (completions === null) {\n      return undefined;\n    }\n\n    const {componentContext, templateContext} = completions;\n\n    const replacementSpan = makeReplacementSpanFromAst(this.node);\n\n    // Merge TS completion results with results from the template scope.\n    let entries: ts.CompletionEntry[] = [];\n    const tsLsCompletions = this.tsLS.getCompletionsAtPosition(\n        componentContext.shimPath, componentContext.positionInShimFile, options);\n    if (tsLsCompletions !== undefined) {\n      for (const tsCompletion of tsLsCompletions.entries) {\n        // Skip completions that are shadowed by a template entity definition.\n        if (templateContext.has(tsCompletion.name)) {\n          continue;\n        }\n        entries.push({\n          ...tsCompletion,\n          // Substitute the TS completion's `replacementSpan` (which uses offsets within the TCB)\n          // with the `replacementSpan` within the template source.\n          replacementSpan,\n        });\n      }\n    }\n\n    for (const [name, entity] of templateContext) {\n      entries.push({\n        name,\n        sortText: name,\n        replacementSpan,\n        kindModifiers: ts.ScriptElementKindModifier.none,\n        kind: unsafeCastDisplayInfoKindToScriptElementKind(\n            entity.kind === CompletionKind.Reference ? DisplayInfoKind.REFERENCE :\n                                                       DisplayInfoKind.VARIABLE),\n      });\n    }\n\n    return {\n      entries,\n      // Although this completion is \"global\" in the sense of an Angular expression (there is no\n      // explicit receiver), it is not \"global\" in a TypeScript sense since Angular expressions have\n      // the component as an implicit receiver.\n      isGlobalCompletion: false,\n      isMemberCompletion: true,\n      isNewIdentifierLocation: false,\n    };\n  }\n\n  /**\n   * Get the details of a specific completion for a property expression in a global context (e.g.\n   * `{{y|}}`).\n   */\n  private getGlobalPropertyExpressionCompletionDetails(\n      this: PropertyExpressionCompletionBuilder, entryName: string,\n      formatOptions: ts.FormatCodeOptions|ts.FormatCodeSettings|undefined,\n      preferences: ts.UserPreferences|undefined): ts.CompletionEntryDetails|undefined {\n    const completions =\n        this.templateTypeChecker.getGlobalCompletions(this.template, this.component);\n    if (completions === null) {\n      return undefined;\n    }\n    const {componentContext, templateContext} = completions;\n\n    if (templateContext.has(entryName)) {\n      const entry = templateContext.get(entryName)!;\n      // Entries that reference a symbol in the template context refer either to local references or\n      // variables.\n      const symbol = this.templateTypeChecker.getSymbolOfNode(entry.node, this.component) as\n              TemplateDeclarationSymbol |\n          null;\n      if (symbol === null) {\n        return undefined;\n      }\n\n      const {kind, displayParts, documentation} =\n          getSymbolDisplayInfo(this.tsLS, this.typeChecker, symbol);\n      return {\n        kind: unsafeCastDisplayInfoKindToScriptElementKind(kind),\n        name: entryName,\n        kindModifiers: ts.ScriptElementKindModifier.none,\n        displayParts,\n        documentation,\n      };\n    } else {\n      return this.tsLS.getCompletionEntryDetails(\n          componentContext.shimPath, componentContext.positionInShimFile, entryName, formatOptions,\n          /* source */ undefined, preferences);\n    }\n  }\n\n  /**\n   * Get the `ts.Symbol` of a specific completion for a property expression in a global context\n   * (e.g.\n   * `{{y|}}`).\n   */\n  private getGlobalPropertyExpressionCompletionSymbol(\n      this: PropertyExpressionCompletionBuilder, entryName: string): ts.Symbol|undefined {\n    const completions =\n        this.templateTypeChecker.getGlobalCompletions(this.template, this.component);\n    if (completions === null) {\n      return undefined;\n    }\n    const {componentContext, templateContext} = completions;\n    if (templateContext.has(entryName)) {\n      const node: TmplAstReference|TmplAstVariable = templateContext.get(entryName)!.node;\n      const symbol = this.templateTypeChecker.getSymbolOfNode(node, this.component) as\n              TemplateDeclarationSymbol |\n          null;\n      if (symbol === null || symbol.tsSymbol === null) {\n        return undefined;\n      }\n      return symbol.tsSymbol;\n    } else {\n      return this.tsLS.getCompletionEntrySymbol(\n          componentContext.shimPath, componentContext.positionInShimFile, entryName,\n          /* source */ undefined);\n    }\n  }\n\n  private isElementTagCompletion(): this is CompletionBuilder<TmplAstElement> {\n    return this.node instanceof TmplAstElement &&\n        this.nodeContext === CompletionNodeContext.ElementTag;\n  }\n\n  private getElementTagCompletion(this: CompletionBuilder<TmplAstElement>):\n      ts.WithMetadata<ts.CompletionInfo>|undefined {\n    const templateTypeChecker = this.compiler.getTemplateTypeChecker();\n\n    // The replacementSpan is the tag name.\n    const replacementSpan: ts.TextSpan = {\n      start: this.node.sourceSpan.start.offset + 1,  // account for leading '<'\n      length: this.node.name.length,\n    };\n\n    const entries: ts.CompletionEntry[] =\n        Array.from(templateTypeChecker.getPotentialElementTags(this.component))\n            .map(([tag, directive]) => ({\n                   kind: tagCompletionKind(directive),\n                   name: tag,\n                   sortText: tag,\n                   replacementSpan,\n                 }));\n\n    return {\n      entries,\n      isGlobalCompletion: false,\n      isMemberCompletion: false,\n      isNewIdentifierLocation: false,\n    };\n  }\n\n  private getElementTagCompletionDetails(\n      this: CompletionBuilder<TmplAstElement>, entryName: string): ts.CompletionEntryDetails\n      |undefined {\n    const templateTypeChecker = this.compiler.getTemplateTypeChecker();\n\n    const tagMap = templateTypeChecker.getPotentialElementTags(this.component);\n    if (!tagMap.has(entryName)) {\n      return undefined;\n    }\n\n    const directive = tagMap.get(entryName)!;\n    let displayParts: ts.SymbolDisplayPart[];\n    let documentation: ts.SymbolDisplayPart[]|undefined = undefined;\n    if (directive === null) {\n      displayParts = [];\n    } else {\n      const displayInfo = getDirectiveDisplayInfo(this.tsLS, directive);\n      displayParts = displayInfo.displayParts;\n      documentation = displayInfo.documentation;\n    }\n\n    return {\n      kind: tagCompletionKind(directive),\n      name: entryName,\n      kindModifiers: ts.ScriptElementKindModifier.none,\n      displayParts,\n      documentation,\n    };\n  }\n\n  private getElementTagCompletionSymbol(this: CompletionBuilder<TmplAstElement>, entryName: string):\n      ts.Symbol|undefined {\n    const templateTypeChecker = this.compiler.getTemplateTypeChecker();\n\n    const tagMap = templateTypeChecker.getPotentialElementTags(this.component);\n    if (!tagMap.has(entryName)) {\n      return undefined;\n    }\n\n    const directive = tagMap.get(entryName)!;\n    return directive?.tsSymbol;\n  }\n\n  private isElementAttributeCompletion(): this is ElementAttributeCompletionBuilder {\n    return (this.nodeContext === CompletionNodeContext.ElementAttributeKey ||\n            this.nodeContext === CompletionNodeContext.TwoWayBinding) &&\n        (this.node instanceof TmplAstElement || this.node instanceof TmplAstBoundAttribute ||\n         this.node instanceof TmplAstTextAttribute || this.node instanceof TmplAstBoundEvent);\n  }\n\n  private getElementAttributeCompletions(this: ElementAttributeCompletionBuilder):\n      ts.WithMetadata<ts.CompletionInfo>|undefined {\n    let element: TmplAstElement|TmplAstTemplate;\n    if (this.node instanceof TmplAstElement) {\n      element = this.node;\n    } else if (\n        this.nodeParent instanceof TmplAstElement || this.nodeParent instanceof TmplAstTemplate) {\n      element = this.nodeParent;\n    } else {\n      // Nothing to do without an element to process.\n      return undefined;\n    }\n\n    let replacementSpan: ts.TextSpan|undefined = undefined;\n    if ((this.node instanceof TmplAstBoundAttribute || this.node instanceof TmplAstBoundEvent ||\n         this.node instanceof TmplAstTextAttribute) &&\n        this.node.keySpan !== undefined) {\n      replacementSpan = makeReplacementSpanFromParseSourceSpan(this.node.keySpan);\n    }\n\n    const attrTable = buildAttributeCompletionTable(\n        this.component, element, this.compiler.getTemplateTypeChecker());\n\n    let entries: ts.CompletionEntry[] = [];\n\n    for (const completion of attrTable.values()) {\n      // First, filter out completions that don't make sense for the current node. For example, if\n      // the user is completing on a property binding `[foo|]`, don't offer output event\n      // completions.\n      switch (completion.kind) {\n        case AttributeCompletionKind.DomAttribute:\n        case AttributeCompletionKind.DomProperty:\n          if (this.node instanceof TmplAstBoundEvent) {\n            continue;\n          }\n          break;\n        case AttributeCompletionKind.DirectiveInput:\n          if (this.node instanceof TmplAstBoundEvent) {\n            continue;\n          }\n          if (!completion.twoWayBindingSupported &&\n              this.nodeContext === CompletionNodeContext.TwoWayBinding) {\n            continue;\n          }\n          break;\n        case AttributeCompletionKind.DirectiveOutput:\n          if (this.node instanceof TmplAstBoundAttribute) {\n            continue;\n          }\n          break;\n        case AttributeCompletionKind.DirectiveAttribute:\n          if (this.node instanceof TmplAstBoundAttribute ||\n              this.node instanceof TmplAstBoundEvent) {\n            continue;\n          }\n          break;\n      }\n\n      // Is the completion in an attribute context (instead of a property context)?\n      const isAttributeContext =\n          (this.node instanceof TmplAstElement || this.node instanceof TmplAstTextAttribute);\n      // Is the completion for an element (not an <ng-template>)?\n      const isElementContext =\n          this.node instanceof TmplAstElement || this.nodeParent instanceof TmplAstElement;\n      addAttributeCompletionEntries(\n          entries, completion, isAttributeContext, isElementContext, replacementSpan);\n    }\n\n    return {\n      entries,\n      isGlobalCompletion: false,\n      isMemberCompletion: false,\n      isNewIdentifierLocation: true,\n    };\n  }\n\n  private getElementAttributeCompletionDetails(\n      this: ElementAttributeCompletionBuilder, entryName: string): ts.CompletionEntryDetails\n      |undefined {\n    // `entryName` here may be `foo` or `[foo]`, depending on which suggested completion the user\n    // chose. Strip off any binding syntax to get the real attribute name.\n    const {name, kind} = stripBindingSugar(entryName);\n\n    let element: TmplAstElement|TmplAstTemplate;\n    if (this.node instanceof TmplAstElement || this.node instanceof TmplAstTemplate) {\n      element = this.node;\n    } else if (\n        this.nodeParent instanceof TmplAstElement || this.nodeParent instanceof TmplAstTemplate) {\n      element = this.nodeParent;\n    } else {\n      // Nothing to do without an element to process.\n      return undefined;\n    }\n\n    const attrTable = buildAttributeCompletionTable(\n        this.component, element, this.compiler.getTemplateTypeChecker());\n\n    if (!attrTable.has(name)) {\n      return undefined;\n    }\n\n    const completion = attrTable.get(name)!;\n    let displayParts: ts.SymbolDisplayPart[];\n    let documentation: ts.SymbolDisplayPart[]|undefined = undefined;\n    let info: DisplayInfo|null;\n    switch (completion.kind) {\n      case AttributeCompletionKind.DomAttribute:\n      case AttributeCompletionKind.DomProperty:\n        // TODO(alxhub): ideally we would show the same documentation as quick info here. However,\n        // since these bindings don't exist in the TCB, there is no straightforward way to retrieve\n        // a `ts.Symbol` for the field in the TS DOM definition.\n        displayParts = [];\n        break;\n      case AttributeCompletionKind.DirectiveAttribute:\n        info = getDirectiveDisplayInfo(this.tsLS, completion.directive);\n        displayParts = info.displayParts;\n        documentation = info.documentation;\n        break;\n      case AttributeCompletionKind.DirectiveInput:\n      case AttributeCompletionKind.DirectiveOutput:\n        const propertySymbol = getAttributeCompletionSymbol(completion, this.typeChecker);\n        if (propertySymbol === null) {\n          return undefined;\n        }\n\n        info = getTsSymbolDisplayInfo(\n            this.tsLS, this.typeChecker, propertySymbol,\n            completion.kind === AttributeCompletionKind.DirectiveInput ? DisplayInfoKind.PROPERTY :\n                                                                         DisplayInfoKind.EVENT,\n            completion.directive.tsSymbol.name);\n        if (info === null) {\n          return undefined;\n        }\n        displayParts = info.displayParts;\n        documentation = info.documentation;\n    }\n\n    return {\n      name: entryName,\n      kind: unsafeCastDisplayInfoKindToScriptElementKind(kind),\n      kindModifiers: ts.ScriptElementKindModifier.none,\n      displayParts: [],\n      documentation,\n    };\n  }\n\n  private getElementAttributeCompletionSymbol(\n      this: ElementAttributeCompletionBuilder, attribute: string): ts.Symbol|undefined {\n    const {name} = stripBindingSugar(attribute);\n\n    let element: TmplAstElement|TmplAstTemplate;\n    if (this.node instanceof TmplAstElement || this.node instanceof TmplAstTemplate) {\n      element = this.node;\n    } else if (\n        this.nodeParent instanceof TmplAstElement || this.nodeParent instanceof TmplAstTemplate) {\n      element = this.nodeParent;\n    } else {\n      // Nothing to do without an element to process.\n      return undefined;\n    }\n\n    const attrTable = buildAttributeCompletionTable(\n        this.component, element, this.compiler.getTemplateTypeChecker());\n\n    if (!attrTable.has(name)) {\n      return undefined;\n    }\n\n    const completion = attrTable.get(name)!;\n    return getAttributeCompletionSymbol(completion, this.typeChecker) ?? undefined;\n  }\n\n  private isPipeCompletion(): this is PipeCompletionBuilder {\n    return this.node instanceof BindingPipe;\n  }\n\n  private getPipeCompletions(this: PipeCompletionBuilder):\n      ts.WithMetadata<ts.CompletionInfo>|undefined {\n    const pipes = this.templateTypeChecker.getPipesInScope(this.component);\n    if (pipes === null) {\n      return undefined;\n    }\n\n    const replacementSpan = makeReplacementSpanFromAst(this.node);\n\n    const entries: ts.CompletionEntry[] =\n        pipes.map(pipe => ({\n                    name: pipe.name,\n                    sortText: pipe.name,\n                    kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.PIPE),\n                    replacementSpan,\n                  }));\n    return {\n      entries,\n      isGlobalCompletion: false,\n      isMemberCompletion: false,\n      isNewIdentifierLocation: false,\n    };\n  }\n}\n\nfunction makeReplacementSpanFromParseSourceSpan(span: ParseSourceSpan): ts.TextSpan {\n  return {\n    start: span.start.offset,\n    length: span.end.offset - span.start.offset,\n  };\n}\n\nfunction makeReplacementSpanFromAst(node: PropertyRead|PropertyWrite|MethodCall|SafePropertyRead|\n                                    SafeMethodCall|BindingPipe|EmptyExpr|LiteralPrimitive|\n                                    BoundEvent): ts.TextSpan|undefined {\n  if ((node instanceof EmptyExpr || node instanceof LiteralPrimitive ||\n       node instanceof BoundEvent)) {\n    // empty nodes do not replace any existing text\n    return undefined;\n  }\n\n  return {\n    start: node.nameSpan.start,\n    length: node.nameSpan.end - node.nameSpan.start,\n  };\n}\n\nfunction tagCompletionKind(directive: DirectiveInScope|null): ts.ScriptElementKind {\n  let kind: DisplayInfoKind;\n  if (directive === null) {\n    kind = DisplayInfoKind.ELEMENT;\n  } else if (directive.isComponent) {\n    kind = DisplayInfoKind.COMPONENT;\n  } else {\n    kind = DisplayInfoKind.DIRECTIVE;\n  }\n  return unsafeCastDisplayInfoKindToScriptElementKind(kind);\n}\n\nconst BINDING_SUGAR = /[\\[\\(\\)\\]]/g;\n\nfunction stripBindingSugar(binding: string): {name: string, kind: DisplayInfoKind} {\n  const name = binding.replace(BINDING_SUGAR, '');\n  if (binding.startsWith('[')) {\n    return {name, kind: DisplayInfoKind.PROPERTY};\n  } else if (binding.startsWith('(')) {\n    return {name, kind: DisplayInfoKind.EVENT};\n  } else {\n    return {name, kind: DisplayInfoKind.ATTRIBUTE};\n  }\n}\n"]}
\No newline at end of file