UNPKG

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