UNPKG

29.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/src/utils", ["require", "exports", "tslib", "@angular/compiler", "path"], factory);
15 }
16})(function (require, exports) {
17 "use strict";
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":"AAAA;;;;;;GAMG;;;;;;;;;;;;;;IAEH,8CAA6U;IAC7U,2BAA6B;IAU7B,SAAS,iBAAiB,CAAC,KAAU;QACnC,OAAO,KAAK,IAAI,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC;IAChC,CAAC;IAKD,SAAgB,MAAM,CAAC,IAAiC;QACtD,IAAI,CAAC,IAAI;YAAE,OAAO,SAAS,CAAC;QAC5B,IAAI,iBAAiB,CAAC,IAAI,CAAC,EAAE;YAC3B,OAAO,EAAC,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,MAAM,EAAC,CAAC;SACzD;aAAM;YACL,IAAI,IAAI,CAAC,aAAa,EAAE;gBACtB,OAAO,EAAC,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,MAAM,EAAE,GAAG,EAAE,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,MAAM,EAAC,CAAC;aAClF;iBAAM,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE;gBAChD,OAAO;oBACL,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,MAAM;oBACnC,GAAG,EAAE,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC,CAAE,CAAC,GAAG;iBAC1D,CAAC;aACH;YACD,OAAO,EAAC,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,MAAM,EAAE,GAAG,EAAE,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,MAAM,EAAC,CAAC;SAC/E;IACH,CAAC;IAfD,wBAeC;IAED,SAAgB,MAAM,CAAC,QAAgB,EAAE,IAAW,EAAE,SAAmB;QACvE,OAAO,IAAI,IAAI,IAAI;YACf,CAAC,SAAS,CAAC,CAAC,CAAC,QAAQ,IAAI,IAAI,CAAC,KAAK,IAAI,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC;gBAC/C,QAAQ,IAAI,IAAI,CAAC,KAAK,IAAI,QAAQ,IAAI,IAAI,CAAC,GAAG,CAAC,CAAC;IACnE,CAAC;IAJD,wBAIC;IAED,SAAgB,UAAU,CAAC,IAAU,EAAE,MAAc;QACnD,OAAO,EAAC,KAAK,EAAE,IAAI,CAAC,KAAK,GAAG,MAAM,EAAE,GAAG,EAAE,IAAI,CAAC,GAAG,GAAG,MAAM,EAAC,CAAC;IAC9D,CAAC;IAFD,gCAEC;IAED,SAAgB,UAAU,CAAC,KAAW,EAAE,KAAW;QACjD,OAAO,KAAK,CAAC,KAAK,IAAI,KAAK,CAAC,KAAK,IAAI,KAAK,CAAC,GAAG,IAAI,KAAK,CAAC,GAAG,CAAC;IAC9D,CAAC;IAFD,gCAEC;IAED,SAAgB,qBAAqB,CAAC,IAAyB;;;;YAC7D,KAAoB,IAAA,KAAA,iBAAA,IAAI,CAAC,MAAM,CAAA,gBAAA,4BAAE;gBAA5B,IAAM,KAAK,WAAA;gBACd,IAAM,SAAS,GAAG,yBAAc,OAAC,KAAK,CAAC,KAAK,0CAAE,UAAU,CAAC,CAAC;gBAC1D,IAAI,SAAS,KAAK,sBAAW,CAAC,WAAW,CAAC,IAAI;oBAC1C,SAAS,KAAK,sBAAW,CAAC,gBAAgB,CAAC,IAAI,EAAE;oBACnD,OAAO,IAAI,CAAC;iBACb;aACF;;;;;;;;;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IATD,sDASC;IAED,SAAgB,YAAY,CAAC,IAAe;;QAC1C,IAAM,GAAG,GAAG,IAAI,GAAG,EAAwC,CAAC;QAC5D,IAAM,OAAO,GAAkB,EAAE,CAAC;;YAClC,KAAwB,IAAA,KAAA,iBAAA,IAAI,CAAC,UAAU,CAAA,gBAAA,4BAAE;gBAApC,IAAM,SAAS,WAAA;gBAClB,IAAM,SAAS,GAAkB,sBAAW,CAAC,KAAK,CAAC,SAAS,CAAC,QAAS,CAAC,CAAC;;oBACxE,KAAuB,IAAA,6BAAA,iBAAA,SAAS,CAAA,CAAA,oCAAA,2DAAE;wBAA7B,IAAM,QAAQ,sBAAA;wBACjB,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;wBACvB,GAAG,CAAC,GAAG,CAAC,QAAQ,EAAE,SAAS,CAAC,CAAC;qBAC9B;;;;;;;;;aACF;;;;;;;;;QACD,OAAO,EAAC,SAAS,EAAE,OAAO,EAAE,GAAG,KAAA,EAAC,CAAC;IACnC,CAAC;IAXD,oCAWC;IAED,SAAgB,8BAA8B,CAAC,IAAe;QAC5D,OAAO;YACL,QAAQ,EAAE,IAAI,CAAC,QAAQ,CAAC,QAAQ;YAChC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,KAAK;YAChC,KAAK,EAAE,IAAI,CAAC,QAAQ,CAAC,KAAK;YAC1B,OAAO,EAAE,IAAI,CAAC,QAAQ,CAAC,OAAO;YAC9B,OAAO,EAAE,IAAI,CAAC,OAAO;YACrB,WAAW,EAAE,IAAI,CAAC,WAAW;YAC7B,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,MAAM;SAC7B,CAAC;IACJ,CAAC;IAVD,wEAUC;IAED,SAAgB,iBAAiB,CAAC,GAAkB,EAAE,QAAgB;QACpE,IAAM,IAAI,GAAkB,EAAE,CAAC;QAC/B,IAAM,OAAO,GAAG;YAAkB,mCAA2B;YAAzC;;YA8CpB,CAAC;YA7CC,uBAAK,GAAL,UAAM,GAAgB;gBACpB,IAAI,IAAI,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;gBACvB,IAAI,MAAM,CAAC,QAAQ,EAAE,IAAI,CAAC,EAAE;oBAC1B,IAAM,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC;oBACxB,IAAI,CAAC,GAAG,IAAI,UAAU,CAAC,IAAI,EAAE,MAAM,CAAC,IAAI,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE;wBACnD,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;qBAChB;iBACF;qBAAM;oBACL,oEAAoE;oBACpE,OAAO,IAAI,CAAC;iBACb;YACH,CAAC;YAED,uCAAqB,GAArB,UAAsB,GAAwB,EAAE,OAAY;gBAC1D,OAAO,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,UAAA,KAAK;oBACtC,2CAA2C;oBAC3C,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;oBACjB,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;oBACtB,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;gBACtB,CAAC,CAAC,CAAC;YACL,CAAC;YAED,8BAAY,GAAZ,UAAa,GAAe,EAAE,OAAY;gBACxC,OAAO,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,UAAA,KAAK;oBACtC,oBAAoB;oBACpB,KAAK,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;oBACjB,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;oBAClB,KAAK,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;oBACnB,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;oBACtB,KAAK,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;oBACtB,KAAK,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;gBACtB,CAAC,CAAC,CAAC;YACL,CAAC;YAED,gCAAc,GAAd,UAAe,GAAiB,EAAE,OAAY;gBAC5C,4CAA4C;gBAC5C,IAAM,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,UAAA,KAAK;oBAC9C,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;gBACpB,CAAC,CAAC,CAAC;gBACH,4DAA4D;gBAC5D,IAAI,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,KAAK,GAAG,EAAE;oBACjC,IAAI,CAAC,GAAG,EAAE,CAAC;iBACZ;gBACD,OAAO,MAAM,CAAC;YAChB,CAAC;YACH,cAAC;QAAD,CAAC,AA9CmB,CAAc,sCAA2B,EA8C5D,CAAC;QAEF,2BAAgB,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC;QAE/B,OAAO,IAAI,kBAAO,CAAc,IAAI,EAAE,QAAQ,CAAC,CAAC;IAClD,CAAC;IArDD,8CAqDC;IAED;;;;;OAKG;IACH,SAAgB,uBAAuB,CAAC,KAAa,EAAE,QAAgB;QACrE,IAAM,IAAI,GAAW,EAAE,CAAC;QACxB,IAAM,OAAO,GAAG;YAAkB,mCAAgB;YAA9B;;YAWpB,CAAC;YAVC,uBAAK,GAAL,UAAM,GAAS;gBACb,IAAM,IAAI,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC;gBACzB,IAAI,MAAM,CAAC,QAAQ,EAAE,IAAI,CAAC,EAAE;oBAC1B,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;iBAChB;qBAAM;oBACL,qEAAqE;oBACrE,aAAa;oBACb,OAAO,IAAI,CAAC;iBACb;YACH,CAAC;YACH,cAAC;QAAD,CAAC,AAXmB,CAAc,2BAAgB,EAWjD,CAAC;QACF,mBAAQ,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QACzB,OAAO,IAAI,kBAAO,CAAO,IAAI,EAAE,QAAQ,CAAC,CAAC;IAC3C,CAAC;IAhBD,0DAgBC;IAGD;;OAEG;IACH,SAAgB,SAAS,CAAC,GAA6B;;QACrD,IAAM,MAAM,GAA6B,EAAE,CAAC;;YAC5C,KAAmB,IAAA,KAAA,iBAAA,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAA,gBAAA,4BAAE;gBAAhC,IAAM,MAAI,WAAA;gBACb,IAAM,CAAC,GAAG,GAAG,CAAC,MAAI,CAAC,CAAC;gBACpB,MAAM,CAAC,CAAC,CAAC,GAAG,MAAI,CAAC;aAClB;;;;;;;;;QACD,OAAO,MAAM,CAAC;IAChB,CAAC;IAPD,8BAOC;IAGD;;;;OAIG;IACH,SAAgB,iBAAiB,CAC7B,OAAsB,EAAE,IAAqB,EAAE,KAAkB;;QACnE,IAAM,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,qBAAU,CAAC,CAAC;QACvC,IAAI,OAAO,EAAE;;gBACX,KAAwB,IAAA,KAAA,iBAAA,OAAO,CAAC,UAAU,CAAA,gBAAA,4BAAE;oBAAvC,IAAM,SAAS,WAAA;oBAClB,IAAM,eAAe,GAAG,SAAS,CAAC,SAAS,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;oBAC/D,IAAM,SAAS,GAAG,eAAe,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;oBAChD,IAAI,SAAS,EAAE;wBACb,IAAM,WAAW,GAAG,KAAK,CAAC,aAAa,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;wBAC5E,IAAI,WAAW,EAAE;4BACf,OAAO,WAAW,CAAC,OAAO,EAAE,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;yBAC7C;qBACF;iBACF;;;;;;;;;SACF;IACH,CAAC;IAfD,8CAeC;IAED;;;;OAIG;IACH,SAAgB,uBAAuB,CAAC,IAA0B;QAChE,IAAM,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC;QACtB,OAAO,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC,QAAQ,CAAC,EAAE,GAAG,CAAC,CAAC;IAClG,CAAC;IAHD,0DAGC","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