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/expression_diagnostics", ["require", "exports", "tslib", "@angular/compiler", "@angular/language-service/src/diagnostic_messages", "@angular/language-service/src/expression_type", "@angular/language-service/src/symbols", "@angular/language-service/src/utils"], factory);
|
15 | }
|
16 | })(function (require, exports) {
|
17 | ;
|
18 | Object.defineProperty(exports, "__esModule", { value: true });
|
19 | exports.getExpressionScope = exports.getTemplateExpressionDiagnostics = void 0;
|
20 | var tslib_1 = require("tslib");
|
21 | var compiler_1 = require("@angular/compiler");
|
22 | var diagnostic_messages_1 = require("@angular/language-service/src/diagnostic_messages");
|
23 | var expression_type_1 = require("@angular/language-service/src/expression_type");
|
24 | var symbols_1 = require("@angular/language-service/src/symbols");
|
25 | var utils_1 = require("@angular/language-service/src/utils");
|
26 | function getTemplateExpressionDiagnostics(info) {
|
27 | var visitor = new ExpressionDiagnosticsVisitor(info, function (path) { return getExpressionScope(info, path); });
|
28 | compiler_1.templateVisitAll(visitor, info.templateAst);
|
29 | return visitor.diagnostics;
|
30 | }
|
31 | exports.getTemplateExpressionDiagnostics = getTemplateExpressionDiagnostics;
|
32 | function getReferences(info) {
|
33 | var result = [];
|
34 | function processReferences(references) {
|
35 | var e_1, _a;
|
36 | var _loop_1 = function (reference) {
|
37 | var type = undefined;
|
38 | if (reference.value) {
|
39 | type = info.query.getTypeSymbol(compiler_1.tokenReference(reference.value));
|
40 | }
|
41 | result.push({
|
42 | name: reference.name,
|
43 | kind: 'reference',
|
44 | type: type || info.query.getBuiltinType(symbols_1.BuiltinType.Any),
|
45 | get definition() {
|
46 | return getDefinitionOf(info, reference);
|
47 | }
|
48 | });
|
49 | };
|
50 | try {
|
51 | for (var references_1 = tslib_1.__values(references), references_1_1 = references_1.next(); !references_1_1.done; references_1_1 = references_1.next()) {
|
52 | var reference = references_1_1.value;
|
53 | _loop_1(reference);
|
54 | }
|
55 | }
|
56 | catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
57 | finally {
|
58 | try {
|
59 | if (references_1_1 && !references_1_1.done && (_a = references_1.return)) _a.call(references_1);
|
60 | }
|
61 | finally { if (e_1) throw e_1.error; }
|
62 | }
|
63 | }
|
64 | var visitor = new /** @class */ (function (_super) {
|
65 | tslib_1.__extends(class_1, _super);
|
66 | function class_1() {
|
67 | return _super !== null && _super.apply(this, arguments) || this;
|
68 | }
|
69 | class_1.prototype.visitEmbeddedTemplate = function (ast, context) {
|
70 | _super.prototype.visitEmbeddedTemplate.call(this, ast, context);
|
71 | processReferences(ast.references);
|
72 | };
|
73 | class_1.prototype.visitElement = function (ast, context) {
|
74 | _super.prototype.visitElement.call(this, ast, context);
|
75 | processReferences(ast.references);
|
76 | };
|
77 | return class_1;
|
78 | }(compiler_1.RecursiveTemplateAstVisitor));
|
79 | compiler_1.templateVisitAll(visitor, info.templateAst);
|
80 | return result;
|
81 | }
|
82 | function getDefinitionOf(info, ast) {
|
83 | if (info.fileName) {
|
84 | var templateOffset = info.offset;
|
85 | return [{
|
86 | fileName: info.fileName,
|
87 | span: {
|
88 | start: ast.sourceSpan.start.offset + templateOffset,
|
89 | end: ast.sourceSpan.end.offset + templateOffset
|
90 | }
|
91 | }];
|
92 | }
|
93 | }
|
94 | /**
|
95 | * Resolve all variable declarations in a template by traversing the specified
|
96 | * `path`.
|
97 | * @param info
|
98 | * @param path template AST path
|
99 | */
|
100 | function getVarDeclarations(info, path) {
|
101 | var e_2, _a;
|
102 | var results = [];
|
103 | for (var current = path.head; current; current = path.childOf(current)) {
|
104 | if (!(current instanceof compiler_1.EmbeddedTemplateAst)) {
|
105 | continue;
|
106 | }
|
107 | var _loop_2 = function (variable) {
|
108 | var symbol = getVariableTypeFromDirectiveContext(variable.value, info.query, current);
|
109 | var kind = info.query.getTypeKind(symbol);
|
110 | if (kind === symbols_1.BuiltinType.Any || kind === symbols_1.BuiltinType.Unbound) {
|
111 | // For special cases such as ngFor and ngIf, the any type is not very useful.
|
112 | // We can do better by resolving the binding value.
|
113 | var symbolsInScope = info.query.mergeSymbolTable([
|
114 | info.members,
|
115 | // Since we are traversing the AST path from head to tail, any variables
|
116 | // that have been declared so far are also in scope.
|
117 | info.query.createSymbolTable(results),
|
118 | ]);
|
119 | symbol = refinedVariableType(variable.value, symbolsInScope, info, current);
|
120 | }
|
121 | results.push({
|
122 | name: variable.name,
|
123 | kind: 'variable',
|
124 | type: symbol,
|
125 | get definition() {
|
126 | return getDefinitionOf(info, variable);
|
127 | },
|
128 | });
|
129 | };
|
130 | try {
|
131 | for (var _b = (e_2 = void 0, tslib_1.__values(current.variables)), _c = _b.next(); !_c.done; _c = _b.next()) {
|
132 | var variable = _c.value;
|
133 | _loop_2(variable);
|
134 | }
|
135 | }
|
136 | catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
137 | finally {
|
138 | try {
|
139 | if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
140 | }
|
141 | finally { if (e_2) throw e_2.error; }
|
142 | }
|
143 | }
|
144 | return results;
|
145 | }
|
146 | /**
|
147 | * Resolve the type for the variable in `templateElement` by finding the structural
|
148 | * directive which has the context member. Returns any when not found.
|
149 | * @param value variable value name
|
150 | * @param query type symbol query
|
151 | * @param templateElement
|
152 | */
|
153 | function getVariableTypeFromDirectiveContext(value, query, templateElement) {
|
154 | var e_3, _a;
|
155 | try {
|
156 | for (var _b = tslib_1.__values(templateElement.directives), _c = _b.next(); !_c.done; _c = _b.next()) {
|
157 | var directive = _c.value.directive;
|
158 | var context = query.getTemplateContext(directive.type.reference);
|
159 | if (context) {
|
160 | var member = context.get(value);
|
161 | if (member && member.type) {
|
162 | return member.type;
|
163 | }
|
164 | }
|
165 | }
|
166 | }
|
167 | catch (e_3_1) { e_3 = { error: e_3_1 }; }
|
168 | finally {
|
169 | try {
|
170 | if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
171 | }
|
172 | finally { if (e_3) throw e_3.error; }
|
173 | }
|
174 | return query.getBuiltinType(symbols_1.BuiltinType.Any);
|
175 | }
|
176 | /**
|
177 | * Resolve a more specific type for the variable in `templateElement` by inspecting
|
178 | * all variables that are in scope in the `mergedTable`. This function is a special
|
179 | * case for `ngFor` and `ngIf`. If resolution fails, return the `any` type.
|
180 | * @param value variable value name
|
181 | * @param mergedTable symbol table for all variables in scope
|
182 | * @param info available template information
|
183 | * @param templateElement
|
184 | */
|
185 | function refinedVariableType(value, mergedTable, info, templateElement) {
|
186 | if (value === '$implicit') {
|
187 | // Special case: ngFor directive
|
188 | var ngForDirective = templateElement.directives.find(function (d) {
|
189 | var name = compiler_1.identifierName(d.directive.type);
|
190 | return name == 'NgFor' || name == 'NgForOf';
|
191 | });
|
192 | if (ngForDirective) {
|
193 | var ngForOfBinding = ngForDirective.inputs.find(function (i) { return i.directiveName == 'ngForOf'; });
|
194 | if (ngForOfBinding) {
|
195 | // Check if there is a known type for the ngFor binding.
|
196 | var bindingType = new expression_type_1.AstType(mergedTable, info.query, {}, info.source).getType(ngForOfBinding.value);
|
197 | if (bindingType) {
|
198 | var result = info.query.getElementType(bindingType);
|
199 | if (result) {
|
200 | return result;
|
201 | }
|
202 | }
|
203 | }
|
204 | }
|
205 | }
|
206 | if (value === 'ngIf' || value === '$implicit') {
|
207 | var ngIfDirective = templateElement.directives.find(function (d) { return compiler_1.identifierName(d.directive.type) === 'NgIf'; });
|
208 | if (ngIfDirective) {
|
209 | // Special case: ngIf directive. The NgIf structural directive owns a template context with
|
210 | // "$implicit" and "ngIf" members. These properties are typed as generics. Until the language
|
211 | // service uses an Ivy and TypecheckBlock backend, we cannot bind these values to a concrete
|
212 | // type without manual inference. To get the concrete type, look up the type of the "ngIf"
|
213 | // import on the NgIf directive bound to the template.
|
214 | //
|
215 | // See @angular/common/ng_if.ts for more information.
|
216 | var ngIfBinding = ngIfDirective.inputs.find(function (i) { return i.directiveName === 'ngIf'; });
|
217 | if (ngIfBinding) {
|
218 | // Check if there is a known type bound to the ngIf input.
|
219 | var bindingType = new expression_type_1.AstType(mergedTable, info.query, {}, info.source).getType(ngIfBinding.value);
|
220 | if (bindingType) {
|
221 | return bindingType;
|
222 | }
|
223 | }
|
224 | }
|
225 | }
|
226 | // We can't do better, return any
|
227 | return info.query.getBuiltinType(symbols_1.BuiltinType.Any);
|
228 | }
|
229 | function getEventDeclaration(info, path) {
|
230 | var event = path.tail;
|
231 | if (!(event instanceof compiler_1.BoundEventAst)) {
|
232 | // No event available in this context.
|
233 | return;
|
234 | }
|
235 | var genericEvent = {
|
236 | name: '$event',
|
237 | kind: 'variable',
|
238 | type: info.query.getBuiltinType(symbols_1.BuiltinType.Any),
|
239 | };
|
240 | var outputSymbol = utils_1.findOutputBinding(event, path, info.query);
|
241 | if (!outputSymbol) {
|
242 | // The `$event` variable doesn't belong to an output, so its type can't be refined.
|
243 | // TODO: type `$event` variables in bindings to DOM events.
|
244 | return genericEvent;
|
245 | }
|
246 | // The raw event type is wrapped in a generic, like EventEmitter<T> or Observable<T>.
|
247 | var ta = outputSymbol.typeArguments();
|
248 | if (!ta || ta.length !== 1)
|
249 | return genericEvent;
|
250 | var eventType = ta[0];
|
251 | return tslib_1.__assign(tslib_1.__assign({}, genericEvent), { type: eventType });
|
252 | }
|
253 | /**
|
254 | * Returns the symbols available in a particular scope of a template.
|
255 | * @param info parsed template information
|
256 | * @param path path of template nodes narrowing to the context the expression scope should be
|
257 | * derived for.
|
258 | */
|
259 | function getExpressionScope(info, path) {
|
260 | var result = info.members;
|
261 | var references = getReferences(info);
|
262 | var variables = getVarDeclarations(info, path);
|
263 | var event = getEventDeclaration(info, path);
|
264 | if (references.length || variables.length || event) {
|
265 | var referenceTable = info.query.createSymbolTable(references);
|
266 | var variableTable = info.query.createSymbolTable(variables);
|
267 | var eventsTable = info.query.createSymbolTable(event ? [event] : []);
|
268 | result = info.query.mergeSymbolTable([result, referenceTable, variableTable, eventsTable]);
|
269 | }
|
270 | return result;
|
271 | }
|
272 | exports.getExpressionScope = getExpressionScope;
|
273 | var ExpressionDiagnosticsVisitor = /** @class */ (function (_super) {
|
274 | tslib_1.__extends(ExpressionDiagnosticsVisitor, _super);
|
275 | function ExpressionDiagnosticsVisitor(info, getExpressionScope) {
|
276 | var _this = _super.call(this) || this;
|
277 | _this.info = info;
|
278 | _this.getExpressionScope = getExpressionScope;
|
279 | _this.diagnostics = [];
|
280 | _this.path = new compiler_1.AstPath([]);
|
281 | return _this;
|
282 | }
|
283 | ExpressionDiagnosticsVisitor.prototype.visitDirective = function (ast, context) {
|
284 | // Override the default child visitor to ignore the host properties of a directive.
|
285 | if (ast.inputs && ast.inputs.length) {
|
286 | compiler_1.templateVisitAll(this, ast.inputs, context);
|
287 | }
|
288 | };
|
289 | ExpressionDiagnosticsVisitor.prototype.visitBoundText = function (ast) {
|
290 | this.push(ast);
|
291 | this.diagnoseExpression(ast.value, ast.sourceSpan.start.offset, false);
|
292 | this.pop();
|
293 | };
|
294 | ExpressionDiagnosticsVisitor.prototype.visitDirectiveProperty = function (ast) {
|
295 | this.push(ast);
|
296 | this.diagnoseExpression(ast.value, this.attributeValueLocation(ast), false);
|
297 | this.pop();
|
298 | };
|
299 | ExpressionDiagnosticsVisitor.prototype.visitElementProperty = function (ast) {
|
300 | this.push(ast);
|
301 | this.diagnoseExpression(ast.value, this.attributeValueLocation(ast), false);
|
302 | this.pop();
|
303 | };
|
304 | ExpressionDiagnosticsVisitor.prototype.visitEvent = function (ast) {
|
305 | this.push(ast);
|
306 | this.diagnoseExpression(ast.handler, this.attributeValueLocation(ast), true);
|
307 | this.pop();
|
308 | };
|
309 | ExpressionDiagnosticsVisitor.prototype.visitVariable = function (ast) {
|
310 | var directive = this.directiveSummary;
|
311 | if (directive && ast.value) {
|
312 | var context = this.info.query.getTemplateContext(directive.type.reference);
|
313 | if (context && !context.has(ast.value)) {
|
314 | var missingMember = ast.value === '$implicit' ? 'an implicit value' : "a member called '" + ast.value + "'";
|
315 | var span = this.absSpan(spanOf(ast.sourceSpan));
|
316 | this.diagnostics.push(diagnostic_messages_1.createDiagnostic(span, diagnostic_messages_1.Diagnostic.template_context_missing_member, directive.type.reference.name, missingMember));
|
317 | }
|
318 | }
|
319 | };
|
320 | ExpressionDiagnosticsVisitor.prototype.visitElement = function (ast, context) {
|
321 | this.push(ast);
|
322 | _super.prototype.visitElement.call(this, ast, context);
|
323 | this.pop();
|
324 | };
|
325 | ExpressionDiagnosticsVisitor.prototype.visitEmbeddedTemplate = function (ast, context) {
|
326 | var previousDirectiveSummary = this.directiveSummary;
|
327 | this.push(ast);
|
328 | // Find directive that references this template
|
329 | this.directiveSummary =
|
330 | ast.directives.map(function (d) { return d.directive; }).find(function (d) { return hasTemplateReference(d.type); });
|
331 | // Process children
|
332 | _super.prototype.visitEmbeddedTemplate.call(this, ast, context);
|
333 | this.pop();
|
334 | this.directiveSummary = previousDirectiveSummary;
|
335 | };
|
336 | ExpressionDiagnosticsVisitor.prototype.attributeValueLocation = function (ast) {
|
337 | var path = utils_1.getPathToNodeAtPosition(this.info.htmlAst, ast.sourceSpan.start.offset);
|
338 | var last = path.tail;
|
339 | if (last instanceof compiler_1.Attribute && last.valueSpan) {
|
340 | return last.valueSpan.start.offset;
|
341 | }
|
342 | return ast.sourceSpan.start.offset;
|
343 | };
|
344 | ExpressionDiagnosticsVisitor.prototype.diagnoseExpression = function (ast, offset, inEvent) {
|
345 | var e_4, _a;
|
346 | var scope = this.getExpressionScope(this.path, inEvent);
|
347 | var analyzer = new expression_type_1.AstType(scope, this.info.query, { inEvent: inEvent }, this.info.source);
|
348 | try {
|
349 | for (var _b = tslib_1.__values(analyzer.getDiagnostics(ast)), _c = _b.next(); !_c.done; _c = _b.next()) {
|
350 | var diagnostic = _c.value;
|
351 | diagnostic.span = this.absSpan(diagnostic.span, offset);
|
352 | this.diagnostics.push(diagnostic);
|
353 | }
|
354 | }
|
355 | catch (e_4_1) { e_4 = { error: e_4_1 }; }
|
356 | finally {
|
357 | try {
|
358 | if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
359 | }
|
360 | finally { if (e_4) throw e_4.error; }
|
361 | }
|
362 | };
|
363 | ExpressionDiagnosticsVisitor.prototype.push = function (ast) {
|
364 | this.path.push(ast);
|
365 | };
|
366 | ExpressionDiagnosticsVisitor.prototype.pop = function () {
|
367 | this.path.pop();
|
368 | };
|
369 | ExpressionDiagnosticsVisitor.prototype.absSpan = function (span, additionalOffset) {
|
370 | if (additionalOffset === void 0) { additionalOffset = 0; }
|
371 | return {
|
372 | start: span.start + this.info.offset + additionalOffset,
|
373 | end: span.end + this.info.offset + additionalOffset,
|
374 | };
|
375 | };
|
376 | return ExpressionDiagnosticsVisitor;
|
377 | }(compiler_1.RecursiveTemplateAstVisitor));
|
378 | function hasTemplateReference(type) {
|
379 | var e_5, _a;
|
380 | if (type.diDeps) {
|
381 | try {
|
382 | for (var _b = tslib_1.__values(type.diDeps), _c = _b.next(); !_c.done; _c = _b.next()) {
|
383 | var diDep = _c.value;
|
384 | if (diDep.token && diDep.token.identifier &&
|
385 | compiler_1.identifierName(diDep.token.identifier) == 'TemplateRef')
|
386 | return true;
|
387 | }
|
388 | }
|
389 | catch (e_5_1) { e_5 = { error: e_5_1 }; }
|
390 | finally {
|
391 | try {
|
392 | if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
393 | }
|
394 | finally { if (e_5) throw e_5.error; }
|
395 | }
|
396 | }
|
397 | return false;
|
398 | }
|
399 | function spanOf(sourceSpan) {
|
400 | return { start: sourceSpan.start.offset, end: sourceSpan.end.offset };
|
401 | }
|
402 | });
|
403 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"expression_diagnostics.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/expression_diagnostics.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 {AST, AstPath, Attribute, BoundDirectivePropertyAst, BoundElementPropertyAst, BoundEventAst, BoundTextAst, CompileDirectiveSummary, CompileTypeMetadata, DirectiveAst, ElementAst, EmbeddedTemplateAst, identifierName, ParseSourceSpan, RecursiveTemplateAstVisitor, ReferenceAst, TemplateAst, TemplateAstPath, templateVisitAll, tokenReference, VariableAst} from '@angular/compiler';\n\nimport {createDiagnostic, Diagnostic} from './diagnostic_messages';\nimport {AstType} from './expression_type';\nimport {BuiltinType, Definition, Span, Symbol, SymbolDeclaration, SymbolQuery, SymbolTable} from './symbols';\nimport * as ng from './types';\nimport {findOutputBinding, getPathToNodeAtPosition} from './utils';\n\nexport function getTemplateExpressionDiagnostics(info: ng.DiagnosticTemplateInfo): ng.Diagnostic[] {\n  const visitor = new ExpressionDiagnosticsVisitor(\n      info, (path: TemplateAstPath) => getExpressionScope(info, path));\n  templateVisitAll(visitor, info.templateAst);\n  return visitor.diagnostics;\n}\n\nfunction getReferences(info: ng.DiagnosticTemplateInfo): SymbolDeclaration[] {\n  const result: SymbolDeclaration[] = [];\n\n  function processReferences(references: ReferenceAst[]) {\n    for (const reference of references) {\n      let type: Symbol|undefined = undefined;\n      if (reference.value) {\n        type = info.query.getTypeSymbol(tokenReference(reference.value));\n      }\n      result.push({\n        name: reference.name,\n        kind: 'reference',\n        type: type || info.query.getBuiltinType(BuiltinType.Any),\n        get definition() {\n          return getDefinitionOf(info, reference);\n        }\n      });\n    }\n  }\n\n  const visitor = new class extends RecursiveTemplateAstVisitor {\n    visitEmbeddedTemplate(ast: EmbeddedTemplateAst, context: any): any {\n      super.visitEmbeddedTemplate(ast, context);\n      processReferences(ast.references);\n    }\n    visitElement(ast: ElementAst, context: any): any {\n      super.visitElement(ast, context);\n      processReferences(ast.references);\n    }\n  };\n\n  templateVisitAll(visitor, info.templateAst);\n\n  return result;\n}\n\nfunction getDefinitionOf(info: ng.DiagnosticTemplateInfo, ast: TemplateAst): Definition|undefined {\n  if (info.fileName) {\n    const templateOffset = info.offset;\n    return [{\n      fileName: info.fileName,\n      span: {\n        start: ast.sourceSpan.start.offset + templateOffset,\n        end: ast.sourceSpan.end.offset + templateOffset\n      }\n    }];\n  }\n}\n\n/**\n * Resolve all variable declarations in a template by traversing the specified\n * `path`.\n * @param info\n * @param path template AST path\n */\nfunction getVarDeclarations(\n    info: ng.DiagnosticTemplateInfo, path: TemplateAstPath): SymbolDeclaration[] {\n  const results: SymbolDeclaration[] = [];\n  for (let current = path.head; current; current = path.childOf(current)) {\n    if (!(current instanceof EmbeddedTemplateAst)) {\n      continue;\n    }\n    for (const variable of current.variables) {\n      let symbol = getVariableTypeFromDirectiveContext(variable.value, info.query, current);\n\n      const kind = info.query.getTypeKind(symbol);\n      if (kind === BuiltinType.Any || kind === BuiltinType.Unbound) {\n        // For special cases such as ngFor and ngIf, the any type is not very useful.\n        // We can do better by resolving the binding value.\n        const symbolsInScope = info.query.mergeSymbolTable([\n          info.members,\n          // Since we are traversing the AST path from head to tail, any variables\n          // that have been declared so far are also in scope.\n          info.query.createSymbolTable(results),\n        ]);\n        symbol = refinedVariableType(variable.value, symbolsInScope, info, current);\n      }\n      results.push({\n        name: variable.name,\n        kind: 'variable',\n        type: symbol,\n        get definition() {\n          return getDefinitionOf(info, variable);\n        },\n      });\n    }\n  }\n  return results;\n}\n\n/**\n * Resolve the type for the variable in `templateElement` by finding the structural\n * directive which has the context member. Returns any when not found.\n * @param value variable value name\n * @param query type symbol query\n * @param templateElement\n */\nfunction getVariableTypeFromDirectiveContext(\n    value: string, query: SymbolQuery, templateElement: EmbeddedTemplateAst): Symbol {\n  for (const {directive} of templateElement.directives) {\n    const context = query.getTemplateContext(directive.type.reference);\n    if (context) {\n      const member = context.get(value);\n      if (member && member.type) {\n        return member.type;\n      }\n    }\n  }\n\n  return query.getBuiltinType(BuiltinType.Any);\n}\n\n/**\n * Resolve a more specific type for the variable in `templateElement` by inspecting\n * all variables that are in scope in the `mergedTable`. This function is a special\n * case for `ngFor` and `ngIf`. If resolution fails, return the `any` type.\n * @param value variable value name\n * @param mergedTable symbol table for all variables in scope\n * @param info available template information\n * @param templateElement\n */\nfunction refinedVariableType(\n    value: string, mergedTable: SymbolTable, info: ng.DiagnosticTemplateInfo,\n    templateElement: EmbeddedTemplateAst): Symbol {\n  if (value === '$implicit') {\n    // Special case: ngFor directive\n    const ngForDirective = templateElement.directives.find(d => {\n      const name = identifierName(d.directive.type);\n      return name == 'NgFor' || name == 'NgForOf';\n    });\n    if (ngForDirective) {\n      const ngForOfBinding = ngForDirective.inputs.find(i => i.directiveName == 'ngForOf');\n      if (ngForOfBinding) {\n        // Check if there is a known type for the ngFor binding.\n        const bindingType =\n            new AstType(mergedTable, info.query, {}, info.source).getType(ngForOfBinding.value);\n        if (bindingType) {\n          const result = info.query.getElementType(bindingType);\n          if (result) {\n            return result;\n          }\n        }\n      }\n    }\n  }\n\n  if (value === 'ngIf' || value === '$implicit') {\n    const ngIfDirective =\n        templateElement.directives.find(d => identifierName(d.directive.type) === 'NgIf');\n    if (ngIfDirective) {\n      // Special case: ngIf directive. The NgIf structural directive owns a template context with\n      // \"$implicit\" and \"ngIf\" members. These properties are typed as generics. Until the language\n      // service uses an Ivy and TypecheckBlock backend, we cannot bind these values to a concrete\n      // type without manual inference. To get the concrete type, look up the type of the \"ngIf\"\n      // import on the NgIf directive bound to the template.\n      //\n      // See @angular/common/ng_if.ts for more information.\n      const ngIfBinding = ngIfDirective.inputs.find(i => i.directiveName === 'ngIf');\n      if (ngIfBinding) {\n        // Check if there is a known type bound to the ngIf input.\n        const bindingType =\n            new AstType(mergedTable, info.query, {}, info.source).getType(ngIfBinding.value);\n        if (bindingType) {\n          return bindingType;\n        }\n      }\n    }\n  }\n\n  // We can't do better, return any\n  return info.query.getBuiltinType(BuiltinType.Any);\n}\n\nfunction getEventDeclaration(\n    info: ng.DiagnosticTemplateInfo, path: TemplateAstPath): SymbolDeclaration|undefined {\n  const event = path.tail;\n  if (!(event instanceof BoundEventAst)) {\n    // No event available in this context.\n    return;\n  }\n\n  const genericEvent: SymbolDeclaration = {\n    name: '$event',\n    kind: 'variable',\n    type: info.query.getBuiltinType(BuiltinType.Any),\n  };\n\n  const outputSymbol = findOutputBinding(event, path, info.query);\n  if (!outputSymbol) {\n    // The `$event` variable doesn't belong to an output, so its type can't be refined.\n    // TODO: type `$event` variables in bindings to DOM events.\n    return genericEvent;\n  }\n\n  // The raw event type is wrapped in a generic, like EventEmitter<T> or Observable<T>.\n  const ta = outputSymbol.typeArguments();\n  if (!ta || ta.length !== 1) return genericEvent;\n  const eventType = ta[0];\n\n  return {...genericEvent, type: eventType};\n}\n\n/**\n * Returns the symbols available in a particular scope of a template.\n * @param info parsed template information\n * @param path path of template nodes narrowing to the context the expression scope should be\n * derived for.\n */\nexport function getExpressionScope(\n    info: ng.DiagnosticTemplateInfo, path: TemplateAstPath): SymbolTable {\n  let result = info.members;\n  const references = getReferences(info);\n  const variables = getVarDeclarations(info, path);\n  const event = getEventDeclaration(info, path);\n  if (references.length || variables.length || event) {\n    const referenceTable = info.query.createSymbolTable(references);\n    const variableTable = info.query.createSymbolTable(variables);\n    const eventsTable = info.query.createSymbolTable(event ? [event] : []);\n    result = info.query.mergeSymbolTable([result, referenceTable, variableTable, eventsTable]);\n  }\n  return result;\n}\n\nclass ExpressionDiagnosticsVisitor extends RecursiveTemplateAstVisitor {\n  private path: TemplateAstPath;\n  private directiveSummary: CompileDirectiveSummary|undefined;\n\n  diagnostics: ng.Diagnostic[] = [];\n\n  constructor(\n      private info: ng.DiagnosticTemplateInfo,\n      private getExpressionScope: (path: TemplateAstPath, includeEvent: boolean) => SymbolTable) {\n    super();\n    this.path = new AstPath<TemplateAst>([]);\n  }\n\n  visitDirective(ast: DirectiveAst, context: any): any {\n    // Override the default child visitor to ignore the host properties of a directive.\n    if (ast.inputs && ast.inputs.length) {\n      templateVisitAll(this, ast.inputs, context);\n    }\n  }\n\n  visitBoundText(ast: BoundTextAst): void {\n    this.push(ast);\n    this.diagnoseExpression(ast.value, ast.sourceSpan.start.offset, false);\n    this.pop();\n  }\n\n  visitDirectiveProperty(ast: BoundDirectivePropertyAst): void {\n    this.push(ast);\n    this.diagnoseExpression(ast.value, this.attributeValueLocation(ast), false);\n    this.pop();\n  }\n\n  visitElementProperty(ast: BoundElementPropertyAst): void {\n    this.push(ast);\n    this.diagnoseExpression(ast.value, this.attributeValueLocation(ast), false);\n    this.pop();\n  }\n\n  visitEvent(ast: BoundEventAst): void {\n    this.push(ast);\n    this.diagnoseExpression(ast.handler, this.attributeValueLocation(ast), true);\n    this.pop();\n  }\n\n  visitVariable(ast: VariableAst): void {\n    const directive = this.directiveSummary;\n    if (directive && ast.value) {\n      const context = this.info.query.getTemplateContext(directive.type.reference)!;\n      if (context && !context.has(ast.value)) {\n        const missingMember =\n            ast.value === '$implicit' ? 'an implicit value' : `a member called '${ast.value}'`;\n\n        const span = this.absSpan(spanOf(ast.sourceSpan));\n        this.diagnostics.push(createDiagnostic(\n            span, Diagnostic.template_context_missing_member, directive.type.reference.name,\n            missingMember));\n      }\n    }\n  }\n\n  visitElement(ast: ElementAst, context: any): void {\n    this.push(ast);\n    super.visitElement(ast, context);\n    this.pop();\n  }\n\n  visitEmbeddedTemplate(ast: EmbeddedTemplateAst, context: any): any {\n    const previousDirectiveSummary = this.directiveSummary;\n\n    this.push(ast);\n\n    // Find directive that references this template\n    this.directiveSummary =\n        ast.directives.map(d => d.directive).find(d => hasTemplateReference(d.type))!;\n\n    // Process children\n    super.visitEmbeddedTemplate(ast, context);\n\n    this.pop();\n\n    this.directiveSummary = previousDirectiveSummary;\n  }\n\n  private attributeValueLocation(ast: TemplateAst) {\n    const path = getPathToNodeAtPosition(this.info.htmlAst, ast.sourceSpan.start.offset);\n    const last = path.tail;\n    if (last instanceof Attribute && last.valueSpan) {\n      return last.valueSpan.start.offset;\n    }\n    return ast.sourceSpan.start.offset;\n  }\n\n  private diagnoseExpression(ast: AST, offset: number, inEvent: boolean) {\n    const scope = this.getExpressionScope(this.path, inEvent);\n    const analyzer = new AstType(scope, this.info.query, {inEvent}, this.info.source);\n    for (const diagnostic of analyzer.getDiagnostics(ast)) {\n      diagnostic.span = this.absSpan(diagnostic.span, offset);\n      this.diagnostics.push(diagnostic);\n    }\n  }\n\n  private push(ast: TemplateAst) {\n    this.path.push(ast);\n  }\n\n  private pop() {\n    this.path.pop();\n  }\n\n  private absSpan(span: Span, additionalOffset: number = 0): Span {\n    return {\n      start: span.start + this.info.offset + additionalOffset,\n      end: span.end + this.info.offset + additionalOffset,\n    };\n  }\n}\n\nfunction hasTemplateReference(type: CompileTypeMetadata): boolean {\n  if (type.diDeps) {\n    for (let diDep of type.diDeps) {\n      if (diDep.token && diDep.token.identifier &&\n          identifierName(diDep.token!.identifier!) == 'TemplateRef')\n        return true;\n    }\n  }\n  return false;\n}\n\nfunction spanOf(sourceSpan: ParseSourceSpan): Span {\n  return {start: sourceSpan.start.offset, end: sourceSpan.end.offset};\n}\n"]} |
\ | No newline at end of file |