@angular/compiler-cli 11.2.9 → 11.2.13

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.
Files changed (60) hide show
  1. package/linker/src/file_linker/partial_linkers/partial_linker_selector.d.ts +1 -1
  2. package/linker/src/file_linker/partial_linkers/partial_linker_selector.js +4 -4
  3. package/ngcc/src/analysis/decoration_analyzer.js +6 -6
  4. package/ngcc/src/packages/build_marker.d.ts +1 -1
  5. package/ngcc/src/packages/build_marker.js +1 -1
  6. package/package.json +2 -2
  7. package/src/ngtsc/annotations/src/component.d.ts +2 -3
  8. package/src/ngtsc/annotations/src/component.js +4 -5
  9. package/src/ngtsc/annotations/src/directive.d.ts +3 -4
  10. package/src/ngtsc/annotations/src/directive.js +6 -7
  11. package/src/ngtsc/annotations/src/injectable.d.ts +1 -3
  12. package/src/ngtsc/annotations/src/injectable.js +9 -9
  13. package/src/ngtsc/annotations/src/metadata.d.ts +1 -2
  14. package/src/ngtsc/annotations/src/metadata.js +5 -5
  15. package/src/ngtsc/annotations/src/ng_module.d.ts +2 -3
  16. package/src/ngtsc/annotations/src/ng_module.js +4 -5
  17. package/src/ngtsc/annotations/src/pipe.d.ts +1 -3
  18. package/src/ngtsc/annotations/src/pipe.js +4 -5
  19. package/src/ngtsc/annotations/src/util.d.ts +5 -5
  20. package/src/ngtsc/annotations/src/util.js +12 -11
  21. package/src/ngtsc/core/api/src/interfaces.d.ts +4 -1
  22. package/src/ngtsc/core/api/src/interfaces.js +1 -1
  23. package/src/ngtsc/core/src/compiler.js +9 -10
  24. package/src/ngtsc/core/src/host.d.ts +1 -1
  25. package/src/ngtsc/diagnostics/src/error_code.d.ts +5 -0
  26. package/src/ngtsc/diagnostics/src/error_code.js +6 -1
  27. package/src/ngtsc/imports/index.d.ts +1 -1
  28. package/src/ngtsc/imports/index.js +2 -3
  29. package/src/ngtsc/imports/src/default.d.ts +10 -37
  30. package/src/ngtsc/imports/src/default.js +22 -37
  31. package/src/ngtsc/incremental/src/state.js +6 -4
  32. package/src/ngtsc/resource/src/loader.js +3 -2
  33. package/src/ngtsc/transform/src/transform.d.ts +2 -2
  34. package/src/ngtsc/transform/src/transform.js +18 -16
  35. package/src/ngtsc/translator/index.d.ts +1 -1
  36. package/src/ngtsc/translator/index.js +1 -1
  37. package/src/ngtsc/translator/src/translator.d.ts +3 -3
  38. package/src/ngtsc/translator/src/translator.js +3 -3
  39. package/src/ngtsc/typecheck/api/checker.d.ts +1 -1
  40. package/src/ngtsc/typecheck/api/checker.js +1 -1
  41. package/src/ngtsc/typecheck/api/completion.d.ts +5 -0
  42. package/src/ngtsc/typecheck/api/completion.js +1 -1
  43. package/src/ngtsc/typecheck/src/checker.d.ts +2 -2
  44. package/src/ngtsc/typecheck/src/checker.js +3 -3
  45. package/src/ngtsc/typecheck/src/completion.d.ts +17 -7
  46. package/src/ngtsc/typecheck/src/completion.js +82 -47
  47. package/src/ngtsc/typecheck/src/context.js +26 -14
  48. package/src/ngtsc/typecheck/src/expression.js +4 -2
  49. package/src/ngtsc/typecheck/src/host.js +5 -7
  50. package/src/ngtsc/typecheck/src/tcb_util.d.ts +24 -2
  51. package/src/ngtsc/typecheck/src/tcb_util.js +40 -11
  52. package/src/ngtsc/typecheck/src/type_check_block.d.ts +28 -1
  53. package/src/ngtsc/typecheck/src/type_check_block.js +61 -25
  54. package/src/ngtsc/typecheck/src/type_check_file.d.ts +2 -1
  55. package/src/ngtsc/typecheck/src/type_check_file.js +3 -3
  56. package/src/ngtsc/typecheck/src/type_constructor.js +4 -8
  57. package/src/ngtsc/util/src/typescript.d.ts +4 -0
  58. package/src/ngtsc/util/src/typescript.js +13 -2
  59. package/src/transformers/program.js +70 -3
  60. package/src/version.js +1 -1
@@ -7,7 +7,7 @@
7
7
  */
8
8
  /// <amd-module name="@angular/compiler-cli/src/ngtsc/typecheck/src/tcb_util" />
9
9
  import { AbsoluteSourceSpan, ParseSourceSpan } from '@angular/compiler';
10
- import { ClassDeclaration } from '@angular/compiler-cli/src/ngtsc/reflection';
10
+ import { ClassDeclaration, ReflectionHost } from '@angular/compiler-cli/src/ngtsc/reflection';
11
11
  import * as ts from 'typescript';
12
12
  import { Reference } from '../../imports';
13
13
  import { FullTemplateMapping, SourceLocation, TemplateId, TemplateSourceMapping } from '../api';
@@ -29,7 +29,28 @@ export interface TemplateSourceResolver {
29
29
  */
30
30
  toParseSourceSpan(id: TemplateId, span: AbsoluteSourceSpan): ParseSourceSpan | null;
31
31
  }
32
- export declare function requiresInlineTypeCheckBlock(node: ClassDeclaration<ts.ClassDeclaration>, usedPipes: Map<string, Reference<ClassDeclaration<ts.ClassDeclaration>>>): boolean;
32
+ /**
33
+ * Indicates whether a particular component requires an inline type check block.
34
+ *
35
+ * This is not a boolean state as inlining might only be required to get the best possible
36
+ * type-checking, but the component could theoretically still be checked without it.
37
+ */
38
+ export declare enum TcbInliningRequirement {
39
+ /**
40
+ * There is no way to type check this component without inlining.
41
+ */
42
+ MustInline = 0,
43
+ /**
44
+ * Inlining should be used due to the component's generic bounds, but a non-inlining fallback
45
+ * method can be used if that's not possible.
46
+ */
47
+ ShouldInlineForGenericBounds = 1,
48
+ /**
49
+ * There is no requirement for this component's TCB to be inlined.
50
+ */
51
+ None = 2
52
+ }
53
+ export declare function requiresInlineTypeCheckBlock(node: ClassDeclaration<ts.ClassDeclaration>, usedPipes: Map<string, Reference<ClassDeclaration<ts.ClassDeclaration>>>, reflector: ReflectionHost): TcbInliningRequirement;
33
54
  /** Maps a shim position back to a template location. */
34
55
  export declare function getTemplateMapping(shimSf: ts.SourceFile, position: number, resolver: TemplateSourceResolver, isDiagnosticRequest: boolean): FullTemplateMapping | null;
35
56
  export declare function findTypeCheckBlock(file: ts.SourceFile, id: TemplateId, isDiagnosticRequest: boolean): ts.Node | null;
@@ -40,3 +61,4 @@ export declare function findTypeCheckBlock(file: ts.SourceFile, id: TemplateId,
40
61
  * returns null.
41
62
  */
42
63
  export declare function findSourceLocation(node: ts.Node, sourceFile: ts.SourceFile, isDiagnosticsRequest: boolean): SourceLocation | null;
64
+ export declare function checkIfGenericTypeBoundsAreContextFree(node: ClassDeclaration<ts.ClassDeclaration>, reflector: ReflectionHost): boolean;
@@ -11,37 +11,61 @@
11
11
  if (v !== undefined) module.exports = v;
12
12
  }
13
13
  else if (typeof define === "function" && define.amd) {
14
- define("@angular/compiler-cli/src/ngtsc/typecheck/src/tcb_util", ["require", "exports", "tslib", "typescript", "@angular/compiler-cli/src/ngtsc/util/src/typescript", "@angular/compiler-cli/src/ngtsc/typecheck/src/comments", "@angular/compiler-cli/src/ngtsc/typecheck/src/ts_util"], factory);
14
+ define("@angular/compiler-cli/src/ngtsc/typecheck/src/tcb_util", ["require", "exports", "tslib", "typescript", "@angular/compiler-cli/src/ngtsc/util/src/typescript", "@angular/compiler-cli/src/ngtsc/typecheck/src/comments", "@angular/compiler-cli/src/ngtsc/typecheck/src/ts_util", "@angular/compiler-cli/src/ngtsc/typecheck/src/type_parameter_emitter"], factory);
15
15
  }
16
16
  })(function (require, exports) {
17
17
  "use strict";
18
18
  Object.defineProperty(exports, "__esModule", { value: true });
19
- exports.findSourceLocation = exports.findTypeCheckBlock = exports.getTemplateMapping = exports.requiresInlineTypeCheckBlock = void 0;
19
+ exports.checkIfGenericTypeBoundsAreContextFree = exports.findSourceLocation = exports.findTypeCheckBlock = exports.getTemplateMapping = exports.requiresInlineTypeCheckBlock = exports.TcbInliningRequirement = void 0;
20
20
  var tslib_1 = require("tslib");
21
21
  var ts = require("typescript");
22
22
  var typescript_1 = require("@angular/compiler-cli/src/ngtsc/util/src/typescript");
23
23
  var comments_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/src/comments");
24
24
  var ts_util_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/src/ts_util");
25
- function requiresInlineTypeCheckBlock(node, usedPipes) {
25
+ var type_parameter_emitter_1 = require("@angular/compiler-cli/src/ngtsc/typecheck/src/type_parameter_emitter");
26
+ /**
27
+ * Indicates whether a particular component requires an inline type check block.
28
+ *
29
+ * This is not a boolean state as inlining might only be required to get the best possible
30
+ * type-checking, but the component could theoretically still be checked without it.
31
+ */
32
+ var TcbInliningRequirement;
33
+ (function (TcbInliningRequirement) {
34
+ /**
35
+ * There is no way to type check this component without inlining.
36
+ */
37
+ TcbInliningRequirement[TcbInliningRequirement["MustInline"] = 0] = "MustInline";
38
+ /**
39
+ * Inlining should be used due to the component's generic bounds, but a non-inlining fallback
40
+ * method can be used if that's not possible.
41
+ */
42
+ TcbInliningRequirement[TcbInliningRequirement["ShouldInlineForGenericBounds"] = 1] = "ShouldInlineForGenericBounds";
43
+ /**
44
+ * There is no requirement for this component's TCB to be inlined.
45
+ */
46
+ TcbInliningRequirement[TcbInliningRequirement["None"] = 2] = "None";
47
+ })(TcbInliningRequirement = exports.TcbInliningRequirement || (exports.TcbInliningRequirement = {}));
48
+ function requiresInlineTypeCheckBlock(node, usedPipes, reflector) {
26
49
  // In order to qualify for a declared TCB (not inline) two conditions must be met:
27
50
  // 1) the class must be exported
28
- // 2) it must not have constrained generic types
51
+ // 2) it must not have contextual generic type bounds
29
52
  if (!ts_util_1.checkIfClassIsExported(node)) {
30
53
  // Condition 1 is false, the class is not exported.
31
- return true;
54
+ return TcbInliningRequirement.MustInline;
32
55
  }
33
- else if (!ts_util_1.checkIfGenericTypesAreUnbound(node)) {
34
- // Condition 2 is false, the class has constrained generic types
35
- return true;
56
+ else if (!checkIfGenericTypeBoundsAreContextFree(node, reflector)) {
57
+ // Condition 2 is false, the class has constrained generic types. It should be checked with an
58
+ // inline TCB if possible, but can potentially use fallbacks to avoid inlining if not.
59
+ return TcbInliningRequirement.ShouldInlineForGenericBounds;
36
60
  }
37
61
  else if (Array.from(usedPipes.values())
38
62
  .some(function (pipeRef) { return !ts_util_1.checkIfClassIsExported(pipeRef.node); })) {
39
63
  // If one of the pipes used by the component is not exported, a non-inline TCB will not be able
40
64
  // to import it, so this requires an inline TCB.
41
- return true;
65
+ return TcbInliningRequirement.MustInline;
42
66
  }
43
67
  else {
44
- return false;
68
+ return TcbInliningRequirement.None;
45
69
  }
46
70
  }
47
71
  exports.requiresInlineTypeCheckBlock = requiresInlineTypeCheckBlock;
@@ -132,5 +156,10 @@
132
156
  return commentText;
133
157
  }) || null;
134
158
  }
159
+ function checkIfGenericTypeBoundsAreContextFree(node, reflector) {
160
+ // Generic type parameters are considered context free if they can be emitted into any context.
161
+ return new type_parameter_emitter_1.TypeParameterEmitter(node.typeParameters, reflector).canEmit();
162
+ }
163
+ exports.checkIfGenericTypeBoundsAreContextFree = checkIfGenericTypeBoundsAreContextFree;
135
164
  });
136
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tcb_util.js","sourceRoot":"","sources":["../../../../../../../../../packages/compiler-cli/src/ngtsc/typecheck/src/tcb_util.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 {AbsoluteSourceSpan, ParseSourceSpan} from '@angular/compiler';\nimport {ClassDeclaration} from '@angular/compiler-cli/src/ngtsc/reflection';\nimport * as ts from 'typescript';\n\nimport {Reference} from '../../imports';\nimport {getTokenAtPosition} from '../../util/src/typescript';\nimport {FullTemplateMapping, SourceLocation, TemplateId, TemplateSourceMapping} from '../api';\n\nimport {hasIgnoreForDiagnosticsMarker, readSpanComment} from './comments';\nimport {checkIfClassIsExported, checkIfGenericTypesAreUnbound} from './ts_util';\n\n/**\n * Adapter interface which allows the template type-checking diagnostics code to interpret offsets\n * in a TCB and map them back to original locations in the template.\n */\nexport interface TemplateSourceResolver {\n  getTemplateId(node: ts.ClassDeclaration): TemplateId;\n\n  /**\n   * For the given template id, retrieve the original source mapping which describes how the offsets\n   * in the template should be interpreted.\n   */\n  getSourceMapping(id: TemplateId): TemplateSourceMapping;\n\n  /**\n   * Convert an absolute source span associated with the given template id into a full\n   * `ParseSourceSpan`. The returned parse span has line and column numbers in addition to only\n   * absolute offsets and gives access to the original template source.\n   */\n  toParseSourceSpan(id: TemplateId, span: AbsoluteSourceSpan): ParseSourceSpan|null;\n}\n\nexport function requiresInlineTypeCheckBlock(\n    node: ClassDeclaration<ts.ClassDeclaration>,\n    usedPipes: Map<string, Reference<ClassDeclaration<ts.ClassDeclaration>>>): boolean {\n  // In order to qualify for a declared TCB (not inline) two conditions must be met:\n  // 1) the class must be exported\n  // 2) it must not have constrained generic types\n  if (!checkIfClassIsExported(node)) {\n    // Condition 1 is false, the class is not exported.\n    return true;\n  } else if (!checkIfGenericTypesAreUnbound(node)) {\n    // Condition 2 is false, the class has constrained generic types\n    return true;\n  } else if (Array.from(usedPipes.values())\n                 .some(pipeRef => !checkIfClassIsExported(pipeRef.node))) {\n    // If one of the pipes used by the component is not exported, a non-inline TCB will not be able\n    // to import it, so this requires an inline TCB.\n    return true;\n  } else {\n    return false;\n  }\n}\n\n/** Maps a shim position back to a template location. */\nexport function getTemplateMapping(\n    shimSf: ts.SourceFile, position: number, resolver: TemplateSourceResolver,\n    isDiagnosticRequest: boolean): FullTemplateMapping|null {\n  const node = getTokenAtPosition(shimSf, position);\n  const sourceLocation = findSourceLocation(node, shimSf, isDiagnosticRequest);\n  if (sourceLocation === null) {\n    return null;\n  }\n\n  const mapping = resolver.getSourceMapping(sourceLocation.id);\n  const span = resolver.toParseSourceSpan(sourceLocation.id, sourceLocation.span);\n  if (span === null) {\n    return null;\n  }\n  // TODO(atscott): Consider adding a context span by walking up from `node` until we get a\n  // different span.\n  return {sourceLocation, templateSourceMapping: mapping, span};\n}\n\nexport function findTypeCheckBlock(\n    file: ts.SourceFile, id: TemplateId, isDiagnosticRequest: boolean): ts.Node|null {\n  for (const stmt of file.statements) {\n    if (ts.isFunctionDeclaration(stmt) && getTemplateId(stmt, file, isDiagnosticRequest) === id) {\n      return stmt;\n    }\n  }\n  return null;\n}\n\n/**\n * Traverses up the AST starting from the given node to extract the source location from comments\n * that have been emitted into the TCB. If the node does not exist within a TCB, or if an ignore\n * marker comment is found up the tree (and this is part of a diagnostic request), this function\n * returns null.\n */\nexport function findSourceLocation(\n    node: ts.Node, sourceFile: ts.SourceFile, isDiagnosticsRequest: boolean): SourceLocation|null {\n  // Search for comments until the TCB's function declaration is encountered.\n  while (node !== undefined && !ts.isFunctionDeclaration(node)) {\n    if (hasIgnoreForDiagnosticsMarker(node, sourceFile) && isDiagnosticsRequest) {\n      // There's an ignore marker on this node, so the diagnostic should not be reported.\n      return null;\n    }\n\n    const span = readSpanComment(node, sourceFile);\n    if (span !== null) {\n      // Once the positional information has been extracted, search further up the TCB to extract\n      // the unique id that is attached with the TCB's function declaration.\n      const id = getTemplateId(node, sourceFile, isDiagnosticsRequest);\n      if (id === null) {\n        return null;\n      }\n      return {id, span};\n    }\n\n    node = node.parent;\n  }\n\n  return null;\n}\n\nfunction getTemplateId(\n    node: ts.Node, sourceFile: ts.SourceFile, isDiagnosticRequest: boolean): TemplateId|null {\n  // Walk up to the function declaration of the TCB, the file information is attached there.\n  while (!ts.isFunctionDeclaration(node)) {\n    if (hasIgnoreForDiagnosticsMarker(node, sourceFile) && isDiagnosticRequest) {\n      // There's an ignore marker on this node, so the diagnostic should not be reported.\n      return null;\n    }\n    node = node.parent;\n\n    // Bail once we have reached the root.\n    if (node === undefined) {\n      return null;\n    }\n  }\n\n  const start = node.getFullStart();\n  return ts.forEachLeadingCommentRange(sourceFile.text, start, (pos, end, kind) => {\n    if (kind !== ts.SyntaxKind.MultiLineCommentTrivia) {\n      return null;\n    }\n    const commentText = sourceFile.text.substring(pos + 2, end - 2);\n    return commentText;\n  }) as TemplateId || null;\n}\n"]}
165
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tcb_util.js","sourceRoot":"","sources":["../../../../../../../../../packages/compiler-cli/src/ngtsc/typecheck/src/tcb_util.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 {AbsoluteSourceSpan, ParseSourceSpan} from '@angular/compiler';\nimport {ClassDeclaration, ReflectionHost} from '@angular/compiler-cli/src/ngtsc/reflection';\nimport * as ts from 'typescript';\n\nimport {Reference} from '../../imports';\nimport {getTokenAtPosition} from '../../util/src/typescript';\nimport {FullTemplateMapping, SourceLocation, TemplateId, TemplateSourceMapping} from '../api';\n\nimport {hasIgnoreForDiagnosticsMarker, readSpanComment} from './comments';\nimport {checkIfClassIsExported, checkIfGenericTypesAreUnbound} from './ts_util';\nimport {TypeParameterEmitter} from './type_parameter_emitter';\n\n/**\n * Adapter interface which allows the template type-checking diagnostics code to interpret offsets\n * in a TCB and map them back to original locations in the template.\n */\nexport interface TemplateSourceResolver {\n  getTemplateId(node: ts.ClassDeclaration): TemplateId;\n\n  /**\n   * For the given template id, retrieve the original source mapping which describes how the offsets\n   * in the template should be interpreted.\n   */\n  getSourceMapping(id: TemplateId): TemplateSourceMapping;\n\n  /**\n   * Convert an absolute source span associated with the given template id into a full\n   * `ParseSourceSpan`. The returned parse span has line and column numbers in addition to only\n   * absolute offsets and gives access to the original template source.\n   */\n  toParseSourceSpan(id: TemplateId, span: AbsoluteSourceSpan): ParseSourceSpan|null;\n}\n\n/**\n * Indicates whether a particular component requires an inline type check block.\n *\n * This is not a boolean state as inlining might only be required to get the best possible\n * type-checking, but the component could theoretically still be checked without it.\n */\nexport enum TcbInliningRequirement {\n  /**\n   * There is no way to type check this component without inlining.\n   */\n  MustInline,\n\n  /**\n   * Inlining should be used due to the component's generic bounds, but a non-inlining fallback\n   * method can be used if that's not possible.\n   */\n  ShouldInlineForGenericBounds,\n\n  /**\n   * There is no requirement for this component's TCB to be inlined.\n   */\n  None,\n}\n\nexport function requiresInlineTypeCheckBlock(\n    node: ClassDeclaration<ts.ClassDeclaration>,\n    usedPipes: Map<string, Reference<ClassDeclaration<ts.ClassDeclaration>>>,\n    reflector: ReflectionHost): TcbInliningRequirement {\n  // In order to qualify for a declared TCB (not inline) two conditions must be met:\n  // 1) the class must be exported\n  // 2) it must not have contextual generic type bounds\n  if (!checkIfClassIsExported(node)) {\n    // Condition 1 is false, the class is not exported.\n    return TcbInliningRequirement.MustInline;\n  } else if (!checkIfGenericTypeBoundsAreContextFree(node, reflector)) {\n    // Condition 2 is false, the class has constrained generic types. It should be checked with an\n    // inline TCB if possible, but can potentially use fallbacks to avoid inlining if not.\n    return TcbInliningRequirement.ShouldInlineForGenericBounds;\n  } else if (Array.from(usedPipes.values())\n                 .some(pipeRef => !checkIfClassIsExported(pipeRef.node))) {\n    // If one of the pipes used by the component is not exported, a non-inline TCB will not be able\n    // to import it, so this requires an inline TCB.\n    return TcbInliningRequirement.MustInline;\n  } else {\n    return TcbInliningRequirement.None;\n  }\n}\n\n/** Maps a shim position back to a template location. */\nexport function getTemplateMapping(\n    shimSf: ts.SourceFile, position: number, resolver: TemplateSourceResolver,\n    isDiagnosticRequest: boolean): FullTemplateMapping|null {\n  const node = getTokenAtPosition(shimSf, position);\n  const sourceLocation = findSourceLocation(node, shimSf, isDiagnosticRequest);\n  if (sourceLocation === null) {\n    return null;\n  }\n\n  const mapping = resolver.getSourceMapping(sourceLocation.id);\n  const span = resolver.toParseSourceSpan(sourceLocation.id, sourceLocation.span);\n  if (span === null) {\n    return null;\n  }\n  // TODO(atscott): Consider adding a context span by walking up from `node` until we get a\n  // different span.\n  return {sourceLocation, templateSourceMapping: mapping, span};\n}\n\nexport function findTypeCheckBlock(\n    file: ts.SourceFile, id: TemplateId, isDiagnosticRequest: boolean): ts.Node|null {\n  for (const stmt of file.statements) {\n    if (ts.isFunctionDeclaration(stmt) && getTemplateId(stmt, file, isDiagnosticRequest) === id) {\n      return stmt;\n    }\n  }\n  return null;\n}\n\n/**\n * Traverses up the AST starting from the given node to extract the source location from comments\n * that have been emitted into the TCB. If the node does not exist within a TCB, or if an ignore\n * marker comment is found up the tree (and this is part of a diagnostic request), this function\n * returns null.\n */\nexport function findSourceLocation(\n    node: ts.Node, sourceFile: ts.SourceFile, isDiagnosticsRequest: boolean): SourceLocation|null {\n  // Search for comments until the TCB's function declaration is encountered.\n  while (node !== undefined && !ts.isFunctionDeclaration(node)) {\n    if (hasIgnoreForDiagnosticsMarker(node, sourceFile) && isDiagnosticsRequest) {\n      // There's an ignore marker on this node, so the diagnostic should not be reported.\n      return null;\n    }\n\n    const span = readSpanComment(node, sourceFile);\n    if (span !== null) {\n      // Once the positional information has been extracted, search further up the TCB to extract\n      // the unique id that is attached with the TCB's function declaration.\n      const id = getTemplateId(node, sourceFile, isDiagnosticsRequest);\n      if (id === null) {\n        return null;\n      }\n      return {id, span};\n    }\n\n    node = node.parent;\n  }\n\n  return null;\n}\n\nfunction getTemplateId(\n    node: ts.Node, sourceFile: ts.SourceFile, isDiagnosticRequest: boolean): TemplateId|null {\n  // Walk up to the function declaration of the TCB, the file information is attached there.\n  while (!ts.isFunctionDeclaration(node)) {\n    if (hasIgnoreForDiagnosticsMarker(node, sourceFile) && isDiagnosticRequest) {\n      // There's an ignore marker on this node, so the diagnostic should not be reported.\n      return null;\n    }\n    node = node.parent;\n\n    // Bail once we have reached the root.\n    if (node === undefined) {\n      return null;\n    }\n  }\n\n  const start = node.getFullStart();\n  return ts.forEachLeadingCommentRange(sourceFile.text, start, (pos, end, kind) => {\n    if (kind !== ts.SyntaxKind.MultiLineCommentTrivia) {\n      return null;\n    }\n    const commentText = sourceFile.text.substring(pos + 2, end - 2);\n    return commentText;\n  }) as TemplateId || null;\n}\n\nexport function checkIfGenericTypeBoundsAreContextFree(\n    node: ClassDeclaration<ts.ClassDeclaration>, reflector: ReflectionHost): boolean {\n  // Generic type parameters are considered context free if they can be emitted into any context.\n  return new TypeParameterEmitter(node.typeParameters, reflector).canEmit();\n}\n"]}
@@ -14,6 +14,31 @@ import { TemplateId, TypeCheckableDirectiveMeta, TypeCheckBlockMetadata } from '
14
14
  import { DomSchemaChecker } from './dom';
15
15
  import { Environment } from './environment';
16
16
  import { OutOfBandDiagnosticRecorder } from './oob';
17
+ /**
18
+ * Controls how generics for the component context class will be handled during TCB generation.
19
+ */
20
+ export declare enum TcbGenericContextBehavior {
21
+ /**
22
+ * References to generic parameter bounds will be emitted via the `TypeParameterEmitter`.
23
+ *
24
+ * The caller must verify that all parameter bounds are emittable in order to use this mode.
25
+ */
26
+ UseEmitter = 0,
27
+ /**
28
+ * Generic parameter declarations will be copied directly from the `ts.ClassDeclaration` of the
29
+ * component class.
30
+ *
31
+ * The caller must only use the generated TCB code in a context where such copies will still be
32
+ * valid, such as an inline type check block.
33
+ */
34
+ CopyClassNodes = 1,
35
+ /**
36
+ * Any generic parameters for the component context class will be set to `any`.
37
+ *
38
+ * Produces a less useful type, but is always safe to use.
39
+ */
40
+ FallbackToAny = 2
41
+ }
17
42
  /**
18
43
  * Given a `ts.ClassDeclaration` for a component, and metadata regarding that component, compose a
19
44
  * "type check block" function.
@@ -35,8 +60,10 @@ import { OutOfBandDiagnosticRecorder } from './oob';
35
60
  * and bindings.
36
61
  * @param oobRecorder used to record errors regarding template elements which could not be correctly
37
62
  * translated into types during TCB generation.
63
+ * @param genericContextBehavior controls how generic parameters (especially parameters with generic
64
+ * bounds) will be referenced from the generated TCB code.
38
65
  */
39
- export declare function generateTypeCheckBlock(env: Environment, ref: Reference<ClassDeclaration<ts.ClassDeclaration>>, name: ts.Identifier, meta: TypeCheckBlockMetadata, domSchemaChecker: DomSchemaChecker, oobRecorder: OutOfBandDiagnosticRecorder): ts.FunctionDeclaration;
66
+ export declare function generateTypeCheckBlock(env: Environment, ref: Reference<ClassDeclaration<ts.ClassDeclaration>>, name: ts.Identifier, meta: TypeCheckBlockMetadata, domSchemaChecker: DomSchemaChecker, oobRecorder: OutOfBandDiagnosticRecorder, genericContextBehavior: TcbGenericContextBehavior): ts.FunctionDeclaration;
40
67
  /**
41
68
  * A code generation operation that's involved in the construction of a Type Check Block.
42
69
  *