UNPKG

74.5 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/attribute_completions", ["require", "exports", "tslib", "@angular/compiler", "typescript", "@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.getAttributeCompletionSymbol = exports.addAttributeCompletionEntries = exports.buildAttributeCompletionTable = exports.AttributeCompletionKind = void 0;
20 var tslib_1 = require("tslib");
21 var compiler_1 = require("@angular/compiler");
22 var ts = require("typescript");
23 var display_parts_1 = require("@angular/language-service/ivy/display_parts");
24 var utils_1 = require("@angular/language-service/ivy/utils");
25 /**
26 * Differentiates different kinds of `AttributeCompletion`s.
27 */
28 var AttributeCompletionKind;
29 (function (AttributeCompletionKind) {
30 /**
31 * Completion of an attribute from the HTML schema.
32 *
33 * Attributes often have a corresponding DOM property of the same name.
34 */
35 AttributeCompletionKind[AttributeCompletionKind["DomAttribute"] = 0] = "DomAttribute";
36 /**
37 * Completion of a property from the DOM schema.
38 *
39 * `DomProperty` completions are generated only for properties which don't share their name with
40 * an HTML attribute.
41 */
42 AttributeCompletionKind[AttributeCompletionKind["DomProperty"] = 1] = "DomProperty";
43 /**
44 * Completion of an attribute that results in a new directive being matched on an element.
45 */
46 AttributeCompletionKind[AttributeCompletionKind["DirectiveAttribute"] = 2] = "DirectiveAttribute";
47 /**
48 * Completion of an attribute that results in a new structural directive being matched on an
49 * element.
50 */
51 AttributeCompletionKind[AttributeCompletionKind["StructuralDirectiveAttribute"] = 3] = "StructuralDirectiveAttribute";
52 /**
53 * Completion of an input from a directive which is either present on the element, or becomes
54 * present after the addition of this attribute.
55 */
56 AttributeCompletionKind[AttributeCompletionKind["DirectiveInput"] = 4] = "DirectiveInput";
57 /**
58 * Completion of an output from a directive which is either present on the element, or becomes
59 * present after the addition of this attribute.
60 */
61 AttributeCompletionKind[AttributeCompletionKind["DirectiveOutput"] = 5] = "DirectiveOutput";
62 })(AttributeCompletionKind = exports.AttributeCompletionKind || (exports.AttributeCompletionKind = {}));
63 /**
64 * Given an element and its context, produce a `Map` of all possible attribute completions.
65 *
66 * 3 kinds of attributes are considered for completion, from highest to lowest priority:
67 *
68 * 1. Inputs/outputs of directives present on the element already.
69 * 2. Inputs/outputs of directives that are not present on the element, but which would become
70 * present if such a binding is added.
71 * 3. Attributes from the DOM schema for the element.
72 *
73 * The priority of these options determines which completions are added to the `Map`. If a directive
74 * input shares the same name as a DOM attribute, the `Map` will reflect the directive input
75 * completion, not the DOM completion for that name.
76 */
77 function buildAttributeCompletionTable(component, element, checker) {
78 var e_1, _a, e_2, _b, e_3, _c, e_4, _d, e_5, _e, e_6, _f, e_7, _g, e_8, _h;
79 var table = new Map();
80 // Use the `ElementSymbol` or `TemplateSymbol` to iterate over directives present on the node, and
81 // their inputs/outputs. These have the highest priority of completion results.
82 var symbol = checker.getSymbolOfNode(element, component);
83 var presentDirectives = new Set();
84 if (symbol !== null) {
85 try {
86 // An `ElementSymbol` was available. This means inputs and outputs for directives on the
87 // element can be added to the completion table.
88 for (var _j = tslib_1.__values(symbol.directives), _k = _j.next(); !_k.done; _k = _j.next()) {
89 var dirSymbol = _k.value;
90 var directive = dirSymbol.tsSymbol.valueDeclaration;
91 if (!ts.isClassDeclaration(directive)) {
92 continue;
93 }
94 presentDirectives.add(directive);
95 var meta = checker.getDirectiveMetadata(directive);
96 if (meta === null) {
97 continue;
98 }
99 try {
100 for (var _l = (e_2 = void 0, tslib_1.__values(meta.inputs)), _m = _l.next(); !_m.done; _m = _l.next()) {
101 var _o = tslib_1.__read(_m.value, 2), propertyName = _o[0], classPropertyName = _o[1];
102 if (table.has(propertyName)) {
103 continue;
104 }
105 table.set(propertyName, {
106 kind: AttributeCompletionKind.DirectiveInput,
107 propertyName: propertyName,
108 directive: dirSymbol,
109 classPropertyName: classPropertyName,
110 twoWayBindingSupported: meta.outputs.hasBindingPropertyName(propertyName + 'Change'),
111 });
112 }
113 }
114 catch (e_2_1) { e_2 = { error: e_2_1 }; }
115 finally {
116 try {
117 if (_m && !_m.done && (_b = _l.return)) _b.call(_l);
118 }
119 finally { if (e_2) throw e_2.error; }
120 }
121 try {
122 for (var _p = (e_3 = void 0, tslib_1.__values(meta.outputs)), _q = _p.next(); !_q.done; _q = _p.next()) {
123 var _r = tslib_1.__read(_q.value, 2), propertyName = _r[0], classPropertyName = _r[1];
124 if (table.has(propertyName)) {
125 continue;
126 }
127 table.set(propertyName, {
128 kind: AttributeCompletionKind.DirectiveOutput,
129 eventName: propertyName,
130 directive: dirSymbol,
131 classPropertyName: classPropertyName,
132 });
133 }
134 }
135 catch (e_3_1) { e_3 = { error: e_3_1 }; }
136 finally {
137 try {
138 if (_q && !_q.done && (_c = _p.return)) _c.call(_p);
139 }
140 finally { if (e_3) throw e_3.error; }
141 }
142 }
143 }
144 catch (e_1_1) { e_1 = { error: e_1_1 }; }
145 finally {
146 try {
147 if (_k && !_k.done && (_a = _j.return)) _a.call(_j);
148 }
149 finally { if (e_1) throw e_1.error; }
150 }
151 }
152 // Next, explore hypothetical directives and determine if the addition of any single attributes
153 // can cause the directive to match the element.
154 var directivesInScope = checker.getDirectivesInScope(component);
155 if (directivesInScope !== null) {
156 var elementSelector = utils_1.makeElementSelector(element);
157 try {
158 for (var directivesInScope_1 = tslib_1.__values(directivesInScope), directivesInScope_1_1 = directivesInScope_1.next(); !directivesInScope_1_1.done; directivesInScope_1_1 = directivesInScope_1.next()) {
159 var dirInScope = directivesInScope_1_1.value;
160 var directive = dirInScope.tsSymbol.valueDeclaration;
161 // Skip directives that are present on the element.
162 if (!ts.isClassDeclaration(directive) || presentDirectives.has(directive)) {
163 continue;
164 }
165 var meta = checker.getDirectiveMetadata(directive);
166 if (meta === null || meta.selector === null) {
167 continue;
168 }
169 if (!meta.isStructural) {
170 // For non-structural directives, the directive's attribute selector(s) are matched against
171 // a hypothetical version of the element with those attributes. A match indicates that
172 // adding that attribute/input/output binding would cause the directive to become present,
173 // meaning that such a binding is a valid completion.
174 var selectors = compiler_1.CssSelector.parse(meta.selector);
175 var matcher = new compiler_1.SelectorMatcher();
176 matcher.addSelectables(selectors);
177 try {
178 for (var selectors_1 = (e_5 = void 0, tslib_1.__values(selectors)), selectors_1_1 = selectors_1.next(); !selectors_1_1.done; selectors_1_1 = selectors_1.next()) {
179 var selector = selectors_1_1.value;
180 try {
181 for (var _s = (e_6 = void 0, tslib_1.__values(selectorAttributes(selector))), _t = _s.next(); !_t.done; _t = _s.next()) {
182 var _u = tslib_1.__read(_t.value, 2), attrName = _u[0], attrValue = _u[1];
183 if (attrValue !== '') {
184 // This attribute selector requires a value, which is not supported in completion.
185 continue;
186 }
187 if (table.has(attrName)) {
188 // Skip this attribute as there's already a binding for it.
189 continue;
190 }
191 // Check whether adding this attribute would cause the directive to start matching.
192 var newElementSelector = elementSelector + ("[" + attrName + "]");
193 if (!matcher.match(compiler_1.CssSelector.parse(newElementSelector)[0], null)) {
194 // Nope, move on with our lives.
195 continue;
196 }
197 // Adding this attribute causes a new directive to be matched. Decide how to categorize
198 // it based on the directive's inputs and outputs.
199 if (meta.inputs.hasBindingPropertyName(attrName)) {
200 // This attribute corresponds to an input binding.
201 table.set(attrName, {
202 kind: AttributeCompletionKind.DirectiveInput,
203 directive: dirInScope,
204 propertyName: attrName,
205 classPropertyName: meta.inputs.getByBindingPropertyName(attrName)[0].classPropertyName,
206 twoWayBindingSupported: meta.outputs.hasBindingPropertyName(attrName + 'Change'),
207 });
208 }
209 else if (meta.outputs.hasBindingPropertyName(attrName)) {
210 // This attribute corresponds to an output binding.
211 table.set(attrName, {
212 kind: AttributeCompletionKind.DirectiveOutput,
213 directive: dirInScope,
214 eventName: attrName,
215 classPropertyName: meta.outputs.getByBindingPropertyName(attrName)[0].classPropertyName,
216 });
217 }
218 else {
219 // This attribute causes a new directive to be matched, but does not also correspond
220 // to an input or output binding.
221 table.set(attrName, {
222 kind: AttributeCompletionKind.DirectiveAttribute,
223 attribute: attrName,
224 directive: dirInScope,
225 });
226 }
227 }
228 }
229 catch (e_6_1) { e_6 = { error: e_6_1 }; }
230 finally {
231 try {
232 if (_t && !_t.done && (_f = _s.return)) _f.call(_s);
233 }
234 finally { if (e_6) throw e_6.error; }
235 }
236 }
237 }
238 catch (e_5_1) { e_5 = { error: e_5_1 }; }
239 finally {
240 try {
241 if (selectors_1_1 && !selectors_1_1.done && (_e = selectors_1.return)) _e.call(selectors_1);
242 }
243 finally { if (e_5) throw e_5.error; }
244 }
245 }
246 else {
247 // Hypothetically matching a structural directive is a litle different than a plain
248 // directive. Use of the '*' structural directive syntactic sugar means that the actual
249 // directive is applied to a plain <ng-template> node, not the existing element with any
250 // other attributes it might already have.
251 // Additionally, more than one attribute/input might need to be present in order for the
252 // directive to match (e.g. `ngFor` has a selector of `[ngFor][ngForOf]`). This gets a
253 // little tricky.
254 var structuralAttributes = getStructuralAttributes(meta);
255 try {
256 for (var structuralAttributes_1 = (e_7 = void 0, tslib_1.__values(structuralAttributes)), structuralAttributes_1_1 = structuralAttributes_1.next(); !structuralAttributes_1_1.done; structuralAttributes_1_1 = structuralAttributes_1.next()) {
257 var attrName = structuralAttributes_1_1.value;
258 table.set(attrName, {
259 kind: AttributeCompletionKind.StructuralDirectiveAttribute,
260 attribute: attrName,
261 directive: dirInScope,
262 });
263 }
264 }
265 catch (e_7_1) { e_7 = { error: e_7_1 }; }
266 finally {
267 try {
268 if (structuralAttributes_1_1 && !structuralAttributes_1_1.done && (_g = structuralAttributes_1.return)) _g.call(structuralAttributes_1);
269 }
270 finally { if (e_7) throw e_7.error; }
271 }
272 }
273 }
274 }
275 catch (e_4_1) { e_4 = { error: e_4_1 }; }
276 finally {
277 try {
278 if (directivesInScope_1_1 && !directivesInScope_1_1.done && (_d = directivesInScope_1.return)) _d.call(directivesInScope_1);
279 }
280 finally { if (e_4) throw e_4.error; }
281 }
282 }
283 // Finally, add any DOM attributes not already covered by inputs.
284 if (element instanceof compiler_1.TmplAstElement) {
285 try {
286 for (var _v = tslib_1.__values(checker.getPotentialDomBindings(element.name)), _w = _v.next(); !_w.done; _w = _v.next()) {
287 var _x = _w.value, attribute = _x.attribute, property = _x.property;
288 var isAlsoProperty = attribute === property;
289 if (!table.has(attribute)) {
290 table.set(attribute, {
291 kind: AttributeCompletionKind.DomAttribute,
292 attribute: attribute,
293 isAlsoProperty: isAlsoProperty,
294 });
295 }
296 if (!isAlsoProperty && !table.has(property)) {
297 table.set(property, {
298 kind: AttributeCompletionKind.DomProperty,
299 property: property,
300 });
301 }
302 }
303 }
304 catch (e_8_1) { e_8 = { error: e_8_1 }; }
305 finally {
306 try {
307 if (_w && !_w.done && (_h = _v.return)) _h.call(_v);
308 }
309 finally { if (e_8) throw e_8.error; }
310 }
311 }
312 return table;
313 }
314 exports.buildAttributeCompletionTable = buildAttributeCompletionTable;
315 /**
316 * Given an `AttributeCompletion`, add any available completions to a `ts.CompletionEntry` array of
317 * results.
318 *
319 * The kind of completions generated depends on whether the current context is an attribute context
320 * or not. For example, completing on `<element attr|>` will generate two results: `attribute` and
321 * `[attribute]` - either a static attribute can be generated, or a property binding. However,
322 * `<element [attr|]>` is not an attribute context, and so only the property completion `attribute`
323 * is generated. Note that this completion does not have the `[]` property binding sugar as its
324 * implicitly present in a property binding context (we're already completing within an `[attr|]`
325 * expression).
326 */
327 function addAttributeCompletionEntries(entries, completion, isAttributeContext, isElementContext, replacementSpan) {
328 switch (completion.kind) {
329 case AttributeCompletionKind.DirectiveAttribute: {
330 entries.push({
331 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.DIRECTIVE),
332 name: completion.attribute,
333 sortText: completion.attribute,
334 replacementSpan: replacementSpan,
335 });
336 break;
337 }
338 case AttributeCompletionKind.StructuralDirectiveAttribute: {
339 // In an element, the completion is offered with a leading '*' to activate the structural
340 // directive. Once present, the structural attribute will be parsed as a template and not an
341 // element, and the prefix is no longer necessary.
342 var prefix = isElementContext ? '*' : '';
343 entries.push({
344 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.DIRECTIVE),
345 name: prefix + completion.attribute,
346 sortText: prefix + completion.attribute,
347 replacementSpan: replacementSpan,
348 });
349 break;
350 }
351 case AttributeCompletionKind.DirectiveInput: {
352 if (isAttributeContext) {
353 // Offer a completion of a property binding.
354 entries.push({
355 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.PROPERTY),
356 name: "[" + completion.propertyName + "]",
357 sortText: completion.propertyName,
358 replacementSpan: replacementSpan,
359 });
360 // If the directive supports banana-in-a-box for this input, offer that as well.
361 if (completion.twoWayBindingSupported) {
362 entries.push({
363 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.PROPERTY),
364 name: "[(" + completion.propertyName + ")]",
365 // This completion should sort after the property binding.
366 sortText: completion.propertyName + '_1',
367 replacementSpan: replacementSpan,
368 });
369 }
370 // Offer a completion of the input binding as an attribute.
371 entries.push({
372 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.ATTRIBUTE),
373 name: completion.propertyName,
374 // This completion should sort after both property binding options (one-way and two-way).
375 sortText: completion.propertyName + '_2',
376 replacementSpan: replacementSpan,
377 });
378 }
379 else {
380 entries.push({
381 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.PROPERTY),
382 name: completion.propertyName,
383 sortText: completion.propertyName,
384 replacementSpan: replacementSpan,
385 });
386 }
387 break;
388 }
389 case AttributeCompletionKind.DirectiveOutput: {
390 if (isAttributeContext) {
391 entries.push({
392 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.EVENT),
393 name: "(" + completion.eventName + ")",
394 sortText: completion.eventName,
395 replacementSpan: replacementSpan,
396 });
397 }
398 else {
399 entries.push({
400 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.EVENT),
401 name: completion.eventName,
402 sortText: completion.eventName,
403 replacementSpan: replacementSpan,
404 });
405 }
406 break;
407 }
408 case AttributeCompletionKind.DomAttribute: {
409 if (isAttributeContext) {
410 // Offer a completion of an attribute binding.
411 entries.push({
412 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.ATTRIBUTE),
413 name: completion.attribute,
414 sortText: completion.attribute,
415 replacementSpan: replacementSpan,
416 });
417 if (completion.isAlsoProperty) {
418 // Offer a completion of a property binding to the DOM property.
419 entries.push({
420 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.PROPERTY),
421 name: "[" + completion.attribute + "]",
422 // In the case of DOM attributes, the property binding should sort after the attribute
423 // binding.
424 sortText: completion.attribute + '_1',
425 replacementSpan: replacementSpan,
426 });
427 }
428 }
429 else if (completion.isAlsoProperty) {
430 entries.push({
431 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.PROPERTY),
432 name: completion.attribute,
433 sortText: completion.attribute,
434 replacementSpan: replacementSpan,
435 });
436 }
437 break;
438 }
439 case AttributeCompletionKind.DomProperty: {
440 if (!isAttributeContext) {
441 entries.push({
442 kind: display_parts_1.unsafeCastDisplayInfoKindToScriptElementKind(display_parts_1.DisplayInfoKind.PROPERTY),
443 name: completion.property,
444 sortText: completion.property,
445 replacementSpan: replacementSpan,
446 });
447 }
448 }
449 }
450 }
451 exports.addAttributeCompletionEntries = addAttributeCompletionEntries;
452 function getAttributeCompletionSymbol(completion, checker) {
453 var _a;
454 switch (completion.kind) {
455 case AttributeCompletionKind.DomAttribute:
456 case AttributeCompletionKind.DomProperty:
457 return null;
458 case AttributeCompletionKind.DirectiveAttribute:
459 case AttributeCompletionKind.StructuralDirectiveAttribute:
460 return completion.directive.tsSymbol;
461 case AttributeCompletionKind.DirectiveInput:
462 case AttributeCompletionKind.DirectiveOutput:
463 return (_a = checker.getDeclaredTypeOfSymbol(completion.directive.tsSymbol)
464 .getProperty(completion.classPropertyName)) !== null && _a !== void 0 ? _a : null;
465 }
466 }
467 exports.getAttributeCompletionSymbol = getAttributeCompletionSymbol;
468 /**
469 * Iterates over `CssSelector` attributes, which are internally represented in a zipped array style
470 * which is not conducive to straightforward iteration.
471 */
472 function selectorAttributes(selector) {
473 var i;
474 return tslib_1.__generator(this, function (_a) {
475 switch (_a.label) {
476 case 0:
477 i = 0;
478 _a.label = 1;
479 case 1:
480 if (!(i < selector.attrs.length)) return [3 /*break*/, 4];
481 return [4 /*yield*/, [selector.attrs[0], selector.attrs[1]]];
482 case 2:
483 _a.sent();
484 _a.label = 3;
485 case 3:
486 i += 2;
487 return [3 /*break*/, 1];
488 case 4: return [2 /*return*/];
489 }
490 });
491 }
492 function getStructuralAttributes(meta) {
493 var e_9, _a;
494 if (meta.selector === null) {
495 return [];
496 }
497 var structuralAttributes = [];
498 var selectors = compiler_1.CssSelector.parse(meta.selector);
499 var _loop_1 = function (selector) {
500 if (selector.element !== null && selector.element !== 'ng-template') {
501 return "continue";
502 }
503 // Every attribute of this selector must be name-only - no required values.
504 var attributeSelectors = Array.from(selectorAttributes(selector));
505 if (!attributeSelectors.every(function (_a) {
506 var _b = tslib_1.__read(_a, 2), _ = _b[0], attrValue = _b[1];
507 return attrValue === '';
508 })) {
509 return "continue";
510 }
511 // Get every named selector.
512 var attributes = attributeSelectors.map(function (_a) {
513 var _b = tslib_1.__read(_a, 2), attrName = _b[0], _ = _b[1];
514 return attrName;
515 });
516 // Find the shortest attribute. This is the structural directive "base", and all potential
517 // input bindings must begin with the base. E.g. in `*ngFor="let a of b"`, `ngFor` is the
518 // base attribute, and the `of` binding key corresponds to an input of `ngForOf`.
519 var baseAttr = attributes.reduce(function (prev, curr) { return prev === null || curr.length < prev.length ? curr : prev; }, null);
520 if (baseAttr === null) {
521 return "continue";
522 }
523 // Validate that the attributes are compatible with use as a structural directive.
524 var isValid = function (attr) {
525 // The base attribute is valid by default.
526 if (attr === baseAttr) {
527 return true;
528 }
529 // Non-base attributes must all be prefixed with the base attribute.
530 if (!attr.startsWith(baseAttr)) {
531 return false;
532 }
533 // Non-base attributes must also correspond to directive inputs.
534 if (!meta.inputs.hasBindingPropertyName(attr)) {
535 return false;
536 }
537 // This attribute is compatible.
538 return true;
539 };
540 if (!attributes.every(isValid)) {
541 return "continue";
542 }
543 // This attribute is valid as a structural attribute for this directive.
544 structuralAttributes.push(baseAttr);
545 };
546 try {
547 for (var selectors_2 = tslib_1.__values(selectors), selectors_2_1 = selectors_2.next(); !selectors_2_1.done; selectors_2_1 = selectors_2.next()) {
548 var selector = selectors_2_1.value;
549 _loop_1(selector);
550 }
551 }
552 catch (e_9_1) { e_9 = { error: e_9_1 }; }
553 finally {
554 try {
555 if (selectors_2_1 && !selectors_2_1.done && (_a = selectors_2.return)) _a.call(selectors_2);
556 }
557 finally { if (e_9) throw e_9.error; }
558 }
559 return structuralAttributes;
560 }
561});
562//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attribute_completions.js","sourceRoot":"","sources":["../../../../../../packages/language-service/ivy/attribute_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 {CssSelector, SelectorMatcher, TmplAstElement, TmplAstTemplate} from '@angular/compiler';\nimport {DirectiveInScope, ElementSymbol, TemplateSymbol, TemplateTypeChecker, TypeCheckableDirectiveMeta} from '@angular/compiler-cli/src/ngtsc/typecheck/api';\nimport * as ts from 'typescript';\n\nimport {DisplayInfoKind, unsafeCastDisplayInfoKindToScriptElementKind} from './display_parts';\nimport {makeElementSelector} from './utils';\n\n/**\n * Differentiates different kinds of `AttributeCompletion`s.\n */\nexport enum AttributeCompletionKind {\n  /**\n   * Completion of an attribute from the HTML schema.\n   *\n   * Attributes often have a corresponding DOM property of the same name.\n   */\n  DomAttribute,\n\n  /**\n   * Completion of a property from the DOM schema.\n   *\n   * `DomProperty` completions are generated only for properties which don't share their name with\n   * an HTML attribute.\n   */\n  DomProperty,\n\n  /**\n   * Completion of an attribute that results in a new directive being matched on an element.\n   */\n  DirectiveAttribute,\n\n  /**\n   * Completion of an attribute that results in a new structural directive being matched on an\n   * element.\n   */\n  StructuralDirectiveAttribute,\n\n  /**\n   * Completion of an input from a directive which is either present on the element, or becomes\n   * present after the addition of this attribute.\n   */\n  DirectiveInput,\n\n  /**\n   * Completion of an output from a directive which is either present on the element, or becomes\n   * present after the addition of this attribute.\n   */\n  DirectiveOutput,\n}\n\n/**\n * Completion of an attribute from the DOM schema.\n */\nexport interface DomAttributeCompletion {\n  kind: AttributeCompletionKind.DomAttribute;\n\n  /**\n   * Name of the HTML attribute (not to be confused with the corresponding DOM property name).\n   */\n  attribute: string;\n\n  /**\n   * Whether this attribute is also a DOM property.\n   */\n  isAlsoProperty: boolean;\n}\n\n/**\n * Completion of a DOM property of an element that's distinct from an HTML attribute.\n */\nexport interface DomPropertyCompletion {\n  kind: AttributeCompletionKind.DomProperty;\n\n  /**\n   * Name of the DOM property\n   */\n  property: string;\n}\n\n/**\n * Completion of an attribute which results in a new directive being matched on an element.\n */\nexport interface DirectiveAttributeCompletion {\n  kind: AttributeCompletionKind.DirectiveAttribute|\n      AttributeCompletionKind.StructuralDirectiveAttribute;\n\n  /**\n   * Name of the attribute whose addition causes this directive to match the element.\n   */\n  attribute: string;\n\n  /**\n   * The directive whose selector gave rise to this completion.\n   */\n  directive: DirectiveInScope;\n}\n\n/**\n * Completion of an input of a directive which may either be present on the element, or become\n * present when a binding to this input is added.\n */\nexport interface DirectiveInputCompletion {\n  kind: AttributeCompletionKind.DirectiveInput;\n\n  /**\n   * The public property name of the input (the name which would be used in any binding to that\n   * input).\n   */\n  propertyName: string;\n\n  /**\n   * The directive which has this input.\n   */\n  directive: DirectiveInScope;\n\n  /**\n   * The field name on the directive class which corresponds to this input.\n   *\n   * Currently, in the case where a single property name corresponds to multiple input fields, only\n   * the first such field is represented here. In the future multiple results may be warranted.\n   */\n  classPropertyName: string;\n\n  /**\n   * Whether this input can be used with two-way binding (that is, whether a corresponding change\n   * output exists on the directive).\n   */\n  twoWayBindingSupported: boolean;\n}\n\nexport interface DirectiveOutputCompletion {\n  kind: AttributeCompletionKind.DirectiveOutput;\n\n  /**\n   * The public event name of the output (the name which would be used in any binding to that\n   * output).\n   */\n  eventName: string;\n\n  /**\n   *The directive which has this output.\n   */\n  directive: DirectiveInScope;\n\n  /**\n   * The field name on the directive class which corresponds to this output.\n   */\n  classPropertyName: string;\n}\n\n/**\n * Any named attribute which is available for completion on a given element.\n *\n * Disambiguated by the `kind` property into various types of completions.\n */\nexport type AttributeCompletion = DomAttributeCompletion|DomPropertyCompletion|\n    DirectiveAttributeCompletion|DirectiveInputCompletion|DirectiveOutputCompletion;\n\n/**\n * Given an element and its context, produce a `Map` of all possible attribute completions.\n *\n * 3 kinds of attributes are considered for completion, from highest to lowest priority:\n *\n * 1. Inputs/outputs of directives present on the element already.\n * 2. Inputs/outputs of directives that are not present on the element, but which would become\n *    present if such a binding is added.\n * 3. Attributes from the DOM schema for the element.\n *\n * The priority of these options determines which completions are added to the `Map`. If a directive\n * input shares the same name as a DOM attribute, the `Map` will reflect the directive input\n * completion, not the DOM completion for that name.\n */\nexport function buildAttributeCompletionTable(\n    component: ts.ClassDeclaration, element: TmplAstElement|TmplAstTemplate,\n    checker: TemplateTypeChecker): Map<string, AttributeCompletion> {\n  const table = new Map<string, AttributeCompletion>();\n\n  // Use the `ElementSymbol` or `TemplateSymbol` to iterate over directives present on the node, and\n  // their inputs/outputs. These have the highest priority of completion results.\n  const symbol: ElementSymbol|TemplateSymbol =\n      checker.getSymbolOfNode(element, component) as ElementSymbol | TemplateSymbol;\n  const presentDirectives = new Set<ts.ClassDeclaration>();\n  if (symbol !== null) {\n    // An `ElementSymbol` was available. This means inputs and outputs for directives on the\n    // element can be added to the completion table.\n    for (const dirSymbol of symbol.directives) {\n      const directive = dirSymbol.tsSymbol.valueDeclaration;\n      if (!ts.isClassDeclaration(directive)) {\n        continue;\n      }\n      presentDirectives.add(directive);\n\n      const meta = checker.getDirectiveMetadata(directive);\n      if (meta === null) {\n        continue;\n      }\n\n      for (const [propertyName, classPropertyName] of meta.inputs) {\n        if (table.has(propertyName)) {\n          continue;\n        }\n\n        table.set(propertyName, {\n          kind: AttributeCompletionKind.DirectiveInput,\n          propertyName,\n          directive: dirSymbol,\n          classPropertyName,\n          twoWayBindingSupported: meta.outputs.hasBindingPropertyName(propertyName + 'Change'),\n        });\n      }\n\n      for (const [propertyName, classPropertyName] of meta.outputs) {\n        if (table.has(propertyName)) {\n          continue;\n        }\n\n        table.set(propertyName, {\n          kind: AttributeCompletionKind.DirectiveOutput,\n          eventName: propertyName,\n          directive: dirSymbol,\n          classPropertyName,\n        });\n      }\n    }\n  }\n\n  // Next, explore hypothetical directives and determine if the addition of any single attributes\n  // can cause the directive to match the element.\n  const directivesInScope = checker.getDirectivesInScope(component);\n  if (directivesInScope !== null) {\n    const elementSelector = makeElementSelector(element);\n\n    for (const dirInScope of directivesInScope) {\n      const directive = dirInScope.tsSymbol.valueDeclaration;\n      // Skip directives that are present on the element.\n      if (!ts.isClassDeclaration(directive) || presentDirectives.has(directive)) {\n        continue;\n      }\n\n      const meta = checker.getDirectiveMetadata(directive);\n      if (meta === null || meta.selector === null) {\n        continue;\n      }\n\n      if (!meta.isStructural) {\n        // For non-structural directives, the directive's attribute selector(s) are matched against\n        // a hypothetical version of the element with those attributes. A match indicates that\n        // adding that attribute/input/output binding would cause the directive to become present,\n        // meaning that such a binding is a valid completion.\n        const selectors = CssSelector.parse(meta.selector);\n        const matcher = new SelectorMatcher();\n        matcher.addSelectables(selectors);\n\n        for (const selector of selectors) {\n          for (const [attrName, attrValue] of selectorAttributes(selector)) {\n            if (attrValue !== '') {\n              // This attribute selector requires a value, which is not supported in completion.\n              continue;\n            }\n\n            if (table.has(attrName)) {\n              // Skip this attribute as there's already a binding for it.\n              continue;\n            }\n\n            // Check whether adding this attribute would cause the directive to start matching.\n            const newElementSelector = elementSelector + `[${attrName}]`;\n            if (!matcher.match(CssSelector.parse(newElementSelector)[0], null)) {\n              // Nope, move on with our lives.\n              continue;\n            }\n\n            // Adding this attribute causes a new directive to be matched. Decide how to categorize\n            // it based on the directive's inputs and outputs.\n            if (meta.inputs.hasBindingPropertyName(attrName)) {\n              // This attribute corresponds to an input binding.\n              table.set(attrName, {\n                kind: AttributeCompletionKind.DirectiveInput,\n                directive: dirInScope,\n                propertyName: attrName,\n                classPropertyName:\n                    meta.inputs.getByBindingPropertyName(attrName)![0].classPropertyName,\n                twoWayBindingSupported: meta.outputs.hasBindingPropertyName(attrName + 'Change'),\n              });\n            } else if (meta.outputs.hasBindingPropertyName(attrName)) {\n              // This attribute corresponds to an output binding.\n              table.set(attrName, {\n                kind: AttributeCompletionKind.DirectiveOutput,\n                directive: dirInScope,\n                eventName: attrName,\n                classPropertyName:\n                    meta.outputs.getByBindingPropertyName(attrName)![0].classPropertyName,\n              });\n            } else {\n              // This attribute causes a new directive to be matched, but does not also correspond\n              // to an input or output binding.\n              table.set(attrName, {\n                kind: AttributeCompletionKind.DirectiveAttribute,\n                attribute: attrName,\n                directive: dirInScope,\n              });\n            }\n          }\n        }\n      } else {\n        // Hypothetically matching a structural directive is a litle different than a plain\n        // directive. Use of the '*' structural directive syntactic sugar means that the actual\n        // directive is applied to a plain <ng-template> node, not the existing element with any\n        // other attributes it might already have.\n        // Additionally, more than one attribute/input might need to be present in order for the\n        // directive to match (e.g. `ngFor` has a selector of `[ngFor][ngForOf]`). This gets a\n        // little tricky.\n\n        const structuralAttributes = getStructuralAttributes(meta);\n        for (const attrName of structuralAttributes) {\n          table.set(attrName, {\n            kind: AttributeCompletionKind.StructuralDirectiveAttribute,\n            attribute: attrName,\n            directive: dirInScope,\n          });\n        }\n      }\n    }\n  }\n\n  // Finally, add any DOM attributes not already covered by inputs.\n  if (element instanceof TmplAstElement) {\n    for (const {attribute, property} of checker.getPotentialDomBindings(element.name)) {\n      const isAlsoProperty = attribute === property;\n      if (!table.has(attribute)) {\n        table.set(attribute, {\n          kind: AttributeCompletionKind.DomAttribute,\n          attribute,\n          isAlsoProperty,\n        });\n      }\n      if (!isAlsoProperty && !table.has(property)) {\n        table.set(property, {\n          kind: AttributeCompletionKind.DomProperty,\n          property,\n        });\n      }\n    }\n  }\n\n  return table;\n}\n\n/**\n * Given an `AttributeCompletion`, add any available completions to a `ts.CompletionEntry` array of\n * results.\n *\n * The kind of completions generated depends on whether the current context is an attribute context\n * or not. For example, completing on `<element attr|>` will generate two results: `attribute` and\n * `[attribute]` - either a static attribute can be generated, or a property binding. However,\n * `<element [attr|]>` is not an attribute context, and so only the property completion `attribute`\n * is generated. Note that this completion does not have the `[]` property binding sugar as its\n * implicitly present in a property binding context (we're already completing within an `[attr|]`\n * expression).\n */\nexport function addAttributeCompletionEntries(\n    entries: ts.CompletionEntry[], completion: AttributeCompletion, isAttributeContext: boolean,\n    isElementContext: boolean, replacementSpan: ts.TextSpan|undefined): void {\n  switch (completion.kind) {\n    case AttributeCompletionKind.DirectiveAttribute: {\n      entries.push({\n        kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.DIRECTIVE),\n        name: completion.attribute,\n        sortText: completion.attribute,\n        replacementSpan,\n      });\n      break;\n    }\n    case AttributeCompletionKind.StructuralDirectiveAttribute: {\n      // In an element, the completion is offered with a leading '*' to activate the structural\n      // directive. Once present, the structural attribute will be parsed as a template and not an\n      // element, and the prefix is no longer necessary.\n      const prefix = isElementContext ? '*' : '';\n      entries.push({\n        kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.DIRECTIVE),\n        name: prefix + completion.attribute,\n        sortText: prefix + completion.attribute,\n        replacementSpan,\n      });\n      break;\n    }\n    case AttributeCompletionKind.DirectiveInput: {\n      if (isAttributeContext) {\n        // Offer a completion of a property binding.\n        entries.push({\n          kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.PROPERTY),\n          name: `[${completion.propertyName}]`,\n          sortText: completion.propertyName,\n          replacementSpan,\n        });\n        // If the directive supports banana-in-a-box for this input, offer that as well.\n        if (completion.twoWayBindingSupported) {\n          entries.push({\n            kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.PROPERTY),\n            name: `[(${completion.propertyName})]`,\n            // This completion should sort after the property binding.\n            sortText: completion.propertyName + '_1',\n            replacementSpan,\n          });\n        }\n        // Offer a completion of the input binding as an attribute.\n        entries.push({\n          kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.ATTRIBUTE),\n          name: completion.propertyName,\n          // This completion should sort after both property binding options (one-way and two-way).\n          sortText: completion.propertyName + '_2',\n          replacementSpan,\n        });\n      } else {\n        entries.push({\n          kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.PROPERTY),\n          name: completion.propertyName,\n          sortText: completion.propertyName,\n          replacementSpan,\n        });\n      }\n      break;\n    }\n    case AttributeCompletionKind.DirectiveOutput: {\n      if (isAttributeContext) {\n        entries.push({\n          kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.EVENT),\n          name: `(${completion.eventName})`,\n          sortText: completion.eventName,\n          replacementSpan,\n        });\n      } else {\n        entries.push({\n          kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.EVENT),\n          name: completion.eventName,\n          sortText: completion.eventName,\n          replacementSpan,\n        });\n      }\n      break;\n    }\n    case AttributeCompletionKind.DomAttribute: {\n      if (isAttributeContext) {\n        // Offer a completion of an attribute binding.\n        entries.push({\n          kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.ATTRIBUTE),\n          name: completion.attribute,\n          sortText: completion.attribute,\n          replacementSpan,\n        });\n        if (completion.isAlsoProperty) {\n          // Offer a completion of a property binding to the DOM property.\n          entries.push({\n            kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.PROPERTY),\n            name: `[${completion.attribute}]`,\n            // In the case of DOM attributes, the property binding should sort after the attribute\n            // binding.\n            sortText: completion.attribute + '_1',\n            replacementSpan,\n          });\n        }\n      } else if (completion.isAlsoProperty) {\n        entries.push({\n          kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.PROPERTY),\n          name: completion.attribute,\n          sortText: completion.attribute,\n          replacementSpan,\n        });\n      }\n      break;\n    }\n    case AttributeCompletionKind.DomProperty: {\n      if (!isAttributeContext) {\n        entries.push({\n          kind: unsafeCastDisplayInfoKindToScriptElementKind(DisplayInfoKind.PROPERTY),\n          name: completion.property,\n          sortText: completion.property,\n          replacementSpan,\n        });\n      }\n    }\n  }\n}\n\nexport function getAttributeCompletionSymbol(\n    completion: AttributeCompletion, checker: ts.TypeChecker): ts.Symbol|null {\n  switch (completion.kind) {\n    case AttributeCompletionKind.DomAttribute:\n    case AttributeCompletionKind.DomProperty:\n      return null;\n    case AttributeCompletionKind.DirectiveAttribute:\n    case AttributeCompletionKind.StructuralDirectiveAttribute:\n      return completion.directive.tsSymbol;\n    case AttributeCompletionKind.DirectiveInput:\n    case AttributeCompletionKind.DirectiveOutput:\n      return checker.getDeclaredTypeOfSymbol(completion.directive.tsSymbol)\n                 .getProperty(completion.classPropertyName) ??\n          null;\n  }\n}\n\n/**\n * Iterates over `CssSelector` attributes, which are internally represented in a zipped array style\n * which is not conducive to straightforward iteration.\n */\nfunction* selectorAttributes(selector: CssSelector): Iterable<[string, string]> {\n  for (let i = 0; i < selector.attrs.length; i += 2) {\n    yield [selector.attrs[0], selector.attrs[1]];\n  }\n}\n\nfunction getStructuralAttributes(meta: TypeCheckableDirectiveMeta): string[] {\n  if (meta.selector === null) {\n    return [];\n  }\n\n  const structuralAttributes: string[] = [];\n  const selectors = CssSelector.parse(meta.selector);\n  for (const selector of selectors) {\n    if (selector.element !== null && selector.element !== 'ng-template') {\n      // This particular selector does not apply under structural directive syntax.\n      continue;\n    }\n\n    // Every attribute of this selector must be name-only - no required values.\n    const attributeSelectors = Array.from(selectorAttributes(selector));\n    if (!attributeSelectors.every(([_, attrValue]) => attrValue === '')) {\n      continue;\n    }\n\n    // Get every named selector.\n    const attributes = attributeSelectors.map(([attrName, _]) => attrName);\n\n    // Find the shortest attribute. This is the structural directive \"base\", and all potential\n    // input bindings must begin with the base. E.g. in `*ngFor=\"let a of b\"`, `ngFor` is the\n    // base attribute, and the `of` binding key corresponds to an input of `ngForOf`.\n    const baseAttr = attributes.reduce(\n        (prev, curr) => prev === null || curr.length < prev.length ? curr : prev,\n        null as string | null);\n    if (baseAttr === null) {\n      // No attributes in this selector?\n      continue;\n    }\n\n    // Validate that the attributes are compatible with use as a structural directive.\n    const isValid = (attr: string): boolean => {\n      // The base attribute is valid by default.\n      if (attr === baseAttr) {\n        return true;\n      }\n\n      // Non-base attributes must all be prefixed with the base attribute.\n      if (!attr.startsWith(baseAttr)) {\n        return false;\n      }\n\n      // Non-base attributes must also correspond to directive inputs.\n      if (!meta.inputs.hasBindingPropertyName(attr)) {\n        return false;\n      }\n\n      // This attribute is compatible.\n      return true;\n    };\n\n    if (!attributes.every(isValid)) {\n      continue;\n    }\n\n    // This attribute is valid as a structural attribute for this directive.\n    structuralAttributes.push(baseAttr);\n  }\n\n  return structuralAttributes;\n}\n"]}
\No newline at end of file