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/src/utils", ["require", "exports", "tslib", "@angular/compiler", "path"], factory);
|
15 | }
|
16 | })(function (require, exports) {
|
17 | ;
|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
19 | exports.extractAbsoluteFilePath = exports.findOutputBinding = exports.invertMap = exports.getPathToNodeAtPosition = exports.findTemplateAstAt = exports.diagnosticInfoFromTemplateInfo = exports.getSelectors = exports.isStructuralDirective = exports.isNarrower = exports.offsetSpan = exports.inSpan = exports.spanOf = void 0;
|
20 | var tslib_1 = require("tslib");
|
21 | var compiler_1 = require("@angular/compiler");
|
22 | var path = require("path");
|
23 | function isParseSourceSpan(value) {
|
24 | return value && !!value.start;
|
25 | }
|
26 | function spanOf(span) {
|
27 | if (!span)
|
28 | return undefined;
|
29 | if (isParseSourceSpan(span)) {
|
30 | return { start: span.start.offset, end: span.end.offset };
|
31 | }
|
32 | else {
|
33 | if (span.endSourceSpan) {
|
34 | return { start: span.sourceSpan.start.offset, end: span.endSourceSpan.end.offset };
|
35 | }
|
36 | else if (span.children && span.children.length) {
|
37 | return {
|
38 | start: span.sourceSpan.start.offset,
|
39 | end: spanOf(span.children[span.children.length - 1]).end
|
40 | };
|
41 | }
|
42 | return { start: span.sourceSpan.start.offset, end: span.sourceSpan.end.offset };
|
43 | }
|
44 | }
|
45 | exports.spanOf = spanOf;
|
46 | function inSpan(position, span, exclusive) {
|
47 | return span != null &&
|
48 | (exclusive ? position >= span.start && position < span.end :
|
49 | position >= span.start && position <= span.end);
|
50 | }
|
51 | exports.inSpan = inSpan;
|
52 | function offsetSpan(span, amount) {
|
53 | return { start: span.start + amount, end: span.end + amount };
|
54 | }
|
55 | exports.offsetSpan = offsetSpan;
|
56 | function isNarrower(spanA, spanB) {
|
57 | return spanA.start >= spanB.start && spanA.end <= spanB.end;
|
58 | }
|
59 | exports.isNarrower = isNarrower;
|
60 | function isStructuralDirective(type) {
|
61 | var e_1, _a;
|
62 | var _b;
|
63 | try {
|
64 | for (var _c = tslib_1.__values(type.diDeps), _d = _c.next(); !_d.done; _d = _c.next()) {
|
65 | var diDep = _d.value;
|
66 | var diDepName = compiler_1.identifierName((_b = diDep.token) === null || _b === void 0 ? void 0 : _b.identifier);
|
67 | if (diDepName === compiler_1.Identifiers.TemplateRef.name ||
|
68 | diDepName === compiler_1.Identifiers.ViewContainerRef.name) {
|
69 | return true;
|
70 | }
|
71 | }
|
72 | }
|
73 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
74 | finally {
|
75 | try {
|
76 | if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
|
77 | }
|
78 | finally { if (e_1) throw e_1.error; }
|
79 | }
|
80 | return false;
|
81 | }
|
82 | exports.isStructuralDirective = isStructuralDirective;
|
83 | function getSelectors(info) {
|
84 | var e_2, _a, e_3, _b;
|
85 | var map = new Map();
|
86 | var results = [];
|
87 | try {
|
88 | for (var _c = tslib_1.__values(info.directives), _d = _c.next(); !_d.done; _d = _c.next()) {
|
89 | var directive = _d.value;
|
90 | var selectors = compiler_1.CssSelector.parse(directive.selector);
|
91 | try {
|
92 | for (var selectors_1 = (e_3 = void 0, tslib_1.__values(selectors)), selectors_1_1 = selectors_1.next(); !selectors_1_1.done; selectors_1_1 = selectors_1.next()) {
|
93 | var selector = selectors_1_1.value;
|
94 | results.push(selector);
|
95 | map.set(selector, directive);
|
96 | }
|
97 | }
|
98 | catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
99 | finally {
|
100 | try {
|
101 | if (selectors_1_1 && !selectors_1_1.done && (_b = selectors_1.return)) _b.call(selectors_1);
|
102 | }
|
103 | finally { if (e_3) throw e_3.error; }
|
104 | }
|
105 | }
|
106 | }
|
107 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
108 | finally {
|
109 | try {
|
110 | if (_d && !_d.done && (_a = _c.return)) _a.call(_c);
|
111 | }
|
112 | finally { if (e_2) throw e_2.error; }
|
113 | }
|
114 | return { selectors: results, map: map };
|
115 | }
|
116 | exports.getSelectors = getSelectors;
|
117 | function diagnosticInfoFromTemplateInfo(info) {
|
118 | return {
|
119 | fileName: info.template.fileName,
|
120 | offset: info.template.span.start,
|
121 | query: info.template.query,
|
122 | members: info.template.members,
|
123 | htmlAst: info.htmlAst,
|
124 | templateAst: info.templateAst,
|
125 | source: info.template.source,
|
126 | };
|
127 | }
|
128 | exports.diagnosticInfoFromTemplateInfo = diagnosticInfoFromTemplateInfo;
|
129 | function findTemplateAstAt(ast, position) {
|
130 | var path = [];
|
131 | var visitor = new /** @class */ (function (_super) {
|
132 | tslib_1.__extends(class_1, _super);
|
133 | function class_1() {
|
134 | return _super !== null && _super.apply(this, arguments) || this;
|
135 | }
|
136 | class_1.prototype.visit = function (ast) {
|
137 | var span = spanOf(ast);
|
138 | if (inSpan(position, span)) {
|
139 | var len = path.length;
|
140 | if (!len || isNarrower(span, spanOf(path[len - 1]))) {
|
141 | path.push(ast);
|
142 | }
|
143 | }
|
144 | else {
|
145 | // Returning a value here will result in the children being skipped.
|
146 | return true;
|
147 | }
|
148 | };
|
149 | class_1.prototype.visitEmbeddedTemplate = function (ast, context) {
|
150 | return this.visitChildren(context, function (visit) {
|
151 | // Ignore reference, variable and providers
|
152 | visit(ast.attrs);
|
153 | visit(ast.directives);
|
154 | visit(ast.children);
|
155 | });
|
156 | };
|
157 | class_1.prototype.visitElement = function (ast, context) {
|
158 | return this.visitChildren(context, function (visit) {
|
159 | // Ingnore providers
|
160 | visit(ast.attrs);
|
161 | visit(ast.inputs);
|
162 | visit(ast.outputs);
|
163 | visit(ast.references);
|
164 | visit(ast.directives);
|
165 | visit(ast.children);
|
166 | });
|
167 | };
|
168 | class_1.prototype.visitDirective = function (ast, context) {
|
169 | // Ignore the host properties of a directive
|
170 | var result = this.visitChildren(context, function (visit) {
|
171 | visit(ast.inputs);
|
172 | });
|
173 | // We never care about the diretive itself, just its inputs.
|
174 | if (path[path.length - 1] === ast) {
|
175 | path.pop();
|
176 | }
|
177 | return result;
|
178 | };
|
179 | return class_1;
|
180 | }(compiler_1.RecursiveTemplateAstVisitor));
|
181 | compiler_1.templateVisitAll(visitor, ast);
|
182 | return new compiler_1.AstPath(path, position);
|
183 | }
|
184 | exports.findTemplateAstAt = findTemplateAstAt;
|
185 | /**
|
186 | * Find the tightest node at the specified `position` from the AST `nodes`, and
|
187 | * return the path to the node.
|
188 | * @param nodes HTML AST nodes
|
189 | * @param position
|
190 | */
|
191 | function getPathToNodeAtPosition(nodes, position) {
|
192 | var path = [];
|
193 | var visitor = new /** @class */ (function (_super) {
|
194 | tslib_1.__extends(class_2, _super);
|
195 | function class_2() {
|
196 | return _super !== null && _super.apply(this, arguments) || this;
|
197 | }
|
198 | class_2.prototype.visit = function (ast) {
|
199 | var span = spanOf(ast);
|
200 | if (inSpan(position, span)) {
|
201 | path.push(ast);
|
202 | }
|
203 | else {
|
204 | // Returning a truthy value here will skip all children and terminate
|
205 | // the visit.
|
206 | return true;
|
207 | }
|
208 | };
|
209 | return class_2;
|
210 | }(compiler_1.RecursiveVisitor));
|
211 | compiler_1.visitAll(visitor, nodes);
|
212 | return new compiler_1.AstPath(path, position);
|
213 | }
|
214 | exports.getPathToNodeAtPosition = getPathToNodeAtPosition;
|
215 | /**
|
216 | * Inverts an object's key-value pairs.
|
217 | */
|
218 | function invertMap(obj) {
|
219 | var e_4, _a;
|
220 | var result = {};
|
221 | try {
|
222 | for (var _b = tslib_1.__values(Object.keys(obj)), _c = _b.next(); !_c.done; _c = _b.next()) {
|
223 | var name_1 = _c.value;
|
224 | var v = obj[name_1];
|
225 | result[v] = name_1;
|
226 | }
|
227 | }
|
228 | catch (e_4_1) { e_4 = { error: e_4_1 }; }
|
229 | finally {
|
230 | try {
|
231 | if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
232 | }
|
233 | finally { if (e_4) throw e_4.error; }
|
234 | }
|
235 | return result;
|
236 | }
|
237 | exports.invertMap = invertMap;
|
238 | /**
|
239 | * Finds the directive member providing a template output binding, if one exists.
|
240 | * @param info aggregate template AST information
|
241 | * @param path narrowing
|
242 | */
|
243 | function findOutputBinding(binding, path, query) {
|
244 | var e_5, _a;
|
245 | var element = path.first(compiler_1.ElementAst);
|
246 | if (element) {
|
247 | try {
|
248 | for (var _b = tslib_1.__values(element.directives), _c = _b.next(); !_c.done; _c = _b.next()) {
|
249 | var directive = _c.value;
|
250 | var invertedOutputs = invertMap(directive.directive.outputs);
|
251 | var fieldName = invertedOutputs[binding.name];
|
252 | if (fieldName) {
|
253 | var classSymbol = query.getTypeSymbol(directive.directive.type.reference);
|
254 | if (classSymbol) {
|
255 | return classSymbol.members().get(fieldName);
|
256 | }
|
257 | }
|
258 | }
|
259 | }
|
260 | catch (e_5_1) { e_5 = { error: e_5_1 }; }
|
261 | finally {
|
262 | try {
|
263 | if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
264 | }
|
265 | finally { if (e_5) throw e_5.error; }
|
266 | }
|
267 | }
|
268 | }
|
269 | exports.findOutputBinding = findOutputBinding;
|
270 | /**
|
271 | * Returns an absolute path from the text in `node`. If the text is already
|
272 | * an absolute path, return it as is, otherwise join the path with the filename
|
273 | * of the source file.
|
274 | */
|
275 | function extractAbsoluteFilePath(node) {
|
276 | var url = node.text;
|
277 | return path.isAbsolute(url) ? url : path.join(path.dirname(node.getSourceFile().fileName), url);
|
278 | }
|
279 | exports.extractAbsoluteFilePath = extractAbsoluteFilePath;
|
280 | });
|
281 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/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 */\n\nimport {AstPath, BoundEventAst, CompileDirectiveSummary, CompileTypeMetadata, CssSelector, DirectiveAst, ElementAst, EmbeddedTemplateAst, HtmlAstPath, identifierName, Identifiers, Node, ParseSourceSpan, RecursiveTemplateAstVisitor, RecursiveVisitor, TemplateAst, TemplateAstPath, templateVisitAll, visitAll} from '@angular/compiler';\nimport * as path from 'path';\n\nimport {AstResult, DiagnosticTemplateInfo, SelectorInfo, Span, Symbol, SymbolQuery} from './types';\n\ninterface SpanHolder {\n  sourceSpan: ParseSourceSpan;\n  endSourceSpan?: ParseSourceSpan|null;\n  children?: SpanHolder[];\n}\n\nfunction isParseSourceSpan(value: any): value is ParseSourceSpan {\n  return value && !!value.start;\n}\n\nexport function spanOf(span: SpanHolder): Span;\nexport function spanOf(span: ParseSourceSpan): Span;\nexport function spanOf(span: SpanHolder|ParseSourceSpan|undefined): Span|undefined;\nexport function spanOf(span?: SpanHolder|ParseSourceSpan): Span|undefined {\n  if (!span) return undefined;\n  if (isParseSourceSpan(span)) {\n    return {start: span.start.offset, end: span.end.offset};\n  } else {\n    if (span.endSourceSpan) {\n      return {start: span.sourceSpan.start.offset, end: span.endSourceSpan.end.offset};\n    } else if (span.children && span.children.length) {\n      return {\n        start: span.sourceSpan.start.offset,\n        end: spanOf(span.children[span.children.length - 1])!.end\n      };\n    }\n    return {start: span.sourceSpan.start.offset, end: span.sourceSpan.end.offset};\n  }\n}\n\nexport function inSpan(position: number, span?: Span, exclusive?: boolean): boolean {\n  return span != null &&\n      (exclusive ? position >= span.start && position < span.end :\n                   position >= span.start && position <= span.end);\n}\n\nexport function offsetSpan(span: Span, amount: number): Span {\n  return {start: span.start + amount, end: span.end + amount};\n}\n\nexport function isNarrower(spanA: Span, spanB: Span): boolean {\n  return spanA.start >= spanB.start && spanA.end <= spanB.end;\n}\n\nexport function isStructuralDirective(type: CompileTypeMetadata): boolean {\n  for (const diDep of type.diDeps) {\n    const diDepName = identifierName(diDep.token?.identifier);\n    if (diDepName === Identifiers.TemplateRef.name ||\n        diDepName === Identifiers.ViewContainerRef.name) {\n      return true;\n    }\n  }\n  return false;\n}\n\nexport function getSelectors(info: AstResult): SelectorInfo {\n  const map = new Map<CssSelector, CompileDirectiveSummary>();\n  const results: CssSelector[] = [];\n  for (const directive of info.directives) {\n    const selectors: CssSelector[] = CssSelector.parse(directive.selector!);\n    for (const selector of selectors) {\n      results.push(selector);\n      map.set(selector, directive);\n    }\n  }\n  return {selectors: results, map};\n}\n\nexport function diagnosticInfoFromTemplateInfo(info: AstResult): DiagnosticTemplateInfo {\n  return {\n    fileName: info.template.fileName,\n    offset: info.template.span.start,\n    query: info.template.query,\n    members: info.template.members,\n    htmlAst: info.htmlAst,\n    templateAst: info.templateAst,\n    source: info.template.source,\n  };\n}\n\nexport function findTemplateAstAt(ast: TemplateAst[], position: number): TemplateAstPath {\n  const path: TemplateAst[] = [];\n  const visitor = new class extends RecursiveTemplateAstVisitor {\n    visit(ast: TemplateAst): any {\n      let span = spanOf(ast);\n      if (inSpan(position, span)) {\n        const len = path.length;\n        if (!len || isNarrower(span, spanOf(path[len - 1]))) {\n          path.push(ast);\n        }\n      } else {\n        // Returning a value here will result in the children being skipped.\n        return true;\n      }\n    }\n\n    visitEmbeddedTemplate(ast: EmbeddedTemplateAst, context: any): any {\n      return this.visitChildren(context, visit => {\n        // Ignore reference, variable and providers\n        visit(ast.attrs);\n        visit(ast.directives);\n        visit(ast.children);\n      });\n    }\n\n    visitElement(ast: ElementAst, context: any): any {\n      return this.visitChildren(context, visit => {\n        // Ingnore providers\n        visit(ast.attrs);\n        visit(ast.inputs);\n        visit(ast.outputs);\n        visit(ast.references);\n        visit(ast.directives);\n        visit(ast.children);\n      });\n    }\n\n    visitDirective(ast: DirectiveAst, context: any): any {\n      // Ignore the host properties of a directive\n      const result = this.visitChildren(context, visit => {\n        visit(ast.inputs);\n      });\n      // We never care about the diretive itself, just its inputs.\n      if (path[path.length - 1] === ast) {\n        path.pop();\n      }\n      return result;\n    }\n  };\n\n  templateVisitAll(visitor, ast);\n\n  return new AstPath<TemplateAst>(path, position);\n}\n\n/**\n * Find the tightest node at the specified `position` from the AST `nodes`, and\n * return the path to the node.\n * @param nodes HTML AST nodes\n * @param position\n */\nexport function getPathToNodeAtPosition(nodes: Node[], position: number): HtmlAstPath {\n  const path: Node[] = [];\n  const visitor = new class extends RecursiveVisitor {\n    visit(ast: Node) {\n      const span = spanOf(ast);\n      if (inSpan(position, span)) {\n        path.push(ast);\n      } else {\n        // Returning a truthy value here will skip all children and terminate\n        // the visit.\n        return true;\n      }\n    }\n  };\n  visitAll(visitor, nodes);\n  return new AstPath<Node>(path, position);\n}\n\n\n/**\n * Inverts an object's key-value pairs.\n */\nexport function invertMap(obj: {[name: string]: string}): {[name: string]: string} {\n  const result: {[name: string]: string} = {};\n  for (const name of Object.keys(obj)) {\n    const v = obj[name];\n    result[v] = name;\n  }\n  return result;\n}\n\n\n/**\n * Finds the directive member providing a template output binding, if one exists.\n * @param info aggregate template AST information\n * @param path narrowing\n */\nexport function findOutputBinding(\n    binding: BoundEventAst, path: TemplateAstPath, query: SymbolQuery): Symbol|undefined {\n  const element = path.first(ElementAst);\n  if (element) {\n    for (const directive of element.directives) {\n      const invertedOutputs = invertMap(directive.directive.outputs);\n      const fieldName = invertedOutputs[binding.name];\n      if (fieldName) {\n        const classSymbol = query.getTypeSymbol(directive.directive.type.reference);\n        if (classSymbol) {\n          return classSymbol.members().get(fieldName);\n        }\n      }\n    }\n  }\n}\n\n/**\n * Returns an absolute path from the text in `node`. If the text is already\n * an absolute path, return it as is, otherwise join the path with the filename\n * of the source file.\n */\nexport function extractAbsoluteFilePath(node: ts.StringLiteralLike) {\n  const url = node.text;\n  return path.isAbsolute(url) ? url : path.join(path.dirname(node.getSourceFile().fileName), url);\n}\n"]} |
\ | No newline at end of file |