UNPKG

51.9 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/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 "use strict";
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