@angular/language-service 9.0.3 → 9.0.7

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -11,7 +11,7 @@
11
11
  if (v !== undefined) module.exports = v;
12
12
  }
13
13
  else if (typeof define === "function" && define.amd) {
14
- define("@angular/language-service/src/diagnostics", ["require", "exports", "tslib", "path", "typescript", "@angular/language-service/src/expression_diagnostics", "@angular/language-service/src/utils"], factory);
14
+ define("@angular/language-service/src/diagnostics", ["require", "exports", "tslib", "path", "typescript", "@angular/language-service/src/diagnostic_messages", "@angular/language-service/src/expression_diagnostics", "@angular/language-service/src/utils"], factory);
15
15
  }
16
16
  })(function (require, exports) {
17
17
  "use strict";
@@ -19,6 +19,7 @@
19
19
  var tslib_1 = require("tslib");
20
20
  var path = require("path");
21
21
  var ts = require("typescript");
22
+ var diagnostic_messages_1 = require("@angular/language-service/src/diagnostic_messages");
22
23
  var expression_diagnostics_1 = require("@angular/language-service/src/expression_diagnostics");
23
24
  var utils_1 = require("@angular/language-service/src/utils");
24
25
  /**
@@ -45,17 +46,6 @@
45
46
  });
46
47
  }
47
48
  exports.getTemplateDiagnostics = getTemplateDiagnostics;
48
- /**
49
- * Generate an error message that indicates a directive is not part of any
50
- * NgModule.
51
- * @param name class name
52
- * @param isComponent true if directive is an Angular Component
53
- */
54
- function missingDirective(name, isComponent) {
55
- var type = isComponent ? 'Component' : 'Directive';
56
- return type + " '" + name + "' is not included in a module and will not be " +
57
- 'available inside a template. Consider adding it to a NgModule declaration.';
58
- }
59
49
  /**
60
50
  * Performs a variety diagnostics on directive declarations.
61
51
  *
@@ -126,29 +116,17 @@
126
116
  }
127
117
  finally { if (e_4) throw e_4.error; }
128
118
  }
119
+ if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {
120
+ results.push(diagnostic_messages_1.createDiagnostic(declarationSpan, diagnostic_messages_1.Diagnostic.directive_not_in_module, metadata.isComponent ? 'Component' : 'Directive', type.name));
121
+ }
129
122
  if (metadata.isComponent) {
130
- if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {
131
- results.push({
132
- kind: ts.DiagnosticCategory.Suggestion,
133
- message: missingDirective(type.name, metadata.isComponent),
134
- span: declarationSpan,
135
- });
136
- }
137
123
  var _j = metadata.template, template = _j.template, templateUrl = _j.templateUrl, styleUrls = _j.styleUrls;
138
124
  if (template === null && !templateUrl) {
139
- results.push({
140
- kind: ts.DiagnosticCategory.Error,
141
- message: "Component '" + type.name + "' must have a template or templateUrl",
142
- span: declarationSpan,
143
- });
125
+ results.push(diagnostic_messages_1.createDiagnostic(declarationSpan, diagnostic_messages_1.Diagnostic.missing_template_and_templateurl, type.name));
144
126
  }
145
127
  else if (templateUrl) {
146
128
  if (template) {
147
- results.push({
148
- kind: ts.DiagnosticCategory.Error,
149
- message: "Component '" + type.name + "' must not have both template and templateUrl",
150
- span: declarationSpan,
151
- });
129
+ results.push(diagnostic_messages_1.createDiagnostic(declarationSpan, diagnostic_messages_1.Diagnostic.both_template_and_templateurl, type.name));
152
130
  }
153
131
  // Find templateUrl value from the directive call expression, which is the parent of the
154
132
  // directive identifier.
@@ -173,13 +151,6 @@
173
151
  results.push.apply(results, tslib_1.__spread(validateUrls(styleUrlsNode.elements, host.tsLsHost)));
174
152
  }
175
153
  }
176
- else if (!directives.has(declaration.type)) {
177
- results.push({
178
- kind: ts.DiagnosticCategory.Suggestion,
179
- message: missingDirective(type.name, metadata.isComponent),
180
- span: declarationSpan,
181
- });
182
- }
183
154
  }
184
155
  }
185
156
  catch (e_3_1) { e_3 = { error: e_3_1 }; }
@@ -219,12 +190,9 @@
219
190
  var url = path.join(path.dirname(curPath), urlNode.text);
220
191
  if (tsLsHost.fileExists(url))
221
192
  continue;
222
- allErrors.push({
223
- kind: ts.DiagnosticCategory.Error,
224
- message: "URL does not point to a valid file",
225
- // Exclude opening and closing quotes in the url span.
226
- span: { start: urlNode.getStart() + 1, end: urlNode.end - 1 },
227
- });
193
+ // Exclude opening and closing quotes in the url span.
194
+ var urlSpan = { start: urlNode.getStart() + 1, end: urlNode.end - 1 };
195
+ allErrors.push(diagnostic_messages_1.createDiagnostic(urlSpan, diagnostic_messages_1.Diagnostic.invalid_templateurl));
228
196
  }
229
197
  return allErrors;
230
198
  }
@@ -258,4 +226,4 @@
258
226
  }
259
227
  exports.ngDiagnosticToTsDiagnostic = ngDiagnosticToTsDiagnostic;
260
228
  });
261
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diagnostics.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/diagnostics.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. 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 {NgAnalyzedModules} from '@angular/compiler';\nimport * as path from 'path';\nimport * as ts from 'typescript';\n\nimport {AstResult} from './common';\nimport {getTemplateExpressionDiagnostics} from './expression_diagnostics';\nimport * as ng from './types';\nimport {TypeScriptServiceHost} from './typescript_host';\nimport {findPropertyValueOfType, findTightestNode, offsetSpan, spanOf} from './utils';\n\n\n/**\n * Return diagnostic information for the parsed AST of the template.\n * @param ast contains HTML and template AST\n */\nexport function getTemplateDiagnostics(ast: AstResult): ng.Diagnostic[] {\n  const {parseErrors, templateAst, htmlAst, template} = ast;\n  if (parseErrors && parseErrors.length) {\n    return parseErrors.map(e => {\n      return {\n        kind: ts.DiagnosticCategory.Error,\n        span: offsetSpan(spanOf(e.span), template.span.start),\n        message: e.msg,\n      };\n    });\n  }\n  return getTemplateExpressionDiagnostics({\n    templateAst: templateAst,\n    htmlAst: htmlAst,\n    offset: template.span.start,\n    query: template.query,\n    members: template.members,\n  });\n}\n\n/**\n * Generate an error message that indicates a directive is not part of any\n * NgModule.\n * @param name class name\n * @param isComponent true if directive is an Angular Component\n */\nfunction missingDirective(name: string, isComponent: boolean) {\n  const type = isComponent ? 'Component' : 'Directive';\n  return `${type} '${name}' is not included in a module and will not be ` +\n      'available inside a template. Consider adding it to a NgModule declaration.';\n}\n\n/**\n * Performs a variety diagnostics on directive declarations.\n *\n * @param declarations Angular directive declarations\n * @param modules NgModules in the project\n * @param host TypeScript service host used to perform TypeScript queries\n * @return diagnosed errors, if any\n */\nexport function getDeclarationDiagnostics(\n    declarations: ng.Declaration[], modules: NgAnalyzedModules,\n    host: Readonly<TypeScriptServiceHost>): ng.Diagnostic[] {\n  const directives = new Set<ng.StaticSymbol>();\n  for (const ngModule of modules.ngModules) {\n    for (const directive of ngModule.declaredDirectives) {\n      directives.add(directive.reference);\n    }\n  }\n\n  const results: ng.Diagnostic[] = [];\n\n  for (const declaration of declarations) {\n    const {errors, metadata, type, declarationSpan} = declaration;\n\n    const sf = host.getSourceFile(type.filePath);\n    if (!sf) {\n      host.error(`directive ${type.name} exists but has no source file`);\n      return [];\n    }\n    // TypeScript identifier of the directive declaration annotation (e.g. \"Component\" or\n    // \"Directive\") on a directive class.\n    const directiveIdentifier = findTightestNode(sf, declarationSpan.start);\n    if (!directiveIdentifier) {\n      host.error(`directive ${type.name} exists but has no identifier`);\n      return [];\n    }\n\n    for (const error of errors) {\n      results.push({\n        kind: ts.DiagnosticCategory.Error,\n        message: error.message,\n        span: error.span,\n      });\n    }\n    if (metadata.isComponent) {\n      if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {\n        results.push({\n          kind: ts.DiagnosticCategory.Suggestion,\n          message: missingDirective(type.name, metadata.isComponent),\n          span: declarationSpan,\n        });\n      }\n      const {template, templateUrl, styleUrls} = metadata.template !;\n      if (template === null && !templateUrl) {\n        results.push({\n          kind: ts.DiagnosticCategory.Error,\n          message: `Component '${type.name}' must have a template or templateUrl`,\n          span: declarationSpan,\n        });\n      } else if (templateUrl) {\n        if (template) {\n          results.push({\n            kind: ts.DiagnosticCategory.Error,\n            message: `Component '${type.name}' must not have both template and templateUrl`,\n            span: declarationSpan,\n          });\n        }\n\n        // Find templateUrl value from the directive call expression, which is the parent of the\n        // directive identifier.\n        //\n        // TODO: We should create an enum of the various properties a directive can have to use\n        // instead of string literals. We can then perform a mass migration of all literal usages.\n        const templateUrlNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'templateUrl', ts.isLiteralExpression);\n        if (!templateUrlNode) {\n          host.error(`templateUrl ${templateUrl} exists but its TypeScript node doesn't`);\n          return [];\n        }\n\n        results.push(...validateUrls([templateUrlNode], host.tsLsHost));\n      }\n\n      if (styleUrls.length > 0) {\n        // Find styleUrls value from the directive call expression, which is the parent of the\n        // directive identifier.\n        const styleUrlsNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'styleUrls', ts.isArrayLiteralExpression);\n        if (!styleUrlsNode) {\n          host.error(`styleUrls property exists but its TypeScript node doesn't'`);\n          return [];\n        }\n\n        results.push(...validateUrls(styleUrlsNode.elements, host.tsLsHost));\n      }\n    } else if (!directives.has(declaration.type)) {\n      results.push({\n        kind: ts.DiagnosticCategory.Suggestion,\n        message: missingDirective(type.name, metadata.isComponent),\n        span: declarationSpan,\n      });\n    }\n  }\n\n  return results;\n}\n\n/**\n * Checks that URLs on a directive point to a valid file.\n * Note that this diagnostic check may require a filesystem hit, and thus may be slower than other\n * checks.\n *\n * @param urls urls to check for validity\n * @param tsLsHost TS LS host used for querying filesystem information\n * @return diagnosed url errors, if any\n */\nfunction validateUrls(\n    urls: ArrayLike<ts.Expression>, tsLsHost: Readonly<ts.LanguageServiceHost>): ng.Diagnostic[] {\n  if (!tsLsHost.fileExists) {\n    return [];\n  }\n\n  const allErrors: ng.Diagnostic[] = [];\n  // TODO(ayazhafiz): most of this logic can be unified with the logic in\n  // definitions.ts#getUrlFromProperty. Create a utility function to be used by both.\n  for (let i = 0; i < urls.length; ++i) {\n    const urlNode = urls[i];\n    if (!ts.isStringLiteralLike(urlNode)) {\n      // If a non-string value is assigned to a URL node (like `templateUrl`), a type error will be\n      // picked up by the TS Language Server.\n      continue;\n    }\n    const curPath = urlNode.getSourceFile().fileName;\n    const url = path.join(path.dirname(curPath), urlNode.text);\n    if (tsLsHost.fileExists(url)) continue;\n\n    allErrors.push({\n      kind: ts.DiagnosticCategory.Error,\n      message: `URL does not point to a valid file`,\n      // Exclude opening and closing quotes in the url span.\n      span: {start: urlNode.getStart() + 1, end: urlNode.end - 1},\n    });\n  }\n  return allErrors;\n}\n\n/**\n * Return a recursive data structure that chains diagnostic messages.\n * @param chain\n */\nfunction chainDiagnostics(chain: ng.DiagnosticMessageChain): ts.DiagnosticMessageChain {\n  return {\n    messageText: chain.message,\n    category: ts.DiagnosticCategory.Error,\n    code: 0,\n    next: chain.next ? chain.next.map(chainDiagnostics) : undefined\n  };\n}\n\n/**\n * Convert ng.Diagnostic to ts.Diagnostic.\n * @param d diagnostic\n * @param file\n */\nexport function ngDiagnosticToTsDiagnostic(\n    d: ng.Diagnostic, file: ts.SourceFile | undefined): ts.Diagnostic {\n  return {\n    file,\n    start: d.span.start,\n    length: d.span.end - d.span.start,\n    messageText: typeof d.message === 'string' ? d.message : chainDiagnostics(d.message),\n    category: d.kind,\n    code: 0,\n    source: 'ng',\n  };\n}\n"]}
229
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"diagnostics.js","sourceRoot":"","sources":["../../../../../../packages/language-service/src/diagnostics.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. 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 {NgAnalyzedModules} from '@angular/compiler';\nimport * as path from 'path';\nimport * as ts from 'typescript';\n\nimport {AstResult} from './common';\nimport {Diagnostic, createDiagnostic} from './diagnostic_messages';\nimport {getTemplateExpressionDiagnostics} from './expression_diagnostics';\nimport * as ng from './types';\nimport {TypeScriptServiceHost} from './typescript_host';\nimport {findPropertyValueOfType, findTightestNode, offsetSpan, spanOf} from './utils';\n\n\n\n/**\n * Return diagnostic information for the parsed AST of the template.\n * @param ast contains HTML and template AST\n */\nexport function getTemplateDiagnostics(ast: AstResult): ng.Diagnostic[] {\n  const {parseErrors, templateAst, htmlAst, template} = ast;\n  if (parseErrors && parseErrors.length) {\n    return parseErrors.map(e => {\n      return {\n        kind: ts.DiagnosticCategory.Error,\n        span: offsetSpan(spanOf(e.span), template.span.start),\n        message: e.msg,\n      };\n    });\n  }\n  return getTemplateExpressionDiagnostics({\n    templateAst: templateAst,\n    htmlAst: htmlAst,\n    offset: template.span.start,\n    query: template.query,\n    members: template.members,\n  });\n}\n\n/**\n * Performs a variety diagnostics on directive declarations.\n *\n * @param declarations Angular directive declarations\n * @param modules NgModules in the project\n * @param host TypeScript service host used to perform TypeScript queries\n * @return diagnosed errors, if any\n */\nexport function getDeclarationDiagnostics(\n    declarations: ng.Declaration[], modules: NgAnalyzedModules,\n    host: Readonly<TypeScriptServiceHost>): ng.Diagnostic[] {\n  const directives = new Set<ng.StaticSymbol>();\n  for (const ngModule of modules.ngModules) {\n    for (const directive of ngModule.declaredDirectives) {\n      directives.add(directive.reference);\n    }\n  }\n\n  const results: ng.Diagnostic[] = [];\n\n  for (const declaration of declarations) {\n    const {errors, metadata, type, declarationSpan} = declaration;\n\n    const sf = host.getSourceFile(type.filePath);\n    if (!sf) {\n      host.error(`directive ${type.name} exists but has no source file`);\n      return [];\n    }\n    // TypeScript identifier of the directive declaration annotation (e.g. \"Component\" or\n    // \"Directive\") on a directive class.\n    const directiveIdentifier = findTightestNode(sf, declarationSpan.start);\n    if (!directiveIdentifier) {\n      host.error(`directive ${type.name} exists but has no identifier`);\n      return [];\n    }\n\n    for (const error of errors) {\n      results.push({\n        kind: ts.DiagnosticCategory.Error,\n        message: error.message,\n        span: error.span,\n      });\n    }\n\n    if (!modules.ngModuleByPipeOrDirective.has(declaration.type)) {\n      results.push(createDiagnostic(\n          declarationSpan, Diagnostic.directive_not_in_module,\n          metadata.isComponent ? 'Component' : 'Directive', type.name));\n    }\n\n    if (metadata.isComponent) {\n      const {template, templateUrl, styleUrls} = metadata.template !;\n      if (template === null && !templateUrl) {\n        results.push(createDiagnostic(\n            declarationSpan, Diagnostic.missing_template_and_templateurl, type.name));\n      } else if (templateUrl) {\n        if (template) {\n          results.push(createDiagnostic(\n              declarationSpan, Diagnostic.both_template_and_templateurl, type.name));\n        }\n\n        // Find templateUrl value from the directive call expression, which is the parent of the\n        // directive identifier.\n        //\n        // TODO: We should create an enum of the various properties a directive can have to use\n        // instead of string literals. We can then perform a mass migration of all literal usages.\n        const templateUrlNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'templateUrl', ts.isLiteralExpression);\n        if (!templateUrlNode) {\n          host.error(`templateUrl ${templateUrl} exists but its TypeScript node doesn't`);\n          return [];\n        }\n\n        results.push(...validateUrls([templateUrlNode], host.tsLsHost));\n      }\n\n      if (styleUrls.length > 0) {\n        // Find styleUrls value from the directive call expression, which is the parent of the\n        // directive identifier.\n        const styleUrlsNode = findPropertyValueOfType(\n            directiveIdentifier.parent, 'styleUrls', ts.isArrayLiteralExpression);\n        if (!styleUrlsNode) {\n          host.error(`styleUrls property exists but its TypeScript node doesn't'`);\n          return [];\n        }\n\n        results.push(...validateUrls(styleUrlsNode.elements, host.tsLsHost));\n      }\n    }\n  }\n\n  return results;\n}\n\n/**\n * Checks that URLs on a directive point to a valid file.\n * Note that this diagnostic check may require a filesystem hit, and thus may be slower than other\n * checks.\n *\n * @param urls urls to check for validity\n * @param tsLsHost TS LS host used for querying filesystem information\n * @return diagnosed url errors, if any\n */\nfunction validateUrls(\n    urls: ArrayLike<ts.Expression>, tsLsHost: Readonly<ts.LanguageServiceHost>): ng.Diagnostic[] {\n  if (!tsLsHost.fileExists) {\n    return [];\n  }\n\n  const allErrors: ng.Diagnostic[] = [];\n  // TODO(ayazhafiz): most of this logic can be unified with the logic in\n  // definitions.ts#getUrlFromProperty. Create a utility function to be used by both.\n  for (let i = 0; i < urls.length; ++i) {\n    const urlNode = urls[i];\n    if (!ts.isStringLiteralLike(urlNode)) {\n      // If a non-string value is assigned to a URL node (like `templateUrl`), a type error will be\n      // picked up by the TS Language Server.\n      continue;\n    }\n    const curPath = urlNode.getSourceFile().fileName;\n    const url = path.join(path.dirname(curPath), urlNode.text);\n    if (tsLsHost.fileExists(url)) continue;\n\n    // Exclude opening and closing quotes in the url span.\n    const urlSpan = {start: urlNode.getStart() + 1, end: urlNode.end - 1};\n    allErrors.push(createDiagnostic(urlSpan, Diagnostic.invalid_templateurl));\n  }\n  return allErrors;\n}\n\n/**\n * Return a recursive data structure that chains diagnostic messages.\n * @param chain\n */\nfunction chainDiagnostics(chain: ng.DiagnosticMessageChain): ts.DiagnosticMessageChain {\n  return {\n    messageText: chain.message,\n    category: ts.DiagnosticCategory.Error,\n    code: 0,\n    next: chain.next ? chain.next.map(chainDiagnostics) : undefined\n  };\n}\n\n/**\n * Convert ng.Diagnostic to ts.Diagnostic.\n * @param d diagnostic\n * @param file\n */\nexport function ngDiagnosticToTsDiagnostic(\n    d: ng.Diagnostic, file: ts.SourceFile | undefined): ts.Diagnostic {\n  return {\n    file,\n    start: d.span.start,\n    length: d.span.end - d.span.start,\n    messageText: typeof d.message === 'string' ? d.message : chainDiagnostics(d.message),\n    category: d.kind,\n    code: 0,\n    source: 'ng',\n  };\n}\n"]}
@@ -8,7 +8,7 @@
8
8
  /// <amd-module name="@angular/language-service/src/expression_diagnostics" />
9
9
  import { Node, TemplateAst, TemplateAstPath } from '@angular/compiler';
10
10
  import { SymbolQuery, SymbolTable } from './symbols';
11
- import { Diagnostic } from './types';
11
+ import * as ng from './types';
12
12
  export interface DiagnosticTemplateInfo {
13
13
  fileName?: string;
14
14
  offset: number;
@@ -17,7 +17,7 @@ export interface DiagnosticTemplateInfo {
17
17
  htmlAst: Node[];
18
18
  templateAst: TemplateAst[];
19
19
  }
20
- export declare function getTemplateExpressionDiagnostics(info: DiagnosticTemplateInfo): Diagnostic[];
20
+ export declare function getTemplateExpressionDiagnostics(info: DiagnosticTemplateInfo): ng.Diagnostic[];
21
21
  /**
22
22
  * Returns the symbols available in a particular scope of a template.
23
23
  * @param info parsed template information
@@ -11,14 +11,14 @@
11
11
  if (v !== undefined) module.exports = v;
12
12
  }
13
13
  else if (typeof define === "function" && define.amd) {
14
- define("@angular/language-service/src/expression_diagnostics", ["require", "exports", "tslib", "@angular/compiler", "typescript", "@angular/language-service/src/expression_type", "@angular/language-service/src/symbols", "@angular/language-service/src/utils"], factory);
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
15
  }
16
16
  })(function (require, exports) {
17
17
  "use strict";
18
18
  Object.defineProperty(exports, "__esModule", { value: true });
19
19
  var tslib_1 = require("tslib");
20
20
  var compiler_1 = require("@angular/compiler");
21
- var ts = require("typescript");
21
+ var diagnostic_messages_1 = require("@angular/language-service/src/diagnostic_messages");
22
22
  var expression_type_1 = require("@angular/language-service/src/expression_type");
23
23
  var symbols_1 = require("@angular/language-service/src/symbols");
24
24
  var utils_1 = require("@angular/language-service/src/utils");
@@ -102,10 +102,7 @@
102
102
  continue;
103
103
  }
104
104
  var _loop_2 = function (variable) {
105
- var symbol = info.members.get(variable.value);
106
- if (!symbol) {
107
- symbol = getVariableTypeFromDirectiveContext(variable.value, info.query, current);
108
- }
105
+ var symbol = getVariableTypeFromDirectiveContext(variable.value, info.query, current);
109
106
  var kind = info.query.getTypeKind(symbol);
110
107
  if (kind === symbols_1.BuiltinType.Any || kind === symbols_1.BuiltinType.Unbound) {
111
108
  // For special cases such as ngFor and ngIf, the any type is not very useful.
@@ -181,7 +178,7 @@
181
178
  */
182
179
  function refinedVariableType(value, mergedTable, query, templateElement) {
183
180
  if (value === '$implicit') {
184
- // Special case the ngFor directive
181
+ // Special case: ngFor directive
185
182
  var ngForDirective = templateElement.directives.find(function (d) {
186
183
  var name = compiler_1.identifierName(d.directive.type);
187
184
  return name == 'NgFor' || name == 'NgForOf';
@@ -200,12 +197,19 @@
200
197
  }
201
198
  }
202
199
  }
203
- // Special case the ngIf directive ( *ngIf="data$ | async as variable" )
204
- if (value === 'ngIf') {
200
+ if (value === 'ngIf' || value === '$implicit') {
205
201
  var ngIfDirective = templateElement.directives.find(function (d) { return compiler_1.identifierName(d.directive.type) === 'NgIf'; });
206
202
  if (ngIfDirective) {
203
+ // Special case: ngIf directive. The NgIf structural directive owns a template context with
204
+ // "$implicit" and "ngIf" members. These properties are typed as generics. Until the language
205
+ // service uses an Ivy and TypecheckBlock backend, we cannot bind these values to a concrete
206
+ // type without manual inference. To get the concrete type, look up the type of the "ngIf"
207
+ // import on the NgIf directive bound to the template.
208
+ //
209
+ // See @angular/common/ng_if.ts for more information.
207
210
  var ngIfBinding = ngIfDirective.inputs.find(function (i) { return i.directiveName === 'ngIf'; });
208
211
  if (ngIfBinding) {
212
+ // Check if there is a known type bound to the ngIf input.
209
213
  var bindingType = new expression_type_1.AstType(mergedTable, query, {}).getType(ngIfBinding.value);
210
214
  if (bindingType) {
211
215
  return bindingType;
@@ -302,8 +306,8 @@
302
306
  var context = this.info.query.getTemplateContext(directive.type.reference);
303
307
  if (context && !context.has(ast.value)) {
304
308
  var missingMember = ast.value === '$implicit' ? 'an implicit value' : "a member called '" + ast.value + "'";
305
- this.reportDiagnostic("The template context of '" + directive.type.reference.name + "' does not define " + missingMember + ".\n" +
306
- "If the context type is a base type or 'any', consider refining it to a more specific type.", spanOf(ast.sourceSpan), ts.DiagnosticCategory.Suggestion);
309
+ var span = this.absSpan(spanOf(ast.sourceSpan));
310
+ this.diagnostics.push(diagnostic_messages_1.createDiagnostic(span, diagnostic_messages_1.Diagnostic.template_context_missing_member, directive.type.reference.name, missingMember));
307
311
  }
308
312
  }
309
313
  };
@@ -337,10 +341,9 @@
337
341
  var analyzer = new expression_type_1.AstType(scope, this.info.query, { event: event });
338
342
  try {
339
343
  for (var _b = tslib_1.__values(analyzer.getDiagnostics(ast)), _c = _b.next(); !_c.done; _c = _b.next()) {
340
- var _d = _c.value, message = _d.message, span = _d.span, kind = _d.kind;
341
- span.start += offset;
342
- span.end += offset;
343
- this.reportDiagnostic(message, span, kind);
344
+ var diagnostic = _c.value;
345
+ diagnostic.span = this.absSpan(diagnostic.span, offset);
346
+ this.diagnostics.push(diagnostic);
344
347
  }
345
348
  }
346
349
  catch (e_4_1) { e_4 = { error: e_4_1 }; }
@@ -353,11 +356,12 @@
353
356
  };
354
357
  ExpressionDiagnosticsVisitor.prototype.push = function (ast) { this.path.push(ast); };
355
358
  ExpressionDiagnosticsVisitor.prototype.pop = function () { this.path.pop(); };
356
- ExpressionDiagnosticsVisitor.prototype.reportDiagnostic = function (message, span, kind) {
357
- if (kind === void 0) { kind = ts.DiagnosticCategory.Error; }
358
- span.start += this.info.offset;
359
- span.end += this.info.offset;
360
- this.diagnostics.push({ kind: kind, span: span, message: message });
359
+ ExpressionDiagnosticsVisitor.prototype.absSpan = function (span, additionalOffset) {
360
+ if (additionalOffset === void 0) { additionalOffset = 0; }
361
+ return {
362
+ start: span.start + this.info.offset + additionalOffset,
363
+ end: span.end + this.info.offset + additionalOffset,
364
+ };
361
365
  };
362
366
  return ExpressionDiagnosticsVisitor;
363
367
  }(compiler_1.RecursiveTemplateAstVisitor));
@@ -386,4 +390,4 @@
386
390
  return { start: sourceSpan.start.offset, end: sourceSpan.end.offset };
387
391
  }
388
392
  });
389
- //# 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 Inc. 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, Node, ParseSourceSpan, RecursiveTemplateAstVisitor, ReferenceAst, TemplateAst, TemplateAstPath, VariableAst, identifierName, templateVisitAll, tokenReference} from '@angular/compiler';\nimport * as ts from 'typescript';\n\nimport {AstType} from './expression_type';\nimport {BuiltinType, Definition, Span, Symbol, SymbolDeclaration, SymbolQuery, SymbolTable} from './symbols';\nimport {Diagnostic} from './types';\nimport {findOutputBinding, getPathToNodeAtPosition} from './utils';\n\nexport interface DiagnosticTemplateInfo {\n  fileName?: string;\n  offset: number;\n  query: SymbolQuery;\n  members: SymbolTable;\n  htmlAst: Node[];\n  templateAst: TemplateAst[];\n}\n\nexport function getTemplateExpressionDiagnostics(info: DiagnosticTemplateInfo): 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: 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() { return getDefinitionOf(info, reference); }\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: 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: 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 = info.members.get(variable.value);\n      if (!symbol) {\n        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.query, current);\n      }\n      results.push({\n        name: variable.name,\n        kind: 'variable',\n        type: symbol, get definition() { return getDefinitionOf(info, variable); },\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  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 query\n * @param templateElement\n */\nfunction refinedVariableType(\n    value: string, mergedTable: SymbolTable, query: SymbolQuery,\n    templateElement: EmbeddedTemplateAst): Symbol {\n  if (value === '$implicit') {\n    // Special case the 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 = new AstType(mergedTable, query, {}).getType(ngForOfBinding.value);\n        if (bindingType) {\n          const result = query.getElementType(bindingType);\n          if (result) {\n            return result;\n          }\n        }\n      }\n    }\n  }\n\n  // Special case the ngIf directive ( *ngIf=\"data$ | async as variable\" )\n  if (value === 'ngIf') {\n    const ngIfDirective =\n        templateElement.directives.find(d => identifierName(d.directive.type) === 'NgIf');\n    if (ngIfDirective) {\n      const ngIfBinding = ngIfDirective.inputs.find(i => i.directiveName === 'ngIf');\n      if (ngIfBinding) {\n        const bindingType = new AstType(mergedTable, query, {}).getType(ngIfBinding.value);\n        if (bindingType) {\n          return bindingType;\n        }\n      }\n    }\n  }\n\n  // We can't do better, return any\n  return query.getBuiltinType(BuiltinType.Any);\n}\n\nfunction getEventDeclaration(\n    info: 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: 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: Diagnostic[] = [];\n\n  constructor(\n      private info: 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        this.reportDiagnostic(\n            `The template context of '${directive.type.reference.name}' does not define ${missingMember}.\\n` +\n                `If the context type is a base type or 'any', consider refining it to a more specific type.`,\n            spanOf(ast.sourceSpan), ts.DiagnosticCategory.Suggestion);\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, event: boolean) {\n    const scope = this.getExpressionScope(this.path, event);\n    const analyzer = new AstType(scope, this.info.query, {event});\n    for (const {message, span, kind} of analyzer.getDiagnostics(ast)) {\n      span.start += offset;\n      span.end += offset;\n      this.reportDiagnostic(message as string, span, kind);\n    }\n  }\n\n  private push(ast: TemplateAst) { this.path.push(ast); }\n\n  private pop() { this.path.pop(); }\n\n  private reportDiagnostic(\n      message: string, span: Span, kind: ts.DiagnosticCategory = ts.DiagnosticCategory.Error) {\n    span.start += this.info.offset;\n    span.end += this.info.offset;\n    this.diagnostics.push({kind, span, message});\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"]}
393
+ //# 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 Inc. 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, Node, ParseSourceSpan, RecursiveTemplateAstVisitor, ReferenceAst, TemplateAst, TemplateAstPath, VariableAst, identifierName, templateVisitAll, tokenReference} from '@angular/compiler';\n\nimport {Diagnostic, createDiagnostic} 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 interface DiagnosticTemplateInfo {\n  fileName?: string;\n  offset: number;\n  query: SymbolQuery;\n  members: SymbolTable;\n  htmlAst: Node[];\n  templateAst: TemplateAst[];\n}\n\nexport function getTemplateExpressionDiagnostics(info: 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: 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() { return getDefinitionOf(info, reference); }\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: 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: 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.query, current);\n      }\n      results.push({\n        name: variable.name,\n        kind: 'variable',\n        type: symbol, get definition() { return getDefinitionOf(info, variable); },\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 query\n * @param templateElement\n */\nfunction refinedVariableType(\n    value: string, mergedTable: SymbolTable, query: SymbolQuery,\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 = new AstType(mergedTable, query, {}).getType(ngForOfBinding.value);\n        if (bindingType) {\n          const result = 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 = new AstType(mergedTable, query, {}).getType(ngIfBinding.value);\n        if (bindingType) {\n          return bindingType;\n        }\n      }\n    }\n  }\n\n  // We can't do better, return any\n  return query.getBuiltinType(BuiltinType.Any);\n}\n\nfunction getEventDeclaration(\n    info: 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: 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: 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, event: boolean) {\n    const scope = this.getExpressionScope(this.path, event);\n    const analyzer = new AstType(scope, this.info.query, {event});\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) { this.path.push(ast); }\n\n  private pop() { this.path.pop(); }\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"]}
@@ -46,6 +46,5 @@ export declare class AstType implements AstVisitor {
46
46
  private get undefinedType();
47
47
  private resolveMethodCall;
48
48
  private resolvePropertyRead;
49
- private reportDiagnostic;
50
49
  private isAny;
51
50
  }