1 | (function (factory) {
|
2 | if (typeof module === "object" && typeof module.exports === "object") {
|
3 | var v = factory(require, exports);
|
4 | if (v !== undefined) module.exports = v;
|
5 | }
|
6 | else if (typeof define === "function" && define.amd) {
|
7 | define("@angular/language-service/ivy/utils", ["require", "exports", "tslib", "@angular/compiler", "@angular/compiler-cli/src/ngtsc/file_system", "@angular/compiler-cli/src/ngtsc/metadata", "@angular/compiler/src/expression_parser/ast", "@angular/compiler/src/render3/r3_ast", "typescript", "@angular/language-service/ivy/display_parts", "@angular/language-service/ivy/ts_utils"], factory);
|
8 | }
|
9 | })(function (require, exports) {
|
10 | ;
|
11 | Object.defineProperty(exports, "__esModule", { value: true });
|
12 | exports.getTemplateLocationFromShimLocation = exports.isWithin = exports.isExternalTemplate = exports.isTypeScriptFile = exports.flatMap = exports.isDollarEvent = exports.filterAliasImports = exports.getDirectiveMatchesForAttribute = exports.makeElementSelector = exports.getDirectiveMatchesForElementTag = exports.getTemplateInfoAtPosition = exports.isExpressionNode = exports.isTemplateNode = exports.isWithinKeyValue = exports.isWithinKey = exports.isTemplateNodeWithKeyAndValue = exports.toTextSpan = exports.getTextSpanOfNode = void 0;
|
13 | var tslib_1 = require("tslib");
|
14 | /**
|
15 | * @license
|
16 | * Copyright Google LLC All Rights Reserved.
|
17 | *
|
18 | * Use of this source code is governed by an MIT-style license that can be
|
19 | * found in the LICENSE file at https://angular.io/license
|
20 | */
|
21 | var compiler_1 = require("@angular/compiler");
|
22 | var file_system_1 = require("@angular/compiler-cli/src/ngtsc/file_system");
|
23 | var metadata_1 = require("@angular/compiler-cli/src/ngtsc/metadata");
|
24 | var e = require("@angular/compiler/src/expression_parser/ast"); // e for expression AST
|
25 | var t = require("@angular/compiler/src/render3/r3_ast"); // t for template AST
|
26 | var ts = require("typescript");
|
27 | var display_parts_1 = require("@angular/language-service/ivy/display_parts");
|
28 | var ts_utils_1 = require("@angular/language-service/ivy/ts_utils");
|
29 | function getTextSpanOfNode(node) {
|
30 | if (isTemplateNodeWithKeyAndValue(node)) {
|
31 | return toTextSpan(node.keySpan);
|
32 | }
|
33 | else if (node instanceof e.PropertyWrite || node instanceof e.MethodCall ||
|
34 | node instanceof e.BindingPipe || node instanceof e.PropertyRead) {
|
35 | // The `name` part of a `PropertyWrite`, `MethodCall`, and `BindingPipe` does not
|
36 | // have its own AST so there is no way to retrieve a `Symbol` for just the `name` via a specific
|
37 | // node.
|
38 | return toTextSpan(node.nameSpan);
|
39 | }
|
40 | else {
|
41 | return toTextSpan(node.sourceSpan);
|
42 | }
|
43 | }
|
44 | exports.getTextSpanOfNode = getTextSpanOfNode;
|
45 | function toTextSpan(span) {
|
46 | var start, end;
|
47 | if (span instanceof compiler_1.AbsoluteSourceSpan || span instanceof e.ParseSpan) {
|
48 | start = span.start;
|
49 | end = span.end;
|
50 | }
|
51 | else {
|
52 | start = span.start.offset;
|
53 | end = span.end.offset;
|
54 | }
|
55 | return { start: start, length: end - start };
|
56 | }
|
57 | exports.toTextSpan = toTextSpan;
|
58 | function isTemplateNodeWithKeyAndValue(node) {
|
59 | return isTemplateNode(node) && node.hasOwnProperty('keySpan');
|
60 | }
|
61 | exports.isTemplateNodeWithKeyAndValue = isTemplateNodeWithKeyAndValue;
|
62 | function isWithinKey(position, node) {
|
63 | var keySpan = node.keySpan, valueSpan = node.valueSpan;
|
64 | if (valueSpan === undefined && node instanceof compiler_1.TmplAstBoundEvent) {
|
65 | valueSpan = node.handlerSpan;
|
66 | }
|
67 | var isWithinKeyValue = isWithin(position, keySpan) || !!(valueSpan && isWithin(position, valueSpan));
|
68 | return isWithinKeyValue;
|
69 | }
|
70 | exports.isWithinKey = isWithinKey;
|
71 | function isWithinKeyValue(position, node) {
|
72 | var keySpan = node.keySpan, valueSpan = node.valueSpan;
|
73 | if (valueSpan === undefined && node instanceof compiler_1.TmplAstBoundEvent) {
|
74 | valueSpan = node.handlerSpan;
|
75 | }
|
76 | var isWithinKeyValue = isWithin(position, keySpan) || !!(valueSpan && isWithin(position, valueSpan));
|
77 | return isWithinKeyValue;
|
78 | }
|
79 | exports.isWithinKeyValue = isWithinKeyValue;
|
80 | function isTemplateNode(node) {
|
81 | // Template node implements the Node interface so we cannot use instanceof.
|
82 | return node.sourceSpan instanceof compiler_1.ParseSourceSpan;
|
83 | }
|
84 | exports.isTemplateNode = isTemplateNode;
|
85 | function isExpressionNode(node) {
|
86 | return node instanceof e.AST;
|
87 | }
|
88 | exports.isExpressionNode = isExpressionNode;
|
89 | function getInlineTemplateInfoAtPosition(sf, position, compiler) {
|
90 | var expression = ts_utils_1.findTightestNode(sf, position);
|
91 | if (expression === undefined) {
|
92 | return undefined;
|
93 | }
|
94 | var classDecl = ts_utils_1.getParentClassDeclaration(expression);
|
95 | if (classDecl === undefined) {
|
96 | return undefined;
|
97 | }
|
98 | // Return `undefined` if the position is not on the template expression or the template resource
|
99 | // is not inline.
|
100 | var resources = compiler.getComponentResources(classDecl);
|
101 | if (resources === null || metadata_1.isExternalResource(resources.template) ||
|
102 | expression !== resources.template.expression) {
|
103 | return undefined;
|
104 | }
|
105 | var template = compiler.getTemplateTypeChecker().getTemplate(classDecl);
|
106 | if (template === null) {
|
107 | return undefined;
|
108 | }
|
109 | return { template: template, component: classDecl };
|
110 | }
|
111 | /**
|
112 | * Retrieves the `ts.ClassDeclaration` at a location along with its template nodes.
|
113 | */
|
114 | function getTemplateInfoAtPosition(fileName, position, compiler) {
|
115 | if (isTypeScriptFile(fileName)) {
|
116 | var sf = compiler.getNextProgram().getSourceFile(fileName);
|
117 | if (sf === undefined) {
|
118 | return undefined;
|
119 | }
|
120 | return getInlineTemplateInfoAtPosition(sf, position, compiler);
|
121 | }
|
122 | else {
|
123 | return getFirstComponentForTemplateFile(fileName, compiler);
|
124 | }
|
125 | }
|
126 | exports.getTemplateInfoAtPosition = getTemplateInfoAtPosition;
|
127 | /**
|
128 | * First, attempt to sort component declarations by file name.
|
129 | * If the files are the same, sort by start location of the declaration.
|
130 | */
|
131 | function tsDeclarationSortComparator(a, b) {
|
132 | var aFile = a.getSourceFile().fileName;
|
133 | var bFile = b.getSourceFile().fileName;
|
134 | if (aFile < bFile) {
|
135 | return -1;
|
136 | }
|
137 | else if (aFile > bFile) {
|
138 | return 1;
|
139 | }
|
140 | else {
|
141 | return b.getFullStart() - a.getFullStart();
|
142 | }
|
143 | }
|
144 | function getFirstComponentForTemplateFile(fileName, compiler) {
|
145 | var e_1, _a;
|
146 | var templateTypeChecker = compiler.getTemplateTypeChecker();
|
147 | var components = compiler.getComponentsWithTemplateFile(fileName);
|
148 | var sortedComponents = Array.from(components).sort(tsDeclarationSortComparator);
|
149 | try {
|
150 | for (var sortedComponents_1 = tslib_1.__values(sortedComponents), sortedComponents_1_1 = sortedComponents_1.next(); !sortedComponents_1_1.done; sortedComponents_1_1 = sortedComponents_1.next()) {
|
151 | var component = sortedComponents_1_1.value;
|
152 | if (!ts.isClassDeclaration(component)) {
|
153 | continue;
|
154 | }
|
155 | var template = templateTypeChecker.getTemplate(component);
|
156 | if (template === null) {
|
157 | continue;
|
158 | }
|
159 | return { template: template, component: component };
|
160 | }
|
161 | }
|
162 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
163 | finally {
|
164 | try {
|
165 | if (sortedComponents_1_1 && !sortedComponents_1_1.done && (_a = sortedComponents_1.return)) _a.call(sortedComponents_1);
|
166 | }
|
167 | finally { if (e_1) throw e_1.error; }
|
168 | }
|
169 | return undefined;
|
170 | }
|
171 | /**
|
172 | * Given an attribute node, converts it to string form.
|
173 | */
|
174 | function toAttributeString(attribute) {
|
175 | var _a, _b;
|
176 | if (attribute instanceof t.BoundEvent) {
|
177 | return "[" + attribute.name + "]";
|
178 | }
|
179 | else {
|
180 | return "[" + attribute.name + "=" + ((_b = (_a = attribute.valueSpan) === null || _a === void 0 ? void 0 : _a.toString()) !== null && _b !== void 0 ? _b : '') + "]";
|
181 | }
|
182 | }
|
183 | function getNodeName(node) {
|
184 | return node instanceof t.Template ? node.tagName : node.name;
|
185 | }
|
186 | /**
|
187 | * Given a template or element node, returns all attributes on the node.
|
188 | */
|
189 | function getAttributes(node) {
|
190 | var attributes = tslib_1.__spread(node.attributes, node.inputs, node.outputs);
|
191 | if (node instanceof t.Template) {
|
192 | attributes.push.apply(attributes, tslib_1.__spread(node.templateAttrs));
|
193 | }
|
194 | return attributes;
|
195 | }
|
196 | /**
|
197 | * Given two `Set`s, returns all items in the `left` which do not appear in the `right`.
|
198 | */
|
199 | function difference(left, right) {
|
200 | var e_2, _a;
|
201 | var result = new Set();
|
202 | try {
|
203 | for (var left_1 = tslib_1.__values(left), left_1_1 = left_1.next(); !left_1_1.done; left_1_1 = left_1.next()) {
|
204 | var dir = left_1_1.value;
|
205 | if (!right.has(dir)) {
|
206 | result.add(dir);
|
207 | }
|
208 | }
|
209 | }
|
210 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
211 | finally {
|
212 | try {
|
213 | if (left_1_1 && !left_1_1.done && (_a = left_1.return)) _a.call(left_1);
|
214 | }
|
215 | finally { if (e_2) throw e_2.error; }
|
216 | }
|
217 | return result;
|
218 | }
|
219 | /**
|
220 | * Given an element or template, determines which directives match because the tag is present. For
|
221 | * example, if a directive selector is `div[myAttr]`, this would match div elements but would not if
|
222 | * the selector were just `[myAttr]`. We find which directives are applied because of this tag by
|
223 | * elimination: compare the directive matches with the tag present against the directive matches
|
224 | * without it. The difference would be the directives which match because the tag is present.
|
225 | *
|
226 | * @param element The element or template node that the attribute/tag is part of.
|
227 | * @param directives The list of directives to match against.
|
228 | * @returns The list of directives matching the tag name via the strategy described above.
|
229 | */
|
230 | // TODO(atscott): Add unit tests for this and the one for attributes
|
231 | function getDirectiveMatchesForElementTag(element, directives) {
|
232 | var attributes = getAttributes(element);
|
233 | var allAttrs = attributes.map(toAttributeString);
|
234 | var allDirectiveMatches = getDirectiveMatchesForSelector(directives, getNodeName(element) + allAttrs.join(''));
|
235 | var matchesWithoutElement = getDirectiveMatchesForSelector(directives, allAttrs.join(''));
|
236 | return difference(allDirectiveMatches, matchesWithoutElement);
|
237 | }
|
238 | exports.getDirectiveMatchesForElementTag = getDirectiveMatchesForElementTag;
|
239 | function makeElementSelector(element) {
|
240 | var attributes = getAttributes(element);
|
241 | var allAttrs = attributes.map(toAttributeString);
|
242 | return getNodeName(element) + allAttrs.join('');
|
243 | }
|
244 | exports.makeElementSelector = makeElementSelector;
|
245 | /**
|
246 | * Given an attribute name, determines which directives match because the attribute is present. We
|
247 | * find which directives are applied because of this attribute by elimination: compare the directive
|
248 | * matches with the attribute present against the directive matches without it. The difference would
|
249 | * be the directives which match because the attribute is present.
|
250 | *
|
251 | * @param name The name of the attribute
|
252 | * @param hostNode The node which the attribute appears on
|
253 | * @param directives The list of directives to match against.
|
254 | * @returns The list of directives matching the tag name via the strategy described above.
|
255 | */
|
256 | function getDirectiveMatchesForAttribute(name, hostNode, directives) {
|
257 | var attributes = getAttributes(hostNode);
|
258 | var allAttrs = attributes.map(toAttributeString);
|
259 | var allDirectiveMatches = getDirectiveMatchesForSelector(directives, getNodeName(hostNode) + allAttrs.join(''));
|
260 | var attrsExcludingName = attributes.filter(function (a) { return a.name !== name; }).map(toAttributeString);
|
261 | var matchesWithoutAttr = getDirectiveMatchesForSelector(directives, getNodeName(hostNode) + attrsExcludingName.join(''));
|
262 | return difference(allDirectiveMatches, matchesWithoutAttr);
|
263 | }
|
264 | exports.getDirectiveMatchesForAttribute = getDirectiveMatchesForAttribute;
|
265 | /**
|
266 | * Given a list of directives and a text to use as a selector, returns the directives which match
|
267 | * for the selector.
|
268 | */
|
269 | function getDirectiveMatchesForSelector(directives, selector) {
|
270 | var selectors = compiler_1.CssSelector.parse(selector);
|
271 | if (selectors.length === 0) {
|
272 | return new Set();
|
273 | }
|
274 | return new Set(directives.filter(function (dir) {
|
275 | if (dir.selector === null) {
|
276 | return false;
|
277 | }
|
278 | var matcher = new compiler_1.SelectorMatcher();
|
279 | matcher.addSelectables(compiler_1.CssSelector.parse(dir.selector));
|
280 | return selectors.some(function (selector) { return matcher.match(selector, null); });
|
281 | }));
|
282 | }
|
283 | /**
|
284 | * Returns a new `ts.SymbolDisplayPart` array which has the alias imports from the tcb filtered
|
285 | * out, i.e. `i0.NgForOf`.
|
286 | */
|
287 | function filterAliasImports(displayParts) {
|
288 | var tcbAliasImportRegex = /i\d+/;
|
289 | function isImportAlias(part) {
|
290 | return part.kind === display_parts_1.ALIAS_NAME && tcbAliasImportRegex.test(part.text);
|
291 | }
|
292 | function isDotPunctuation(part) {
|
293 | return part.kind === display_parts_1.SYMBOL_PUNC && part.text === '.';
|
294 | }
|
295 | return displayParts.filter(function (part, i) {
|
296 | var previousPart = displayParts[i - 1];
|
297 | var nextPart = displayParts[i + 1];
|
298 | var aliasNameFollowedByDot = isImportAlias(part) && nextPart !== undefined && isDotPunctuation(nextPart);
|
299 | var dotPrecededByAlias = isDotPunctuation(part) && previousPart !== undefined && isImportAlias(previousPart);
|
300 | return !aliasNameFollowedByDot && !dotPrecededByAlias;
|
301 | });
|
302 | }
|
303 | exports.filterAliasImports = filterAliasImports;
|
304 | function isDollarEvent(n) {
|
305 | return n instanceof e.PropertyRead && n.name === '$event' &&
|
306 | n.receiver instanceof e.ImplicitReceiver && !(n.receiver instanceof e.ThisReceiver);
|
307 | }
|
308 | exports.isDollarEvent = isDollarEvent;
|
309 | /**
|
310 | * Returns a new array formed by applying a given callback function to each element of the array,
|
311 | * and then flattening the result by one level.
|
312 | */
|
313 | function flatMap(items, f) {
|
314 | var e_3, _a;
|
315 | var results = [];
|
316 | try {
|
317 | for (var items_1 = tslib_1.__values(items), items_1_1 = items_1.next(); !items_1_1.done; items_1_1 = items_1.next()) {
|
318 | var x = items_1_1.value;
|
319 | results.push.apply(results, tslib_1.__spread(f(x)));
|
320 | }
|
321 | }
|
322 | catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
323 | finally {
|
324 | try {
|
325 | if (items_1_1 && !items_1_1.done && (_a = items_1.return)) _a.call(items_1);
|
326 | }
|
327 | finally { if (e_3) throw e_3.error; }
|
328 | }
|
329 | return results;
|
330 | }
|
331 | exports.flatMap = flatMap;
|
332 | function isTypeScriptFile(fileName) {
|
333 | return fileName.endsWith('.ts');
|
334 | }
|
335 | exports.isTypeScriptFile = isTypeScriptFile;
|
336 | function isExternalTemplate(fileName) {
|
337 | return !isTypeScriptFile(fileName);
|
338 | }
|
339 | exports.isExternalTemplate = isExternalTemplate;
|
340 | function isWithin(position, span) {
|
341 | var start, end;
|
342 | if (span instanceof compiler_1.ParseSourceSpan) {
|
343 | start = span.start.offset;
|
344 | end = span.end.offset;
|
345 | }
|
346 | else {
|
347 | start = span.start;
|
348 | end = span.end;
|
349 | }
|
350 | // Note both start and end are inclusive because we want to match conditions
|
351 | // like ¦start and end¦ where ¦ is the cursor.
|
352 | return start <= position && position <= end;
|
353 | }
|
354 | exports.isWithin = isWithin;
|
355 | /**
|
356 | * For a given location in a shim file, retrieves the corresponding file url for the template and
|
357 | * the span in the template.
|
358 | */
|
359 | function getTemplateLocationFromShimLocation(templateTypeChecker, shimPath, positionInShimFile) {
|
360 | var mapping = templateTypeChecker.getTemplateMappingAtShimLocation({ shimPath: shimPath, positionInShimFile: positionInShimFile });
|
361 | if (mapping === null) {
|
362 | return null;
|
363 | }
|
364 | var templateSourceMapping = mapping.templateSourceMapping, span = mapping.span;
|
365 | var templateUrl;
|
366 | if (templateSourceMapping.type === 'direct') {
|
367 | templateUrl = file_system_1.absoluteFromSourceFile(templateSourceMapping.node.getSourceFile());
|
368 | }
|
369 | else if (templateSourceMapping.type === 'external') {
|
370 | templateUrl = file_system_1.absoluteFrom(templateSourceMapping.templateUrl);
|
371 | }
|
372 | else {
|
373 | // This includes indirect mappings, which are difficult to map directly to the code
|
374 | // location. Diagnostics similarly return a synthetic template string for this case rather
|
375 | // than a real location.
|
376 | return null;
|
377 | }
|
378 | return { templateUrl: templateUrl, span: span };
|
379 | }
|
380 | exports.getTemplateLocationFromShimLocation = getTemplateLocationFromShimLocation;
|
381 | });
|
382 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../packages/language-service/ivy/utils.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 */\nimport {AbsoluteSourceSpan, CssSelector, ParseSourceSpan, SelectorMatcher, TmplAstBoundEvent} from '@angular/compiler';\nimport {NgCompiler} from '@angular/compiler-cli/src/ngtsc/core';\nimport {absoluteFrom, absoluteFromSourceFile, AbsoluteFsPath} from '@angular/compiler-cli/src/ngtsc/file_system';\nimport {isExternalResource} from '@angular/compiler-cli/src/ngtsc/metadata';\nimport {DeclarationNode} from '@angular/compiler-cli/src/ngtsc/reflection';\nimport {DirectiveSymbol, TemplateTypeChecker} from '@angular/compiler-cli/src/ngtsc/typecheck/api';\nimport * as e from '@angular/compiler/src/expression_parser/ast';  // e for expression AST\nimport * as t from '@angular/compiler/src/render3/r3_ast';         // t for template AST\nimport * as ts from 'typescript';\n\nimport {ALIAS_NAME, SYMBOL_PUNC} from './display_parts';\nimport {findTightestNode, getParentClassDeclaration} from './ts_utils';\n\nexport function getTextSpanOfNode(node: t.Node|e.AST): ts.TextSpan {\n  if (isTemplateNodeWithKeyAndValue(node)) {\n    return toTextSpan(node.keySpan);\n  } else if (\n      node instanceof e.PropertyWrite || node instanceof e.MethodCall ||\n      node instanceof e.BindingPipe || node instanceof e.PropertyRead) {\n    // The `name` part of a `PropertyWrite`, `MethodCall`, and `BindingPipe` does not\n    // have its own AST so there is no way to retrieve a `Symbol` for just the `name` via a specific\n    // node.\n    return toTextSpan(node.nameSpan);\n  } else {\n    return toTextSpan(node.sourceSpan);\n  }\n}\n\nexport function toTextSpan(span: AbsoluteSourceSpan|ParseSourceSpan|e.ParseSpan): ts.TextSpan {\n  let start: number, end: number;\n  if (span instanceof AbsoluteSourceSpan || span instanceof e.ParseSpan) {\n    start = span.start;\n    end = span.end;\n  } else {\n    start = span.start.offset;\n    end = span.end.offset;\n  }\n  return {start, length: end - start};\n}\n\ninterface NodeWithKeyAndValue extends t.Node {\n  keySpan: ParseSourceSpan;\n  valueSpan?: ParseSourceSpan;\n}\n\nexport function isTemplateNodeWithKeyAndValue(node: t.Node|e.AST): node is NodeWithKeyAndValue {\n  return isTemplateNode(node) && node.hasOwnProperty('keySpan');\n}\n\nexport function isWithinKey(position: number, node: NodeWithKeyAndValue): boolean {\n  let {keySpan, valueSpan} = node;\n  if (valueSpan === undefined && node instanceof TmplAstBoundEvent) {\n    valueSpan = node.handlerSpan;\n  }\n  const isWithinKeyValue =\n      isWithin(position, keySpan) || !!(valueSpan && isWithin(position, valueSpan));\n  return isWithinKeyValue;\n}\n\nexport function isWithinKeyValue(position: number, node: NodeWithKeyAndValue): boolean {\n  let {keySpan, valueSpan} = node;\n  if (valueSpan === undefined && node instanceof TmplAstBoundEvent) {\n    valueSpan = node.handlerSpan;\n  }\n  const isWithinKeyValue =\n      isWithin(position, keySpan) || !!(valueSpan && isWithin(position, valueSpan));\n  return isWithinKeyValue;\n}\n\nexport function isTemplateNode(node: t.Node|e.AST): node is t.Node {\n  // Template node implements the Node interface so we cannot use instanceof.\n  return node.sourceSpan instanceof ParseSourceSpan;\n}\n\nexport function isExpressionNode(node: t.Node|e.AST): node is e.AST {\n  return node instanceof e.AST;\n}\n\nexport interface TemplateInfo {\n  template: t.Node[];\n  component: ts.ClassDeclaration;\n}\n\nfunction getInlineTemplateInfoAtPosition(\n    sf: ts.SourceFile, position: number, compiler: NgCompiler): TemplateInfo|undefined {\n  const expression = findTightestNode(sf, position);\n  if (expression === undefined) {\n    return undefined;\n  }\n  const classDecl = getParentClassDeclaration(expression);\n  if (classDecl === undefined) {\n    return undefined;\n  }\n\n  // Return `undefined` if the position is not on the template expression or the template resource\n  // is not inline.\n  const resources = compiler.getComponentResources(classDecl);\n  if (resources === null || isExternalResource(resources.template) ||\n      expression !== resources.template.expression) {\n    return undefined;\n  }\n\n  const template = compiler.getTemplateTypeChecker().getTemplate(classDecl);\n  if (template === null) {\n    return undefined;\n  }\n\n  return {template, component: classDecl};\n}\n\n/**\n * Retrieves the `ts.ClassDeclaration` at a location along with its template nodes.\n */\nexport function getTemplateInfoAtPosition(\n    fileName: string, position: number, compiler: NgCompiler): TemplateInfo|undefined {\n  if (isTypeScriptFile(fileName)) {\n    const sf = compiler.getNextProgram().getSourceFile(fileName);\n    if (sf === undefined) {\n      return undefined;\n    }\n\n    return getInlineTemplateInfoAtPosition(sf, position, compiler);\n  } else {\n    return getFirstComponentForTemplateFile(fileName, compiler);\n  }\n}\n\n/**\n * First, attempt to sort component declarations by file name.\n * If the files are the same, sort by start location of the declaration.\n */\nfunction tsDeclarationSortComparator(a: DeclarationNode, b: DeclarationNode): number {\n  const aFile = a.getSourceFile().fileName;\n  const bFile = b.getSourceFile().fileName;\n  if (aFile < bFile) {\n    return -1;\n  } else if (aFile > bFile) {\n    return 1;\n  } else {\n    return b.getFullStart() - a.getFullStart();\n  }\n}\n\nfunction getFirstComponentForTemplateFile(fileName: string, compiler: NgCompiler): TemplateInfo|\n    undefined {\n  const templateTypeChecker = compiler.getTemplateTypeChecker();\n  const components = compiler.getComponentsWithTemplateFile(fileName);\n  const sortedComponents = Array.from(components).sort(tsDeclarationSortComparator);\n  for (const component of sortedComponents) {\n    if (!ts.isClassDeclaration(component)) {\n      continue;\n    }\n    const template = templateTypeChecker.getTemplate(component);\n    if (template === null) {\n      continue;\n    }\n    return {template, component};\n  }\n\n  return undefined;\n}\n\n/**\n * Given an attribute node, converts it to string form.\n */\nfunction toAttributeString(attribute: t.TextAttribute|t.BoundAttribute|t.BoundEvent): string {\n  if (attribute instanceof t.BoundEvent) {\n    return `[${attribute.name}]`;\n  } else {\n    return `[${attribute.name}=${attribute.valueSpan?.toString() ?? ''}]`;\n  }\n}\n\nfunction getNodeName(node: t.Template|t.Element): string {\n  return node instanceof t.Template ? node.tagName : node.name;\n}\n\n/**\n * Given a template or element node, returns all attributes on the node.\n */\nfunction getAttributes(node: t.Template|\n                       t.Element): Array<t.TextAttribute|t.BoundAttribute|t.BoundEvent> {\n  const attributes: Array<t.TextAttribute|t.BoundAttribute|t.BoundEvent> =\n      [...node.attributes, ...node.inputs, ...node.outputs];\n  if (node instanceof t.Template) {\n    attributes.push(...node.templateAttrs);\n  }\n  return attributes;\n}\n\n/**\n * Given two `Set`s, returns all items in the `left` which do not appear in the `right`.\n */\nfunction difference<T>(left: Set<T>, right: Set<T>): Set<T> {\n  const result = new Set<T>();\n  for (const dir of left) {\n    if (!right.has(dir)) {\n      result.add(dir);\n    }\n  }\n  return result;\n}\n\n/**\n * Given an element or template, determines which directives match because the tag is present. For\n * example, if a directive selector is `div[myAttr]`, this would match div elements but would not if\n * the selector were just `[myAttr]`. We find which directives are applied because of this tag by\n * elimination: compare the directive matches with the tag present against the directive matches\n * without it. The difference would be the directives which match because the tag is present.\n *\n * @param element The element or template node that the attribute/tag is part of.\n * @param directives The list of directives to match against.\n * @returns The list of directives matching the tag name via the strategy described above.\n */\n// TODO(atscott): Add unit tests for this and the one for attributes\nexport function getDirectiveMatchesForElementTag(\n    element: t.Template|t.Element, directives: DirectiveSymbol[]): Set<DirectiveSymbol> {\n  const attributes = getAttributes(element);\n  const allAttrs = attributes.map(toAttributeString);\n  const allDirectiveMatches =\n      getDirectiveMatchesForSelector(directives, getNodeName(element) + allAttrs.join(''));\n  const matchesWithoutElement = getDirectiveMatchesForSelector(directives, allAttrs.join(''));\n  return difference(allDirectiveMatches, matchesWithoutElement);\n}\n\n\nexport function makeElementSelector(element: t.Element|t.Template): string {\n  const attributes = getAttributes(element);\n  const allAttrs = attributes.map(toAttributeString);\n  return getNodeName(element) + allAttrs.join('');\n}\n\n/**\n * Given an attribute name, determines which directives match because the attribute is present. We\n * find which directives are applied because of this attribute by elimination: compare the directive\n * matches with the attribute present against the directive matches without it. The difference would\n * be the directives which match because the attribute is present.\n *\n * @param name The name of the attribute\n * @param hostNode The node which the attribute appears on\n * @param directives The list of directives to match against.\n * @returns The list of directives matching the tag name via the strategy described above.\n */\nexport function getDirectiveMatchesForAttribute(\n    name: string, hostNode: t.Template|t.Element,\n    directives: DirectiveSymbol[]): Set<DirectiveSymbol> {\n  const attributes = getAttributes(hostNode);\n  const allAttrs = attributes.map(toAttributeString);\n  const allDirectiveMatches =\n      getDirectiveMatchesForSelector(directives, getNodeName(hostNode) + allAttrs.join(''));\n  const attrsExcludingName = attributes.filter(a => a.name !== name).map(toAttributeString);\n  const matchesWithoutAttr = getDirectiveMatchesForSelector(\n      directives, getNodeName(hostNode) + attrsExcludingName.join(''));\n  return difference(allDirectiveMatches, matchesWithoutAttr);\n}\n\n/**\n * Given a list of directives and a text to use as a selector, returns the directives which match\n * for the selector.\n */\nfunction getDirectiveMatchesForSelector(\n    directives: DirectiveSymbol[], selector: string): Set<DirectiveSymbol> {\n  const selectors = CssSelector.parse(selector);\n  if (selectors.length === 0) {\n    return new Set();\n  }\n  return new Set(directives.filter((dir: DirectiveSymbol) => {\n    if (dir.selector === null) {\n      return false;\n    }\n\n    const matcher = new SelectorMatcher();\n    matcher.addSelectables(CssSelector.parse(dir.selector));\n\n    return selectors.some(selector => matcher.match(selector, null));\n  }));\n}\n\n/**\n * Returns a new `ts.SymbolDisplayPart` array which has the alias imports from the tcb filtered\n * out, i.e. `i0.NgForOf`.\n */\nexport function filterAliasImports(displayParts: ts.SymbolDisplayPart[]): ts.SymbolDisplayPart[] {\n  const tcbAliasImportRegex = /i\\d+/;\n  function isImportAlias(part: {kind: string, text: string}) {\n    return part.kind === ALIAS_NAME && tcbAliasImportRegex.test(part.text);\n  }\n  function isDotPunctuation(part: {kind: string, text: string}) {\n    return part.kind === SYMBOL_PUNC && part.text === '.';\n  }\n\n  return displayParts.filter((part, i) => {\n    const previousPart = displayParts[i - 1];\n    const nextPart = displayParts[i + 1];\n\n    const aliasNameFollowedByDot =\n        isImportAlias(part) && nextPart !== undefined && isDotPunctuation(nextPart);\n    const dotPrecededByAlias =\n        isDotPunctuation(part) && previousPart !== undefined && isImportAlias(previousPart);\n\n    return !aliasNameFollowedByDot && !dotPrecededByAlias;\n  });\n}\n\nexport function isDollarEvent(n: t.Node|e.AST): n is e.PropertyRead {\n  return n instanceof e.PropertyRead && n.name === '$event' &&\n      n.receiver instanceof e.ImplicitReceiver && !(n.receiver instanceof e.ThisReceiver);\n}\n\n/**\n * Returns a new array formed by applying a given callback function to each element of the array,\n * and then flattening the result by one level.\n */\nexport function flatMap<T, R>(items: T[]|readonly T[], f: (item: T) => R[] | readonly R[]): R[] {\n  const results: R[] = [];\n  for (const x of items) {\n    results.push(...f(x));\n  }\n  return results;\n}\n\nexport function isTypeScriptFile(fileName: string): boolean {\n  return fileName.endsWith('.ts');\n}\n\nexport function isExternalTemplate(fileName: string): boolean {\n  return !isTypeScriptFile(fileName);\n}\n\nexport function isWithin(position: number, span: AbsoluteSourceSpan|ParseSourceSpan): boolean {\n  let start: number, end: number;\n  if (span instanceof ParseSourceSpan) {\n    start = span.start.offset;\n    end = span.end.offset;\n  } else {\n    start = span.start;\n    end = span.end;\n  }\n  // Note both start and end are inclusive because we want to match conditions\n  // like ¦start and end¦ where ¦ is the cursor.\n  return start <= position && position <= end;\n}\n\n/**\n * For a given location in a shim file, retrieves the corresponding file url for the template and\n * the span in the template.\n */\nexport function getTemplateLocationFromShimLocation(\n    templateTypeChecker: TemplateTypeChecker, shimPath: AbsoluteFsPath,\n    positionInShimFile: number): {templateUrl: AbsoluteFsPath, span: ParseSourceSpan}|null {\n  const mapping =\n      templateTypeChecker.getTemplateMappingAtShimLocation({shimPath, positionInShimFile});\n  if (mapping === null) {\n    return null;\n  }\n  const {templateSourceMapping, span} = mapping;\n\n  let templateUrl: AbsoluteFsPath;\n  if (templateSourceMapping.type === 'direct') {\n    templateUrl = absoluteFromSourceFile(templateSourceMapping.node.getSourceFile());\n  } else if (templateSourceMapping.type === 'external') {\n    templateUrl = absoluteFrom(templateSourceMapping.templateUrl);\n  } else {\n    // This includes indirect mappings, which are difficult to map directly to the code\n    // location. Diagnostics similarly return a synthetic template string for this case rather\n    // than a real location.\n    return null;\n  }\n  return {templateUrl, span};\n}"]} |
\ | No newline at end of file |