@angular/core 10.0.0 → 10.0.4

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 (49) hide show
  1. package/bundles/core-testing.umd.js +321 -272
  2. package/bundles/core-testing.umd.js.map +1 -1
  3. package/bundles/core-testing.umd.min.js +11 -25
  4. package/bundles/core-testing.umd.min.js.map +1 -1
  5. package/bundles/core.umd.js +620 -832
  6. package/bundles/core.umd.js.map +1 -1
  7. package/bundles/core.umd.min.js +128 -282
  8. package/bundles/core.umd.min.js.map +1 -1
  9. package/core.d.ts +69 -65
  10. package/core.metadata.json +1 -1
  11. package/esm2015/src/application_init.js +9 -5
  12. package/esm2015/src/application_tokens.js +16 -12
  13. package/esm2015/src/change_detection/change_detector_ref.js +6 -3
  14. package/esm2015/src/change_detection/constants.js +3 -1
  15. package/esm2015/src/di/injectable.js +1 -1
  16. package/esm2015/src/di/metadata.js +2 -2
  17. package/esm2015/src/interface/type.js +2 -2
  18. package/esm2015/src/linker/ng_module_factory.js +3 -5
  19. package/esm2015/src/linker/view_ref.js +3 -6
  20. package/esm2015/src/metadata/directives.js +1 -1
  21. package/esm2015/src/metadata/ng_module.js +1 -1
  22. package/esm2015/src/render3/component_ref.js +4 -13
  23. package/esm2015/src/render3/di.js +12 -4
  24. package/esm2015/src/render3/i18n.js +3 -3
  25. package/esm2015/src/render3/instructions/di.js +1 -1
  26. package/esm2015/src/render3/instructions/listener.js +2 -2
  27. package/esm2015/src/render3/instructions/shared.js +8 -18
  28. package/esm2015/src/render3/node_assert.js +3 -3
  29. package/esm2015/src/render3/node_manipulation.js +11 -7
  30. package/esm2015/src/render3/query.js +2 -2
  31. package/esm2015/src/render3/view_engine_compatibility.js +2 -2
  32. package/esm2015/src/render3/view_ref.js +7 -5
  33. package/esm2015/src/sanitization/html_sanitizer.js +3 -3
  34. package/esm2015/src/sanitization/inert_body.js +39 -82
  35. package/esm2015/src/util/ng_dev_mode.js +2 -2
  36. package/esm2015/src/version.js +1 -1
  37. package/esm2015/testing/src/fake_async_fallback.js +5 -1
  38. package/esm2015/testing/src/r3_test_bed.js +2 -2
  39. package/fesm2015/core.js +118 -159
  40. package/fesm2015/core.js.map +1 -1
  41. package/fesm2015/testing.js +6 -2
  42. package/fesm2015/testing.js.map +1 -1
  43. package/package.json +1 -1
  44. package/schematics/migrations/undecorated-classes-with-decorated-fields/transform.d.ts +7 -0
  45. package/schematics/migrations/undecorated-classes-with-decorated-fields/transform.js +125 -59
  46. package/src/r3_symbols.d.ts +15 -17
  47. package/testing/testing.d.ts +1 -1
  48. package/testing.d.ts +1 -1
  49. package/esm2015/src/util/WrappedValue.js +0 -48
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@angular/core",
3
- "version": "10.0.0",
3
+ "version": "10.0.4",
4
4
  "description": "Angular - the core framework",
5
5
  "author": "angular",
6
6
  "license": "MIT",
@@ -52,5 +52,12 @@ export declare class UndecoratedClassesWithDecoratedFieldsTransform {
52
52
  * considered Angular features..
53
53
  */
54
54
  private _determineClassKind;
55
+ /**
56
+ * Checks whether a given class has been reported as ambiguous in previous
57
+ * migration run. e.g. when build targets are migrated first, and then test
58
+ * targets that have an overlap with build source files, the same class
59
+ * could be detected as ambiguous.
60
+ */
61
+ private _hasBeenReportedAsAmbiguous;
55
62
  }
56
63
  export {};
@@ -47,12 +47,23 @@
47
47
  */
48
48
  const AMBIGUOUS_LIFECYCLE_HOOKS = new Set(['ngOnDestroy']);
49
49
  /** Describes how a given class is used in the context of Angular. */
50
- var ClassKind;
51
- (function (ClassKind) {
52
- ClassKind[ClassKind["DIRECTIVE"] = 0] = "DIRECTIVE";
53
- ClassKind[ClassKind["AMBIGUOUS"] = 1] = "AMBIGUOUS";
54
- ClassKind[ClassKind["UNKNOWN"] = 2] = "UNKNOWN";
55
- })(ClassKind || (ClassKind = {}));
50
+ var InferredKind;
51
+ (function (InferredKind) {
52
+ InferredKind[InferredKind["DIRECTIVE"] = 0] = "DIRECTIVE";
53
+ InferredKind[InferredKind["AMBIGUOUS"] = 1] = "AMBIGUOUS";
54
+ InferredKind[InferredKind["UNKNOWN"] = 2] = "UNKNOWN";
55
+ })(InferredKind || (InferredKind = {}));
56
+ /** Describes possible types of Angular declarations. */
57
+ var DeclarationType;
58
+ (function (DeclarationType) {
59
+ DeclarationType[DeclarationType["DIRECTIVE"] = 0] = "DIRECTIVE";
60
+ DeclarationType[DeclarationType["COMPONENT"] = 1] = "COMPONENT";
61
+ DeclarationType[DeclarationType["ABSTRACT_DIRECTIVE"] = 2] = "ABSTRACT_DIRECTIVE";
62
+ DeclarationType[DeclarationType["PIPE"] = 3] = "PIPE";
63
+ DeclarationType[DeclarationType["INJECTABLE"] = 4] = "INJECTABLE";
64
+ })(DeclarationType || (DeclarationType = {}));
65
+ /** TODO message that is added to ambiguous classes using Angular features. */
66
+ const AMBIGUOUS_CLASS_TODO = 'Add Angular decorator.';
56
67
  class UndecoratedClassesWithDecoratedFieldsTransform {
57
68
  constructor(typeChecker, getUpdateRecorder) {
58
69
  this.typeChecker = typeChecker;
@@ -69,8 +80,8 @@
69
80
  * indicating that a given class uses Angular features. https://hackmd.io/vuQfavzfRG6KUCtU7oK_EA
70
81
  */
71
82
  migrate(sourceFiles) {
72
- const { result, ambiguous } = this._findUndecoratedAbstractDirectives(sourceFiles);
73
- result.forEach(node => {
83
+ const { detectedAbstractDirectives, ambiguousClasses } = this._findUndecoratedAbstractDirectives(sourceFiles);
84
+ detectedAbstractDirectives.forEach(node => {
74
85
  const sourceFile = node.getSourceFile();
75
86
  const recorder = this.getUpdateRecorder(sourceFile);
76
87
  const directiveExpr = this.importManager.addImportToSourceFile(sourceFile, 'Directive', '@angular/core');
@@ -81,11 +92,17 @@
81
92
  // determine whether the class is used as directive, service or pipe. The migration
82
93
  // could potentially determine the type by checking NgModule definitions or inheritance
83
94
  // of other known declarations, but this is out of scope and a TODO/failure is sufficient.
84
- return Array.from(ambiguous).reduce((failures, node) => {
95
+ return Array.from(ambiguousClasses).reduce((failures, node) => {
96
+ // If the class has been reported as ambiguous before, skip adding a TODO and
97
+ // printing an error. A class could be visited multiple times when it's part
98
+ // of multiple build targets in the CLI project.
99
+ if (this._hasBeenReportedAsAmbiguous(node)) {
100
+ return failures;
101
+ }
85
102
  const sourceFile = node.getSourceFile();
86
103
  const recorder = this.getUpdateRecorder(sourceFile);
87
104
  // Add a TODO to the class that uses Angular features but is not decorated.
88
- recorder.addClassTodo(node, `Add Angular decorator.`);
105
+ recorder.addClassTodo(node, AMBIGUOUS_CLASS_TODO);
89
106
  // Add an error for the class that will be printed in the `ng update` output.
90
107
  return failures.concat({
91
108
  node,
@@ -104,58 +121,78 @@
104
121
  * directives. Those are ambiguous and could be either Directive, Pipe or service.
105
122
  */
106
123
  _findUndecoratedAbstractDirectives(sourceFiles) {
107
- const result = new Set();
124
+ const ambiguousClasses = new Set();
125
+ const declarations = new WeakMap();
126
+ const detectedAbstractDirectives = new Set();
108
127
  const undecoratedClasses = new Set();
109
- const nonAbstractDirectives = new WeakSet();
110
- const abstractDirectives = new WeakSet();
111
- const ambiguous = new Set();
112
128
  const visitNode = (node) => {
113
129
  node.forEachChild(visitNode);
114
130
  if (!ts.isClassDeclaration(node)) {
115
131
  return;
116
132
  }
117
- const { isDirectiveOrComponent, isAbstractDirective, kind } = this._analyzeClassDeclaration(node);
118
- if (isDirectiveOrComponent) {
119
- if (isAbstractDirective) {
120
- abstractDirectives.add(node);
121
- }
122
- else {
123
- nonAbstractDirectives.add(node);
124
- }
133
+ const { inferredKind, decoratedType } = this._analyzeClassDeclaration(node);
134
+ if (decoratedType !== null) {
135
+ declarations.set(node, decoratedType);
136
+ return;
137
+ }
138
+ if (inferredKind === InferredKind.DIRECTIVE) {
139
+ detectedAbstractDirectives.add(node);
125
140
  }
126
- else if (kind === ClassKind.DIRECTIVE) {
127
- abstractDirectives.add(node);
128
- result.add(node);
141
+ else if (inferredKind === InferredKind.AMBIGUOUS) {
142
+ ambiguousClasses.add(node);
129
143
  }
130
144
  else {
131
- if (kind === ClassKind.AMBIGUOUS) {
132
- ambiguous.add(node);
133
- }
134
145
  undecoratedClasses.add(node);
135
146
  }
136
147
  };
137
148
  sourceFiles.forEach(sourceFile => sourceFile.forEachChild(visitNode));
138
- // We collected all undecorated class declarations which inherit from abstract directives.
139
- // For such abstract directives, the derived classes also need to be migrated.
140
- undecoratedClasses.forEach(node => {
141
- for (const { node: baseClass } of find_base_classes_1.findBaseClassDeclarations(node, this.typeChecker)) {
142
- // If the undecorated class inherits from a non-abstract directive, skip the current
143
- // class. We do this because undecorated classes which inherit metadata from non-abstract
144
- // directives are handled in the `undecorated-classes-with-di` migration that copies
145
- // inherited metadata into an explicit decorator.
146
- if (nonAbstractDirectives.has(baseClass)) {
149
+ /**
150
+ * Checks the inheritance of the given set of classes. It removes classes from the
151
+ * detected abstract directives set when they inherit from a non-abstract Angular
152
+ * declaration. e.g. an abstract directive can never extend from a component.
153
+ *
154
+ * If a class inherits from an abstract directive though, we will migrate them too
155
+ * as derived classes also need to be decorated. This has been done for a simpler mental
156
+ * model and reduced complexity in the Angular compiler. See migration plan document.
157
+ */
158
+ const checkInheritanceOfClasses = (classes) => {
159
+ classes.forEach(node => {
160
+ for (const { node: baseClass } of find_base_classes_1.findBaseClassDeclarations(node, this.typeChecker)) {
161
+ if (!declarations.has(baseClass)) {
162
+ continue;
163
+ }
164
+ // If the undecorated class inherits from an abstract directive, always migrate it.
165
+ // Derived undecorated classes of abstract directives are always also considered
166
+ // abstract directives and need to be decorated too. This is necessary as otherwise
167
+ // the inheritance chain cannot be resolved by the Angular compiler. e.g. when it
168
+ // flattens directive metadata for type checking. In the other case, we never want
169
+ // to migrate a class if it extends from a non-abstract Angular declaration. That
170
+ // is an unsupported pattern as of v9 and was previously handled with the
171
+ // `undecorated-classes-with-di` migration (which copied the inherited decorator).
172
+ if (declarations.get(baseClass) === DeclarationType.ABSTRACT_DIRECTIVE) {
173
+ detectedAbstractDirectives.add(node);
174
+ }
175
+ else {
176
+ detectedAbstractDirectives.delete(node);
177
+ }
178
+ ambiguousClasses.delete(node);
147
179
  break;
148
180
  }
149
- else if (abstractDirectives.has(baseClass)) {
150
- result.add(node);
151
- // In case the undecorated class previously could not be detected as directive,
152
- // remove it from the ambiguous set as we now know that it's a guaranteed directive.
153
- ambiguous.delete(node);
154
- break;
155
- }
156
- }
157
- });
158
- return { result, ambiguous };
181
+ });
182
+ };
183
+ // Check inheritance of any detected abstract directive. We want to remove
184
+ // classes that are not eligible abstract directives due to inheritance. i.e.
185
+ // if a class extends from a component, it cannot be a derived abstract directive.
186
+ checkInheritanceOfClasses(detectedAbstractDirectives);
187
+ // Update the class declarations to reflect the detected abstract directives. This is
188
+ // then used later when we check for undecorated classes that inherit from an abstract
189
+ // directive and need to be decorated.
190
+ detectedAbstractDirectives.forEach(n => declarations.set(n, DeclarationType.ABSTRACT_DIRECTIVE));
191
+ // Check ambiguous and undecorated classes if they inherit from an abstract directive.
192
+ // If they do, we want to migrate them too. See function definition for more details.
193
+ checkInheritanceOfClasses(ambiguousClasses);
194
+ checkInheritanceOfClasses(undecoratedClasses);
195
+ return { detectedAbstractDirectives, ambiguousClasses };
159
196
  }
160
197
  /**
161
198
  * Analyzes the given class declaration by determining whether the class
@@ -163,18 +200,32 @@
163
200
  */
164
201
  _analyzeClassDeclaration(node) {
165
202
  const ngDecorators = node.decorators && ng_decorators_1.getAngularDecorators(this.typeChecker, node.decorators);
166
- const kind = this._determineClassKind(node);
203
+ const inferredKind = this._determineClassKind(node);
167
204
  if (ngDecorators === undefined || ngDecorators.length === 0) {
168
- return { isDirectiveOrComponent: false, isAbstractDirective: false, kind };
205
+ return { decoratedType: null, inferredKind };
169
206
  }
170
207
  const directiveDecorator = ngDecorators.find(({ name }) => name === 'Directive');
171
208
  const componentDecorator = ngDecorators.find(({ name }) => name === 'Component');
209
+ const pipeDecorator = ngDecorators.find(({ name }) => name === 'Pipe');
210
+ const injectableDecorator = ngDecorators.find(({ name }) => name === 'Injectable');
172
211
  const isAbstractDirective = directiveDecorator !== undefined && this._isAbstractDirective(directiveDecorator);
173
- return {
174
- isDirectiveOrComponent: !!directiveDecorator || !!componentDecorator,
175
- isAbstractDirective,
176
- kind,
177
- };
212
+ let decoratedType = null;
213
+ if (isAbstractDirective) {
214
+ decoratedType = DeclarationType.ABSTRACT_DIRECTIVE;
215
+ }
216
+ else if (componentDecorator !== undefined) {
217
+ decoratedType = DeclarationType.COMPONENT;
218
+ }
219
+ else if (directiveDecorator !== undefined) {
220
+ decoratedType = DeclarationType.DIRECTIVE;
221
+ }
222
+ else if (pipeDecorator !== undefined) {
223
+ decoratedType = DeclarationType.PIPE;
224
+ }
225
+ else if (injectableDecorator !== undefined) {
226
+ decoratedType = DeclarationType.INJECTABLE;
227
+ }
228
+ return { decoratedType, inferredKind };
178
229
  }
179
230
  /**
180
231
  * Checks whether the given decorator resolves to an abstract directive. An directive is
@@ -203,32 +254,47 @@
203
254
  * considered Angular features..
204
255
  */
205
256
  _determineClassKind(node) {
206
- let usage = ClassKind.UNKNOWN;
257
+ let usage = InferredKind.UNKNOWN;
207
258
  for (const member of node.members) {
208
259
  const propertyName = member.name !== undefined ? property_name_1.getPropertyNameText(member.name) : null;
209
260
  // If the class declares any of the known directive lifecycle hooks, we can
210
261
  // immediately exit the loop as the class is guaranteed to be a directive.
211
262
  if (propertyName !== null && DIRECTIVE_LIFECYCLE_HOOKS.has(propertyName)) {
212
- return ClassKind.DIRECTIVE;
263
+ return InferredKind.DIRECTIVE;
213
264
  }
214
265
  const ngDecorators = member.decorators !== undefined ?
215
266
  ng_decorators_1.getAngularDecorators(this.typeChecker, member.decorators) :
216
267
  [];
217
268
  for (const { name } of ngDecorators) {
218
269
  if (DIRECTIVE_FIELD_DECORATORS.has(name)) {
219
- return ClassKind.DIRECTIVE;
270
+ return InferredKind.DIRECTIVE;
220
271
  }
221
272
  }
222
273
  // If the class declares any of the lifecycle hooks that do not guarantee that
223
274
  // the given class is a directive, update the kind and continue looking for other
224
275
  // members that would unveil a more specific kind (i.e. being a directive).
225
276
  if (propertyName !== null && AMBIGUOUS_LIFECYCLE_HOOKS.has(propertyName)) {
226
- usage = ClassKind.AMBIGUOUS;
277
+ usage = InferredKind.AMBIGUOUS;
227
278
  }
228
279
  }
229
280
  return usage;
230
281
  }
282
+ /**
283
+ * Checks whether a given class has been reported as ambiguous in previous
284
+ * migration run. e.g. when build targets are migrated first, and then test
285
+ * targets that have an overlap with build source files, the same class
286
+ * could be detected as ambiguous.
287
+ */
288
+ _hasBeenReportedAsAmbiguous(node) {
289
+ const sourceFile = node.getSourceFile();
290
+ const leadingComments = ts.getLeadingCommentRanges(sourceFile.text, node.pos);
291
+ if (leadingComments === undefined) {
292
+ return false;
293
+ }
294
+ return leadingComments.some(({ kind, pos, end }) => kind === ts.SyntaxKind.SingleLineCommentTrivia &&
295
+ sourceFile.text.substring(pos, end).includes(`TODO: ${AMBIGUOUS_CLASS_TODO}`));
296
+ }
231
297
  }
232
298
  exports.UndecoratedClassesWithDecoratedFieldsTransform = UndecoratedClassesWithDecoratedFieldsTransform;
233
299
  });
234
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transform.js","sourceRoot":"","sources":["../../../../../../../../packages/core/schematics/migrations/undecorated-classes-with-decorated-fields/transform.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 {PartialEvaluator} from '@angular/compiler-cli/src/ngtsc/partial_evaluator';\nimport {reflectObjectLiteral, TypeScriptReflectionHost} from '@angular/compiler-cli/src/ngtsc/reflection';\nimport * as ts from 'typescript';\n\nimport {ImportManager} from '../../utils/import_manager';\nimport {getAngularDecorators, NgDecorator} from '../../utils/ng_decorators';\nimport {findBaseClassDeclarations} from '../../utils/typescript/find_base_classes';\nimport {unwrapExpression} from '../../utils/typescript/functions';\nimport {getPropertyNameText} from '../../utils/typescript/property_name';\n\nimport {UpdateRecorder} from './update_recorder';\n\n/**\n * Set of known decorators that indicate that the current class needs a directive\n * definition. These decorators are always specific to directives.\n */\nconst DIRECTIVE_FIELD_DECORATORS = new Set([\n  'Input', 'Output', 'ViewChild', 'ViewChildren', 'ContentChild', 'ContentChildren', 'HostBinding',\n  'HostListener'\n]);\n\n/**\n * Set of known lifecycle hooks that indicate that the current class needs a directive\n * definition. These lifecycle hooks are always specific to directives.\n */\nconst DIRECTIVE_LIFECYCLE_HOOKS = new Set([\n  'ngOnChanges', 'ngOnInit', 'ngDoCheck', 'ngAfterViewInit', 'ngAfterViewChecked',\n  'ngAfterContentInit', 'ngAfterContentChecked'\n]);\n\n/**\n * Set of known lifecycle hooks that indicate that a given class uses Angular\n * features, but it's ambiguous whether it is a directive or service.\n */\nconst AMBIGUOUS_LIFECYCLE_HOOKS = new Set(['ngOnDestroy']);\n\n/** Describes how a given class is used in the context of Angular. */\nenum ClassKind {\n  DIRECTIVE,\n  AMBIGUOUS,\n  UNKNOWN,\n}\n\n/** Analyzed class declaration. */\ninterface AnalyzedClass {\n  /** Whether the class is decorated with @Directive or @Component. */\n  isDirectiveOrComponent: boolean;\n  /** Whether the class is an abstract directive. */\n  isAbstractDirective: boolean;\n  /** Kind of the given class in terms of Angular. */\n  kind: ClassKind;\n}\n\ninterface AnalysisFailure {\n  node: ts.Node;\n  message: string;\n}\n\nexport class UndecoratedClassesWithDecoratedFieldsTransform {\n  private printer = ts.createPrinter();\n  private importManager = new ImportManager(this.getUpdateRecorder, this.printer);\n  private reflectionHost = new TypeScriptReflectionHost(this.typeChecker);\n  private partialEvaluator = new PartialEvaluator(this.reflectionHost, this.typeChecker, null);\n\n  constructor(\n      private typeChecker: ts.TypeChecker,\n      private getUpdateRecorder: (sf: ts.SourceFile) => UpdateRecorder) {}\n\n  /**\n   * Migrates the specified source files. The transform adds the abstract `@Directive`\n   * decorator to undecorated classes that use Angular features. Class members which\n   * are decorated with any Angular decorator, or class members for lifecycle hooks are\n   * indicating that a given class uses Angular features. https://hackmd.io/vuQfavzfRG6KUCtU7oK_EA\n   */\n  migrate(sourceFiles: ts.SourceFile[]): AnalysisFailure[] {\n    const {result, ambiguous} = this._findUndecoratedAbstractDirectives(sourceFiles);\n\n\n    result.forEach(node => {\n      const sourceFile = node.getSourceFile();\n      const recorder = this.getUpdateRecorder(sourceFile);\n      const directiveExpr =\n          this.importManager.addImportToSourceFile(sourceFile, 'Directive', '@angular/core');\n      const decoratorExpr = ts.createDecorator(ts.createCall(directiveExpr, undefined, undefined));\n      recorder.addClassDecorator(\n          node, this.printer.printNode(ts.EmitHint.Unspecified, decoratorExpr, sourceFile));\n    });\n\n    // Ambiguous classes clearly use Angular features, but the migration is unable to\n    // determine whether the class is used as directive, service or pipe. The migration\n    // could potentially determine the type by checking NgModule definitions or inheritance\n    // of other known declarations, but this is out of scope and a TODO/failure is sufficient.\n    return Array.from(ambiguous).reduce((failures, node) => {\n      const sourceFile = node.getSourceFile();\n      const recorder = this.getUpdateRecorder(sourceFile);\n\n      // Add a TODO to the class that uses Angular features but is not decorated.\n      recorder.addClassTodo(node, `Add Angular decorator.`);\n\n      // Add an error for the class that will be printed in the `ng update` output.\n      return failures.concat({\n        node,\n        message: 'Class uses Angular features but cannot be migrated automatically. Please ' +\n            'add an appropriate Angular decorator.'\n      });\n    }, [] as AnalysisFailure[]);\n  }\n\n  /** Records all changes that were made in the import manager. */\n  recordChanges() {\n    this.importManager.recordChanges();\n  }\n\n  /**\n   * Finds undecorated abstract directives in the specified source files. Also returns\n   * a set of undecorated classes which could not be detected as guaranteed abstract\n   * directives. Those are ambiguous and could be either Directive, Pipe or service.\n   */\n  private _findUndecoratedAbstractDirectives(sourceFiles: ts.SourceFile[]) {\n    const result = new Set<ts.ClassDeclaration>();\n    const undecoratedClasses = new Set<ts.ClassDeclaration>();\n    const nonAbstractDirectives = new WeakSet<ts.ClassDeclaration>();\n    const abstractDirectives = new WeakSet<ts.ClassDeclaration>();\n    const ambiguous = new Set<ts.ClassDeclaration>();\n\n    const visitNode = (node: ts.Node) => {\n      node.forEachChild(visitNode);\n      if (!ts.isClassDeclaration(node)) {\n        return;\n      }\n      const {isDirectiveOrComponent, isAbstractDirective, kind} =\n          this._analyzeClassDeclaration(node);\n      if (isDirectiveOrComponent) {\n        if (isAbstractDirective) {\n          abstractDirectives.add(node);\n        } else {\n          nonAbstractDirectives.add(node);\n        }\n      } else if (kind === ClassKind.DIRECTIVE) {\n        abstractDirectives.add(node);\n        result.add(node);\n      } else {\n        if (kind === ClassKind.AMBIGUOUS) {\n          ambiguous.add(node);\n        }\n        undecoratedClasses.add(node);\n      }\n    };\n\n    sourceFiles.forEach(sourceFile => sourceFile.forEachChild(visitNode));\n\n    // We collected all undecorated class declarations which inherit from abstract directives.\n    // For such abstract directives, the derived classes also need to be migrated.\n    undecoratedClasses.forEach(node => {\n      for (const {node: baseClass} of findBaseClassDeclarations(node, this.typeChecker)) {\n        // If the undecorated class inherits from a non-abstract directive, skip the current\n        // class. We do this because undecorated classes which inherit metadata from non-abstract\n        // directives are handled in the `undecorated-classes-with-di` migration that copies\n        // inherited metadata into an explicit decorator.\n        if (nonAbstractDirectives.has(baseClass)) {\n          break;\n        } else if (abstractDirectives.has(baseClass)) {\n          result.add(node);\n          // In case the undecorated class previously could not be detected as directive,\n          // remove it from the ambiguous set as we now know that it's a guaranteed directive.\n          ambiguous.delete(node);\n          break;\n        }\n      }\n    });\n\n    return {result, ambiguous};\n  }\n\n  /**\n   * Analyzes the given class declaration by determining whether the class\n   * is a directive, is an abstract directive, or uses Angular features.\n   */\n  private _analyzeClassDeclaration(node: ts.ClassDeclaration): AnalyzedClass {\n    const ngDecorators = node.decorators && getAngularDecorators(this.typeChecker, node.decorators);\n    const kind = this._determineClassKind(node);\n    if (ngDecorators === undefined || ngDecorators.length === 0) {\n      return {isDirectiveOrComponent: false, isAbstractDirective: false, kind};\n    }\n    const directiveDecorator = ngDecorators.find(({name}) => name === 'Directive');\n    const componentDecorator = ngDecorators.find(({name}) => name === 'Component');\n    const isAbstractDirective =\n        directiveDecorator !== undefined && this._isAbstractDirective(directiveDecorator);\n    return {\n      isDirectiveOrComponent: !!directiveDecorator || !!componentDecorator,\n      isAbstractDirective,\n      kind,\n    };\n  }\n\n  /**\n   * Checks whether the given decorator resolves to an abstract directive. An directive is\n   * considered \"abstract\" if there is no selector specified.\n   */\n  private _isAbstractDirective({node}: NgDecorator): boolean {\n    const metadataArgs = node.expression.arguments;\n    if (metadataArgs.length === 0) {\n      return true;\n    }\n    const metadataExpr = unwrapExpression(metadataArgs[0]);\n    if (!ts.isObjectLiteralExpression(metadataExpr)) {\n      return false;\n    }\n    const metadata = reflectObjectLiteral(metadataExpr);\n    if (!metadata.has('selector')) {\n      return false;\n    }\n    const selector = this.partialEvaluator.evaluate(metadata.get('selector')!);\n    return selector == null;\n  }\n\n  /**\n   * Determines the kind of a given class in terms of Angular. The method checks\n   * whether the given class has members that indicate the use of Angular features.\n   * e.g. lifecycle hooks or decorated members like `@Input` or `@Output` are\n   * considered Angular features..\n   */\n  private _determineClassKind(node: ts.ClassDeclaration): ClassKind {\n    let usage = ClassKind.UNKNOWN;\n\n    for (const member of node.members) {\n      const propertyName = member.name !== undefined ? getPropertyNameText(member.name) : null;\n\n      // If the class declares any of the known directive lifecycle hooks, we can\n      // immediately exit the loop as the class is guaranteed to be a directive.\n      if (propertyName !== null && DIRECTIVE_LIFECYCLE_HOOKS.has(propertyName)) {\n        return ClassKind.DIRECTIVE;\n      }\n\n      const ngDecorators = member.decorators !== undefined ?\n          getAngularDecorators(this.typeChecker, member.decorators) :\n          [];\n      for (const {name} of ngDecorators) {\n        if (DIRECTIVE_FIELD_DECORATORS.has(name)) {\n          return ClassKind.DIRECTIVE;\n        }\n      }\n\n      // If the class declares any of the lifecycle hooks that do not guarantee that\n      // the given class is a directive, update the kind and continue looking for other\n      // members that would unveil a more specific kind (i.e. being a directive).\n      if (propertyName !== null && AMBIGUOUS_LIFECYCLE_HOOKS.has(propertyName)) {\n        usage = ClassKind.AMBIGUOUS;\n      }\n    }\n\n    return usage;\n  }\n}\n"]}
300
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transform.js","sourceRoot":"","sources":["../../../../../../../../packages/core/schematics/migrations/undecorated-classes-with-decorated-fields/transform.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 {PartialEvaluator} from '@angular/compiler-cli/src/ngtsc/partial_evaluator';\nimport {reflectObjectLiteral, TypeScriptReflectionHost} from '@angular/compiler-cli/src/ngtsc/reflection';\nimport * as ts from 'typescript';\n\nimport {ImportManager} from '../../utils/import_manager';\nimport {getAngularDecorators, NgDecorator} from '../../utils/ng_decorators';\nimport {findBaseClassDeclarations} from '../../utils/typescript/find_base_classes';\nimport {unwrapExpression} from '../../utils/typescript/functions';\nimport {getPropertyNameText} from '../../utils/typescript/property_name';\n\nimport {UpdateRecorder} from './update_recorder';\n\n/**\n * Set of known decorators that indicate that the current class needs a directive\n * definition. These decorators are always specific to directives.\n */\nconst DIRECTIVE_FIELD_DECORATORS = new Set([\n  'Input', 'Output', 'ViewChild', 'ViewChildren', 'ContentChild', 'ContentChildren', 'HostBinding',\n  'HostListener'\n]);\n\n/**\n * Set of known lifecycle hooks that indicate that the current class needs a directive\n * definition. These lifecycle hooks are always specific to directives.\n */\nconst DIRECTIVE_LIFECYCLE_HOOKS = new Set([\n  'ngOnChanges', 'ngOnInit', 'ngDoCheck', 'ngAfterViewInit', 'ngAfterViewChecked',\n  'ngAfterContentInit', 'ngAfterContentChecked'\n]);\n\n/**\n * Set of known lifecycle hooks that indicate that a given class uses Angular\n * features, but it's ambiguous whether it is a directive or service.\n */\nconst AMBIGUOUS_LIFECYCLE_HOOKS = new Set(['ngOnDestroy']);\n\n/** Describes how a given class is used in the context of Angular. */\nenum InferredKind {\n  DIRECTIVE,\n  AMBIGUOUS,\n  UNKNOWN,\n}\n\n/** Describes possible types of Angular declarations. */\nenum DeclarationType {\n  DIRECTIVE,\n  COMPONENT,\n  ABSTRACT_DIRECTIVE,\n  PIPE,\n  INJECTABLE,\n}\n\n/** Analyzed class declaration. */\ninterface AnalyzedClass {\n  /** Type of declaration that is determined through an applied decorator. */\n  decoratedType: DeclarationType|null;\n  /** Inferred class kind in terms of Angular. */\n  inferredKind: InferredKind;\n}\n\ninterface AnalysisFailure {\n  node: ts.Node;\n  message: string;\n}\n\n/** TODO message that is added to ambiguous classes using Angular features. */\nconst AMBIGUOUS_CLASS_TODO = 'Add Angular decorator.';\n\nexport class UndecoratedClassesWithDecoratedFieldsTransform {\n  private printer = ts.createPrinter();\n  private importManager = new ImportManager(this.getUpdateRecorder, this.printer);\n  private reflectionHost = new TypeScriptReflectionHost(this.typeChecker);\n  private partialEvaluator = new PartialEvaluator(this.reflectionHost, this.typeChecker, null);\n\n  constructor(\n      private typeChecker: ts.TypeChecker,\n      private getUpdateRecorder: (sf: ts.SourceFile) => UpdateRecorder) {}\n\n  /**\n   * Migrates the specified source files. The transform adds the abstract `@Directive`\n   * decorator to undecorated classes that use Angular features. Class members which\n   * are decorated with any Angular decorator, or class members for lifecycle hooks are\n   * indicating that a given class uses Angular features. https://hackmd.io/vuQfavzfRG6KUCtU7oK_EA\n   */\n  migrate(sourceFiles: ts.SourceFile[]): AnalysisFailure[] {\n    const {detectedAbstractDirectives, ambiguousClasses} =\n        this._findUndecoratedAbstractDirectives(sourceFiles);\n\n    detectedAbstractDirectives.forEach(node => {\n      const sourceFile = node.getSourceFile();\n      const recorder = this.getUpdateRecorder(sourceFile);\n      const directiveExpr =\n          this.importManager.addImportToSourceFile(sourceFile, 'Directive', '@angular/core');\n      const decoratorExpr = ts.createDecorator(ts.createCall(directiveExpr, undefined, undefined));\n      recorder.addClassDecorator(\n          node, this.printer.printNode(ts.EmitHint.Unspecified, decoratorExpr, sourceFile));\n    });\n\n    // Ambiguous classes clearly use Angular features, but the migration is unable to\n    // determine whether the class is used as directive, service or pipe. The migration\n    // could potentially determine the type by checking NgModule definitions or inheritance\n    // of other known declarations, but this is out of scope and a TODO/failure is sufficient.\n    return Array.from(ambiguousClasses).reduce((failures, node) => {\n      // If the class has been reported as ambiguous before, skip adding a TODO and\n      // printing an error. A class could be visited multiple times when it's part\n      // of multiple build targets in the CLI project.\n      if (this._hasBeenReportedAsAmbiguous(node)) {\n        return failures;\n      }\n\n      const sourceFile = node.getSourceFile();\n      const recorder = this.getUpdateRecorder(sourceFile);\n\n      // Add a TODO to the class that uses Angular features but is not decorated.\n      recorder.addClassTodo(node, AMBIGUOUS_CLASS_TODO);\n\n      // Add an error for the class that will be printed in the `ng update` output.\n      return failures.concat({\n        node,\n        message: 'Class uses Angular features but cannot be migrated automatically. Please ' +\n            'add an appropriate Angular decorator.'\n      });\n    }, [] as AnalysisFailure[]);\n  }\n\n  /** Records all changes that were made in the import manager. */\n  recordChanges() {\n    this.importManager.recordChanges();\n  }\n\n  /**\n   * Finds undecorated abstract directives in the specified source files. Also returns\n   * a set of undecorated classes which could not be detected as guaranteed abstract\n   * directives. Those are ambiguous and could be either Directive, Pipe or service.\n   */\n  private _findUndecoratedAbstractDirectives(sourceFiles: ts.SourceFile[]) {\n    const ambiguousClasses = new Set<ts.ClassDeclaration>();\n    const declarations = new WeakMap<ts.ClassDeclaration, DeclarationType>();\n    const detectedAbstractDirectives = new Set<ts.ClassDeclaration>();\n    const undecoratedClasses = new Set<ts.ClassDeclaration>();\n\n    const visitNode = (node: ts.Node) => {\n      node.forEachChild(visitNode);\n      if (!ts.isClassDeclaration(node)) {\n        return;\n      }\n      const {inferredKind, decoratedType} = this._analyzeClassDeclaration(node);\n\n      if (decoratedType !== null) {\n        declarations.set(node, decoratedType);\n        return;\n      }\n\n      if (inferredKind === InferredKind.DIRECTIVE) {\n        detectedAbstractDirectives.add(node);\n      } else if (inferredKind === InferredKind.AMBIGUOUS) {\n        ambiguousClasses.add(node);\n      } else {\n        undecoratedClasses.add(node);\n      }\n    };\n\n    sourceFiles.forEach(sourceFile => sourceFile.forEachChild(visitNode));\n\n    /**\n     * Checks the inheritance of the given set of classes. It removes classes from the\n     * detected abstract directives set when they inherit from a non-abstract Angular\n     * declaration. e.g. an abstract directive can never extend from a component.\n     *\n     * If a class inherits from an abstract directive though, we will migrate them too\n     * as derived classes also need to be decorated. This has been done for a simpler mental\n     * model and reduced complexity in the Angular compiler. See migration plan document.\n     */\n    const checkInheritanceOfClasses = (classes: Set<ts.ClassDeclaration>) => {\n      classes.forEach(node => {\n        for (const {node: baseClass} of findBaseClassDeclarations(node, this.typeChecker)) {\n          if (!declarations.has(baseClass)) {\n            continue;\n          }\n          // If the undecorated class inherits from an abstract directive, always migrate it.\n          // Derived undecorated classes of abstract directives are always also considered\n          // abstract directives and need to be decorated too. This is necessary as otherwise\n          // the inheritance chain cannot be resolved by the Angular compiler. e.g. when it\n          // flattens directive metadata for type checking. In the other case, we never want\n          // to migrate a class if it extends from a non-abstract Angular declaration. That\n          // is an unsupported pattern as of v9 and was previously handled with the\n          // `undecorated-classes-with-di` migration (which copied the inherited decorator).\n          if (declarations.get(baseClass) === DeclarationType.ABSTRACT_DIRECTIVE) {\n            detectedAbstractDirectives.add(node);\n          } else {\n            detectedAbstractDirectives.delete(node);\n          }\n          ambiguousClasses.delete(node);\n          break;\n        }\n      });\n    };\n\n    // Check inheritance of any detected abstract directive. We want to remove\n    // classes that are not eligible abstract directives due to inheritance. i.e.\n    // if a class extends from a component, it cannot be a derived abstract directive.\n    checkInheritanceOfClasses(detectedAbstractDirectives);\n    // Update the class declarations to reflect the detected abstract directives. This is\n    // then used later when we check for undecorated classes that inherit from an abstract\n    // directive and need to be decorated.\n    detectedAbstractDirectives.forEach(\n        n => declarations.set(n, DeclarationType.ABSTRACT_DIRECTIVE));\n    // Check ambiguous and undecorated classes if they inherit from an abstract directive.\n    // If they do, we want to migrate them too. See function definition for more details.\n    checkInheritanceOfClasses(ambiguousClasses);\n    checkInheritanceOfClasses(undecoratedClasses);\n\n    return {detectedAbstractDirectives, ambiguousClasses};\n  }\n\n  /**\n   * Analyzes the given class declaration by determining whether the class\n   * is a directive, is an abstract directive, or uses Angular features.\n   */\n  private _analyzeClassDeclaration(node: ts.ClassDeclaration): AnalyzedClass {\n    const ngDecorators = node.decorators && getAngularDecorators(this.typeChecker, node.decorators);\n    const inferredKind = this._determineClassKind(node);\n    if (ngDecorators === undefined || ngDecorators.length === 0) {\n      return {decoratedType: null, inferredKind};\n    }\n    const directiveDecorator = ngDecorators.find(({name}) => name === 'Directive');\n    const componentDecorator = ngDecorators.find(({name}) => name === 'Component');\n    const pipeDecorator = ngDecorators.find(({name}) => name === 'Pipe');\n    const injectableDecorator = ngDecorators.find(({name}) => name === 'Injectable');\n    const isAbstractDirective =\n        directiveDecorator !== undefined && this._isAbstractDirective(directiveDecorator);\n\n    let decoratedType: DeclarationType|null = null;\n    if (isAbstractDirective) {\n      decoratedType = DeclarationType.ABSTRACT_DIRECTIVE;\n    } else if (componentDecorator !== undefined) {\n      decoratedType = DeclarationType.COMPONENT;\n    } else if (directiveDecorator !== undefined) {\n      decoratedType = DeclarationType.DIRECTIVE;\n    } else if (pipeDecorator !== undefined) {\n      decoratedType = DeclarationType.PIPE;\n    } else if (injectableDecorator !== undefined) {\n      decoratedType = DeclarationType.INJECTABLE;\n    }\n    return {decoratedType, inferredKind};\n  }\n\n  /**\n   * Checks whether the given decorator resolves to an abstract directive. An directive is\n   * considered \"abstract\" if there is no selector specified.\n   */\n  private _isAbstractDirective({node}: NgDecorator): boolean {\n    const metadataArgs = node.expression.arguments;\n    if (metadataArgs.length === 0) {\n      return true;\n    }\n    const metadataExpr = unwrapExpression(metadataArgs[0]);\n    if (!ts.isObjectLiteralExpression(metadataExpr)) {\n      return false;\n    }\n    const metadata = reflectObjectLiteral(metadataExpr);\n    if (!metadata.has('selector')) {\n      return false;\n    }\n    const selector = this.partialEvaluator.evaluate(metadata.get('selector')!);\n    return selector == null;\n  }\n\n  /**\n   * Determines the kind of a given class in terms of Angular. The method checks\n   * whether the given class has members that indicate the use of Angular features.\n   * e.g. lifecycle hooks or decorated members like `@Input` or `@Output` are\n   * considered Angular features..\n   */\n  private _determineClassKind(node: ts.ClassDeclaration): InferredKind {\n    let usage = InferredKind.UNKNOWN;\n\n    for (const member of node.members) {\n      const propertyName = member.name !== undefined ? getPropertyNameText(member.name) : null;\n\n      // If the class declares any of the known directive lifecycle hooks, we can\n      // immediately exit the loop as the class is guaranteed to be a directive.\n      if (propertyName !== null && DIRECTIVE_LIFECYCLE_HOOKS.has(propertyName)) {\n        return InferredKind.DIRECTIVE;\n      }\n\n      const ngDecorators = member.decorators !== undefined ?\n          getAngularDecorators(this.typeChecker, member.decorators) :\n          [];\n      for (const {name} of ngDecorators) {\n        if (DIRECTIVE_FIELD_DECORATORS.has(name)) {\n          return InferredKind.DIRECTIVE;\n        }\n      }\n\n      // If the class declares any of the lifecycle hooks that do not guarantee that\n      // the given class is a directive, update the kind and continue looking for other\n      // members that would unveil a more specific kind (i.e. being a directive).\n      if (propertyName !== null && AMBIGUOUS_LIFECYCLE_HOOKS.has(propertyName)) {\n        usage = InferredKind.AMBIGUOUS;\n      }\n    }\n\n    return usage;\n  }\n\n  /**\n   * Checks whether a given class has been reported as ambiguous in previous\n   * migration run. e.g. when build targets are migrated first, and then test\n   * targets that have an overlap with build source files, the same class\n   * could be detected as ambiguous.\n   */\n  private _hasBeenReportedAsAmbiguous(node: ts.ClassDeclaration): boolean {\n    const sourceFile = node.getSourceFile();\n    const leadingComments = ts.getLeadingCommentRanges(sourceFile.text, node.pos);\n    if (leadingComments === undefined) {\n      return false;\n    }\n    return leadingComments.some(\n        ({kind, pos, end}) => kind === ts.SyntaxKind.SingleLineCommentTrivia &&\n            sourceFile.text.substring(pos, end).includes(`TODO: ${AMBIGUOUS_CLASS_TODO}`));\n  }\n}\n"]}
@@ -1,5 +1,5 @@
1
1
  /**
2
- * @license Angular v10.0.0
2
+ * @license Angular v10.0.4
3
3
  * (c) 2010-2020 Google LLC. https://angular.io/
4
4
  * License: MIT
5
5
  */
@@ -18,10 +18,13 @@ declare interface AbstractType<T> extends Function {
18
18
  }
19
19
 
20
20
  /**
21
- * Base class for Angular Views, provides change detection functionality.
21
+ * Base class that provides change detection functionality.
22
22
  * A change-detection tree collects all views that are to be checked for changes.
23
23
  * Use the methods to add and remove views from the tree, initiate change-detection,
24
- * and explicitly mark views as _dirty_, meaning that they have changed and need to be rerendered.
24
+ * and explicitly mark views as _dirty_, meaning that they have changed and need to be re-rendered.
25
+ *
26
+ * @see [Using change detection hooks](guide/lifecycle-hooks#using-change-detection-hooks)
27
+ * @see [Defining custom change detection](guide/lifecycle-hooks#defining-custom-change-detection)
25
28
  *
26
29
  * @usageNotes
27
30
  *
@@ -691,25 +694,23 @@ declare abstract class NgModuleFactory_2<T> {
691
694
  }
692
695
 
693
696
  /**
694
- * Represents an instance of an NgModule created via a {@link NgModuleFactory}.
695
- *
696
- * `NgModuleRef` provides access to the NgModule Instance as well other objects related to this
697
- * NgModule Instance.
697
+ * Represents an instance of an `NgModule` created by an `NgModuleFactory`.
698
+ * Provides access to the `NgModule` instance and related objects.
698
699
  *
699
700
  * @publicApi
700
701
  */
701
702
  declare abstract class NgModuleRef<T> {
702
703
  /**
703
- * The injector that contains all of the providers of the NgModule.
704
+ * The injector that contains all of the providers of the `NgModule`.
704
705
  */
705
706
  abstract get injector(): Injector;
706
707
  /**
707
- * The ComponentFactoryResolver to get hold of the ComponentFactories
708
+ * The resolver that can retrieve the component factories
708
709
  * declared in the `entryComponents` property of the module.
709
710
  */
710
711
  abstract get componentFactoryResolver(): ComponentFactoryResolver;
711
712
  /**
712
- * The NgModule instance.
713
+ * The `NgModule` instance.
713
714
  */
714
715
  abstract get instance(): T;
715
716
  /**
@@ -717,7 +718,7 @@ declare abstract class NgModuleRef<T> {
717
718
  */
718
719
  abstract destroy(): void;
719
720
  /**
720
- * Allows to register a callback that will be called when the module is destroyed.
721
+ * Registers a callback to be executed when the module is destroyed.
721
722
  */
722
723
  abstract onDestroy(callback: () => void): void;
723
724
  }
@@ -835,7 +836,7 @@ declare type StaticProvider = ValueProvider | ExistingProvider | StaticClassProv
835
836
  *
836
837
  * Represents a type that a Component or other object is instances of.
837
838
  *
838
- * An example of a `Type` is `MyCustomComponent` class, which in JavaScript is be represented by
839
+ * An example of a `Type` is `MyCustomComponent` class, which in JavaScript is represented by
839
840
  * the `MyCustomComponent` constructor function.
840
841
  *
841
842
  * @publicApi
@@ -888,12 +889,9 @@ declare interface ValueSansProvider {
888
889
  }
889
890
 
890
891
  /**
891
- * Represents an Angular [view](guide/glossary#view),
892
- * specifically the [host view](guide/glossary#view-tree) that is defined by a component.
893
- * Also serves as the base class
894
- * that adds destroy methods for [embedded views](guide/glossary#view-tree).
892
+ * Represents an Angular [view](guide/glossary#view "Definition").
895
893
  *
896
- * @see `EmbeddedViewRef`
894
+ * @see {@link ChangeDetectorRef#usage-notes Change detection usage}
897
895
  *
898
896
  * @publicApi
899
897
  */
@@ -1,5 +1,5 @@
1
1
  /**
2
- * @license Angular v10.0.0
2
+ * @license Angular v10.0.4
3
3
  * (c) 2010-2020 Google LLC. https://angular.io/
4
4
  * License: MIT
5
5
  */
package/testing.d.ts CHANGED
@@ -1,5 +1,5 @@
1
1
  /**
2
- * @license Angular v10.0.0
2
+ * @license Angular v10.0.4
3
3
  * (c) 2010-2020 Google LLC. https://angular.io/
4
4
  * License: MIT
5
5
  */
@@ -1,48 +0,0 @@
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
- /**
9
- * Indicates that the result of a {@link Pipe} transformation has changed even though the
10
- * reference has not changed.
11
- *
12
- * Wrapped values are unwrapped automatically during the change detection, and the unwrapped value
13
- * is stored.
14
- *
15
- * Example:
16
- *
17
- * ```
18
- * if (this._latestValue === this._latestReturnedValue) {
19
- * return this._latestReturnedValue;
20
- * } else {
21
- * this._latestReturnedValue = this._latestValue;
22
- * return WrappedValue.wrap(this._latestValue); // this will force update
23
- * }
24
- * ```
25
- *
26
- * @publicApi
27
- */
28
- export class WrappedValue {
29
- constructor(value) {
30
- this.wrapped = value;
31
- }
32
- /** Creates a wrapped value. */
33
- static wrap(value) {
34
- return new WrappedValue(value);
35
- }
36
- /**
37
- * Returns the underlying value of a wrapped value.
38
- * Returns the given `value` when it is not wrapped.
39
- **/
40
- static unwrap(value) {
41
- return WrappedValue.isWrapped(value) ? value.wrapped : value;
42
- }
43
- /** Returns true if `value` is a wrapped value. */
44
- static isWrapped(value) {
45
- return value instanceof WrappedValue;
46
- }
47
- }
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiV3JhcHBlZFZhbHVlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvY29yZS9zcmMvdXRpbC9XcmFwcGVkVmFsdWUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBRUg7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0FtQkc7QUFDSCxNQUFNLE9BQU8sWUFBWTtJQUl2QixZQUFZLEtBQVU7UUFDcEIsSUFBSSxDQUFDLE9BQU8sR0FBRyxLQUFLLENBQUM7SUFDdkIsQ0FBQztJQUVELCtCQUErQjtJQUMvQixNQUFNLENBQUMsSUFBSSxDQUFDLEtBQVU7UUFDcEIsT0FBTyxJQUFJLFlBQVksQ0FBQyxLQUFLLENBQUMsQ0FBQztJQUNqQyxDQUFDO0lBRUQ7OztRQUdJO0lBQ0osTUFBTSxDQUFDLE1BQU0sQ0FBQyxLQUFVO1FBQ3RCLE9BQU8sWUFBWSxDQUFDLFNBQVMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDLE9BQU8sQ0FBQyxDQUFDLENBQUMsS0FBSyxDQUFDO0lBQy9ELENBQUM7SUFFRCxrREFBa0Q7SUFDbEQsTUFBTSxDQUFDLFNBQVMsQ0FBQyxLQUFVO1FBQ3pCLE9BQU8sS0FBSyxZQUFZLFlBQVksQ0FBQztJQUN2QyxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuLyoqXG4gKiBJbmRpY2F0ZXMgdGhhdCB0aGUgcmVzdWx0IG9mIGEge0BsaW5rIFBpcGV9IHRyYW5zZm9ybWF0aW9uIGhhcyBjaGFuZ2VkIGV2ZW4gdGhvdWdoIHRoZVxuICogcmVmZXJlbmNlIGhhcyBub3QgY2hhbmdlZC5cbiAqXG4gKiBXcmFwcGVkIHZhbHVlcyBhcmUgdW53cmFwcGVkIGF1dG9tYXRpY2FsbHkgZHVyaW5nIHRoZSBjaGFuZ2UgZGV0ZWN0aW9uLCBhbmQgdGhlIHVud3JhcHBlZCB2YWx1ZVxuICogaXMgc3RvcmVkLlxuICpcbiAqIEV4YW1wbGU6XG4gKlxuICogYGBgXG4gKiBpZiAodGhpcy5fbGF0ZXN0VmFsdWUgPT09IHRoaXMuX2xhdGVzdFJldHVybmVkVmFsdWUpIHtcbiAqICAgIHJldHVybiB0aGlzLl9sYXRlc3RSZXR1cm5lZFZhbHVlO1xuICogIH0gZWxzZSB7XG4gKiAgICB0aGlzLl9sYXRlc3RSZXR1cm5lZFZhbHVlID0gdGhpcy5fbGF0ZXN0VmFsdWU7XG4gKiAgICByZXR1cm4gV3JhcHBlZFZhbHVlLndyYXAodGhpcy5fbGF0ZXN0VmFsdWUpOyAvLyB0aGlzIHdpbGwgZm9yY2UgdXBkYXRlXG4gKiAgfVxuICogYGBgXG4gKlxuICogQHB1YmxpY0FwaVxuICovXG5leHBvcnQgY2xhc3MgV3JhcHBlZFZhbHVlIHtcbiAgLyoqIEBkZXByZWNhdGVkIGZyb20gNS4zLCB1c2UgYHVud3JhcCgpYCBpbnN0ZWFkIC0gd2lsbCBzd2l0Y2ggdG8gcHJvdGVjdGVkICovXG4gIHdyYXBwZWQ6IGFueTtcblxuICBjb25zdHJ1Y3Rvcih2YWx1ZTogYW55KSB7XG4gICAgdGhpcy53cmFwcGVkID0gdmFsdWU7XG4gIH1cblxuICAvKiogQ3JlYXRlcyBhIHdyYXBwZWQgdmFsdWUuICovXG4gIHN0YXRpYyB3cmFwKHZhbHVlOiBhbnkpOiBXcmFwcGVkVmFsdWUge1xuICAgIHJldHVybiBuZXcgV3JhcHBlZFZhbHVlKHZhbHVlKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBSZXR1cm5zIHRoZSB1bmRlcmx5aW5nIHZhbHVlIG9mIGEgd3JhcHBlZCB2YWx1ZS5cbiAgICogUmV0dXJucyB0aGUgZ2l2ZW4gYHZhbHVlYCB3aGVuIGl0IGlzIG5vdCB3cmFwcGVkLlxuICAgKiovXG4gIHN0YXRpYyB1bndyYXAodmFsdWU6IGFueSk6IGFueSB7XG4gICAgcmV0dXJuIFdyYXBwZWRWYWx1ZS5pc1dyYXBwZWQodmFsdWUpID8gdmFsdWUud3JhcHBlZCA6IHZhbHVlO1xuICB9XG5cbiAgLyoqIFJldHVybnMgdHJ1ZSBpZiBgdmFsdWVgIGlzIGEgd3JhcHBlZCB2YWx1ZS4gKi9cbiAgc3RhdGljIGlzV3JhcHBlZCh2YWx1ZTogYW55KTogdmFsdWUgaXMgV3JhcHBlZFZhbHVlIHtcbiAgICByZXR1cm4gdmFsdWUgaW5zdGFuY2VvZiBXcmFwcGVkVmFsdWU7XG4gIH1cbn1cbiJdfQ==