@angular/core 17.0.0-next.7 → 17.0.0-rc.0
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.
- package/esm2022/primitives/signals/index.mjs +15 -0
- package/esm2022/primitives/signals/src/computed.mjs +92 -0
- package/esm2022/primitives/signals/src/equality.mjs +14 -0
- package/esm2022/primitives/signals/src/errors.mjs +18 -0
- package/esm2022/primitives/signals/src/graph.mjs +291 -0
- package/esm2022/primitives/signals/src/signal.mjs +78 -0
- package/esm2022/primitives/signals/src/watch.mjs +82 -0
- package/esm2022/primitives/signals/src/weak_ref.mjs +11 -0
- package/esm2022/rxjs-interop/src/to_signal.mjs +45 -14
- package/esm2022/src/application_init.mjs +50 -2
- package/esm2022/src/application_ref.mjs +8 -2
- package/esm2022/src/application_tokens.mjs +16 -1
- package/esm2022/src/core_private_export.mjs +7 -6
- package/esm2022/src/core_reactivity_export_internal.mjs +6 -2
- package/esm2022/src/core_render3_private_export.mjs +3 -3
- package/esm2022/src/debug/debug_node.mjs +5 -9
- package/esm2022/src/defer/cleanup.mjs +70 -0
- package/esm2022/src/defer/discovery.mjs +47 -0
- package/esm2022/src/defer/dom_triggers.mjs +256 -0
- package/esm2022/src/defer/idle_scheduler.mjs +109 -0
- package/esm2022/src/defer/instructions.mjs +641 -0
- package/esm2022/src/defer/interfaces.mjs +79 -0
- package/esm2022/src/defer/timer_scheduler.mjs +192 -0
- package/esm2022/src/defer/utils.mjs +134 -0
- package/esm2022/src/errors.mjs +1 -1
- package/esm2022/src/hydration/api.mjs +1 -2
- package/esm2022/src/hydration/utils.mjs +2 -2
- package/esm2022/src/image_performance_warning.mjs +154 -0
- package/esm2022/src/linker/compiler.mjs +1 -1
- package/esm2022/src/metadata/directives.mjs +1 -1
- package/esm2022/src/render/api.mjs +1 -1
- package/esm2022/src/render3/after_render_hooks.mjs +35 -1
- package/esm2022/src/render3/assert.mjs +16 -1
- package/esm2022/src/render3/component_ref.mjs +12 -3
- package/esm2022/src/render3/debug/framework_injector_profiler.mjs +33 -4
- package/esm2022/src/render3/debug/injector_profiler.mjs +1 -1
- package/esm2022/src/render3/debug/set_debug_info.mjs +20 -0
- package/esm2022/src/render3/definition.mjs +2 -1
- package/esm2022/src/render3/deps_tracker/api.mjs +1 -1
- package/esm2022/src/render3/deps_tracker/deps_tracker.mjs +13 -2
- package/esm2022/src/render3/features/host_directives_feature.mjs +3 -8
- package/esm2022/src/render3/hooks.mjs +5 -5
- package/esm2022/src/render3/index.mjs +4 -2
- package/esm2022/src/render3/instructions/all.mjs +2 -2
- package/esm2022/src/render3/instructions/change_detection.mjs +31 -14
- package/esm2022/src/render3/instructions/control_flow.mjs +42 -23
- package/esm2022/src/render3/instructions/shared.mjs +5 -4
- package/esm2022/src/render3/interfaces/container.mjs +5 -7
- package/esm2022/src/render3/interfaces/definition.mjs +2 -4
- package/esm2022/src/render3/interfaces/i18n.mjs +1 -4
- package/esm2022/src/render3/interfaces/injector.mjs +1 -4
- package/esm2022/src/render3/interfaces/node.mjs +1 -4
- package/esm2022/src/render3/interfaces/projection.mjs +2 -4
- package/esm2022/src/render3/interfaces/query.mjs +2 -4
- package/esm2022/src/render3/interfaces/renderer.mjs +2 -4
- package/esm2022/src/render3/interfaces/renderer_dom.mjs +2 -4
- package/esm2022/src/render3/interfaces/view.mjs +5 -7
- package/esm2022/src/render3/jit/environment.mjs +3 -1
- package/esm2022/src/render3/list_reconciliation.mjs +58 -34
- package/esm2022/src/render3/node_manipulation.mjs +4 -6
- package/esm2022/src/render3/reactive_lview_consumer.mjs +3 -8
- package/esm2022/src/render3/reactivity/api.mjs +15 -0
- package/esm2022/src/render3/reactivity/asserts.mjs +26 -0
- package/esm2022/src/render3/reactivity/computed.mjs +19 -0
- package/esm2022/src/render3/reactivity/effect.mjs +7 -6
- package/esm2022/src/render3/reactivity/signal.mjs +32 -0
- package/esm2022/src/render3/reactivity/untracked.mjs +24 -0
- package/esm2022/src/render3/util/injector_discovery_utils.mjs +43 -14
- package/esm2022/src/render3/util/stringify_utils.mjs +28 -1
- package/esm2022/src/render3/util/view_utils.mjs +41 -25
- package/esm2022/src/render3/view_ref.mjs +3 -2
- package/esm2022/src/util/stringify.mjs +16 -1
- package/esm2022/src/version.mjs +1 -1
- package/esm2022/testing/src/logger.mjs +3 -3
- package/fesm2022/core.mjs +2881 -2270
- package/fesm2022/core.mjs.map +1 -1
- package/fesm2022/primitives/signals.mjs +539 -0
- package/fesm2022/primitives/signals.mjs.map +1 -0
- package/fesm2022/rxjs-interop.mjs +45 -14
- package/fesm2022/rxjs-interop.mjs.map +1 -1
- package/fesm2022/testing.mjs +1 -1
- package/index.d.ts +204 -168
- package/package.json +7 -1
- package/primitives/signals/index.d.ts +281 -0
- package/rxjs-interop/index.d.ts +15 -101
- package/schematics/collection.json +12 -2
- package/schematics/migrations/block-template-entities/bundle.js +551 -197
- package/schematics/migrations/block-template-entities/bundle.js.map +4 -4
- package/schematics/migrations/compiler-options/bundle.js +582 -0
- package/schematics/migrations/compiler-options/bundle.js.map +7 -0
- package/schematics/migrations/transfer-state/bundle.js +592 -0
- package/schematics/migrations/transfer-state/bundle.js.map +7 -0
- package/schematics/migrations.json +10 -0
- package/schematics/ng-generate/control-flow-migration/bundle.js +24309 -0
- package/schematics/ng-generate/control-flow-migration/bundle.js.map +7 -0
- package/schematics/ng-generate/control-flow-migration/schema.json +7 -0
- package/schematics/ng-generate/standalone-migration/bundle.js +1496 -924
- package/schematics/ng-generate/standalone-migration/bundle.js.map +4 -4
- package/testing/index.d.ts +1 -1
- package/esm2022/src/render3/instructions/defer.mjs +0 -1091
- package/esm2022/src/render3/instructions/defer_events.mjs +0 -164
- package/esm2022/src/render3/interfaces/defer.mjs +0 -72
- package/esm2022/src/signals/index.mjs +0 -16
- package/esm2022/src/signals/src/api.mjs +0 -39
- package/esm2022/src/signals/src/computed.mjs +0 -95
- package/esm2022/src/signals/src/errors.mjs +0 -18
- package/esm2022/src/signals/src/graph.mjs +0 -280
- package/esm2022/src/signals/src/signal.mjs +0 -92
- package/esm2022/src/signals/src/untracked.mjs +0 -26
- package/esm2022/src/signals/src/watch.mjs +0 -81
- package/esm2022/src/signals/src/weak_ref.mjs +0 -11
|
@@ -0,0 +1,79 @@
|
|
|
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
|
+
* Describes the state of defer block dependency loading.
|
|
10
|
+
*/
|
|
11
|
+
export var DeferDependenciesLoadingState;
|
|
12
|
+
(function (DeferDependenciesLoadingState) {
|
|
13
|
+
/** Initial state, dependency loading is not yet triggered */
|
|
14
|
+
DeferDependenciesLoadingState[DeferDependenciesLoadingState["NOT_STARTED"] = 0] = "NOT_STARTED";
|
|
15
|
+
/** Dependency loading is in progress */
|
|
16
|
+
DeferDependenciesLoadingState[DeferDependenciesLoadingState["IN_PROGRESS"] = 1] = "IN_PROGRESS";
|
|
17
|
+
/** Dependency loading has completed successfully */
|
|
18
|
+
DeferDependenciesLoadingState[DeferDependenciesLoadingState["COMPLETE"] = 2] = "COMPLETE";
|
|
19
|
+
/** Dependency loading has failed */
|
|
20
|
+
DeferDependenciesLoadingState[DeferDependenciesLoadingState["FAILED"] = 3] = "FAILED";
|
|
21
|
+
})(DeferDependenciesLoadingState || (DeferDependenciesLoadingState = {}));
|
|
22
|
+
/** Slot index where `minimum` parameter value is stored. */
|
|
23
|
+
export const MINIMUM_SLOT = 0;
|
|
24
|
+
/** Slot index where `after` parameter value is stored. */
|
|
25
|
+
export const LOADING_AFTER_SLOT = 1;
|
|
26
|
+
/**
|
|
27
|
+
* Describes the current state of this defer block instance.
|
|
28
|
+
*
|
|
29
|
+
* @publicApi
|
|
30
|
+
* @developerPreview
|
|
31
|
+
*/
|
|
32
|
+
export var DeferBlockState;
|
|
33
|
+
(function (DeferBlockState) {
|
|
34
|
+
/** The placeholder block content is rendered */
|
|
35
|
+
DeferBlockState[DeferBlockState["Placeholder"] = 0] = "Placeholder";
|
|
36
|
+
/** The loading block content is rendered */
|
|
37
|
+
DeferBlockState[DeferBlockState["Loading"] = 1] = "Loading";
|
|
38
|
+
/** The main content block content is rendered */
|
|
39
|
+
DeferBlockState[DeferBlockState["Complete"] = 2] = "Complete";
|
|
40
|
+
/** The error block content is rendered */
|
|
41
|
+
DeferBlockState[DeferBlockState["Error"] = 3] = "Error";
|
|
42
|
+
})(DeferBlockState || (DeferBlockState = {}));
|
|
43
|
+
/**
|
|
44
|
+
* Describes the initial state of this defer block instance.
|
|
45
|
+
*
|
|
46
|
+
* Note: this state is internal only and *must* be represented
|
|
47
|
+
* with a number lower than any value in the `DeferBlockState` enum.
|
|
48
|
+
*/
|
|
49
|
+
export var DeferBlockInternalState;
|
|
50
|
+
(function (DeferBlockInternalState) {
|
|
51
|
+
/** Initial state. Nothing is rendered yet. */
|
|
52
|
+
DeferBlockInternalState[DeferBlockInternalState["Initial"] = -1] = "Initial";
|
|
53
|
+
})(DeferBlockInternalState || (DeferBlockInternalState = {}));
|
|
54
|
+
export const NEXT_DEFER_BLOCK_STATE = 0;
|
|
55
|
+
// Note: it's *important* to keep the state in this slot, because this slot
|
|
56
|
+
// is used by runtime logic to differentiate between LViews, LContainers and
|
|
57
|
+
// other types (see `isLView` and `isLContainer` functions). In case of defer
|
|
58
|
+
// blocks, this slot would always be a number.
|
|
59
|
+
export const DEFER_BLOCK_STATE = 1;
|
|
60
|
+
export const STATE_IS_FROZEN_UNTIL = 2;
|
|
61
|
+
export const LOADING_AFTER_CLEANUP_FN = 3;
|
|
62
|
+
/**
|
|
63
|
+
* Options for configuring defer blocks behavior.
|
|
64
|
+
* @publicApi
|
|
65
|
+
* @developerPreview
|
|
66
|
+
*/
|
|
67
|
+
export var DeferBlockBehavior;
|
|
68
|
+
(function (DeferBlockBehavior) {
|
|
69
|
+
/**
|
|
70
|
+
* Manual triggering mode for defer blocks. Provides control over when defer blocks render
|
|
71
|
+
* and which state they render. This is the default behavior in test environments.
|
|
72
|
+
*/
|
|
73
|
+
DeferBlockBehavior[DeferBlockBehavior["Manual"] = 0] = "Manual";
|
|
74
|
+
/**
|
|
75
|
+
* Playthrough mode for defer blocks. This mode behaves like defer blocks would in a browser.
|
|
76
|
+
*/
|
|
77
|
+
DeferBlockBehavior[DeferBlockBehavior["Playthrough"] = 1] = "Playthrough";
|
|
78
|
+
})(DeferBlockBehavior || (DeferBlockBehavior = {}));
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"interfaces.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/defer/interfaces.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAsBH;;GAEG;AACH,MAAM,CAAN,IAAY,6BAYX;AAZD,WAAY,6BAA6B;IACvC,6DAA6D;IAC7D,+FAAW,CAAA;IAEX,wCAAwC;IACxC,+FAAW,CAAA;IAEX,oDAAoD;IACpD,yFAAQ,CAAA;IAER,oCAAoC;IACpC,qFAAM,CAAA;AACR,CAAC,EAZW,6BAA6B,KAA7B,6BAA6B,QAYxC;AAED,4DAA4D;AAC5D,MAAM,CAAC,MAAM,YAAY,GAAG,CAAC,CAAC;AAE9B,0DAA0D;AAC1D,MAAM,CAAC,MAAM,kBAAkB,GAAG,CAAC,CAAC;AA6DpC;;;;;GAKG;AACH,MAAM,CAAN,IAAY,eAYX;AAZD,WAAY,eAAe;IACzB,gDAAgD;IAChD,mEAAe,CAAA;IAEf,4CAA4C;IAC5C,2DAAW,CAAA;IAEX,iDAAiD;IACjD,6DAAY,CAAA;IAEZ,0CAA0C;IAC1C,uDAAS,CAAA;AACX,CAAC,EAZW,eAAe,KAAf,eAAe,QAY1B;AAED;;;;;GAKG;AACH,MAAM,CAAN,IAAY,uBAGX;AAHD,WAAY,uBAAuB;IACjC,8CAA8C;IAC9C,4EAAY,CAAA;AACd,CAAC,EAHW,uBAAuB,KAAvB,uBAAuB,QAGlC;AAED,MAAM,CAAC,MAAM,sBAAsB,GAAG,CAAC,CAAC;AACxC,2EAA2E;AAC3E,4EAA4E;AAC5E,6EAA6E;AAC7E,8CAA8C;AAC9C,MAAM,CAAC,MAAM,iBAAiB,GAAG,CAAC,CAAC;AACnC,MAAM,CAAC,MAAM,qBAAqB,GAAG,CAAC,CAAC;AACvC,MAAM,CAAC,MAAM,wBAAwB,GAAG,CAAC,CAAC;AAyC1C;;;;GAIG;AACH,MAAM,CAAN,IAAY,kBAWX;AAXD,WAAY,kBAAkB;IAC5B;;;OAGG;IACH,+DAAM,CAAA;IAEN;;OAEG;IACH,yEAAW,CAAA;AACb,CAAC,EAXW,kBAAkB,KAAlB,kBAAkB,QAW7B","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 type {DependencyType} from '../render3/interfaces/definition';\n\n/**\n * Describes the shape of a function generated by the compiler\n * to download dependencies that can be defer-loaded.\n */\nexport type DependencyResolverFn = () => Array<Promise<DependencyType>>;\n\n/**\n * Enumerates all `on` triggers of a defer block.\n */\nexport const enum DeferBlockTriggers {\n  OnIdle,\n  OnTimer,\n  OnImmediate,\n  OnHover,\n  OnInteraction,\n  OnViewport,\n}\n\n/**\n * Describes the state of defer block dependency loading.\n */\nexport enum DeferDependenciesLoadingState {\n  /** Initial state, dependency loading is not yet triggered */\n  NOT_STARTED,\n\n  /** Dependency loading is in progress */\n  IN_PROGRESS,\n\n  /** Dependency loading has completed successfully */\n  COMPLETE,\n\n  /** Dependency loading has failed */\n  FAILED,\n}\n\n/** Slot index where `minimum` parameter value is stored. */\nexport const MINIMUM_SLOT = 0;\n\n/** Slot index where `after` parameter value is stored. */\nexport const LOADING_AFTER_SLOT = 1;\n\n/** Configuration object for a loading block as it is stored in the component constants. */\nexport type DeferredLoadingBlockConfig = [minimumTime: number|null, afterTime: number|null];\n\n/** Configuration object for a placeholder block as it is stored in the component constants. */\nexport type DeferredPlaceholderBlockConfig = [minimumTime: number|null];\n\n/**\n * Describes the data shared across all instances of a defer block.\n */\nexport interface TDeferBlockDetails {\n  /**\n   * Index in an LView and TData arrays where a template for the primary content\n   * can be found.\n   */\n  primaryTmplIndex: number;\n\n  /**\n   * Index in an LView and TData arrays where a template for the loading block can be found.\n   */\n  loadingTmplIndex: number|null;\n\n  /**\n   * Extra configuration parameters (such as `after` and `minimum`) for the loading block.\n   */\n  loadingBlockConfig: DeferredLoadingBlockConfig|null;\n\n  /**\n   * Index in an LView and TData arrays where a template for the placeholder block can be found.\n   */\n  placeholderTmplIndex: number|null;\n\n  /**\n   * Extra configuration parameters (such as `after` and `minimum`) for the placeholder block.\n   */\n  placeholderBlockConfig: DeferredPlaceholderBlockConfig|null;\n\n  /**\n   * Index in an LView and TData arrays where a template for the error block can be found.\n   */\n  errorTmplIndex: number|null;\n\n  /**\n   * Compiler-generated function that loads all dependencies for a defer block.\n   */\n  dependencyResolverFn: DependencyResolverFn|null;\n\n  /**\n   * Keeps track of the current loading state of defer block dependencies.\n   */\n  loadingState: DeferDependenciesLoadingState;\n\n  /**\n   * Dependency loading Promise. This Promise is helpful for cases when there\n   * are multiple instances of a defer block (e.g. if it was used inside of an *ngFor),\n   * which all await the same set of dependencies.\n   */\n  loadingPromise: Promise<unknown>|null;\n}\n\n/**\n * Describes the current state of this defer block instance.\n *\n * @publicApi\n * @developerPreview\n */\nexport enum DeferBlockState {\n  /** The placeholder block content is rendered */\n  Placeholder = 0,\n\n  /** The loading block content is rendered */\n  Loading = 1,\n\n  /** The main content block content is rendered */\n  Complete = 2,\n\n  /** The error block content is rendered */\n  Error = 3,\n}\n\n/**\n * Describes the initial state of this defer block instance.\n *\n * Note: this state is internal only and *must* be represented\n * with a number lower than any value in the `DeferBlockState` enum.\n */\nexport enum DeferBlockInternalState {\n  /** Initial state. Nothing is rendered yet. */\n  Initial = -1,\n}\n\nexport const NEXT_DEFER_BLOCK_STATE = 0;\n// Note: it's *important* to keep the state in this slot, because this slot\n// is used by runtime logic to differentiate between LViews, LContainers and\n// other types (see `isLView` and `isLContainer` functions). In case of defer\n// blocks, this slot would always be a number.\nexport const DEFER_BLOCK_STATE = 1;\nexport const STATE_IS_FROZEN_UNTIL = 2;\nexport const LOADING_AFTER_CLEANUP_FN = 3;\n\n/**\n * Describes instance-specific defer block data.\n *\n * Note: currently there is only the `state` slot, but more slots\n * would be added later to keep track of `after` and `maximum` features\n * (which would require per-instance state).\n */\nexport interface LDeferBlockDetails extends Array<unknown> {\n  /**\n   * Currently rendered block state.\n   */\n  [DEFER_BLOCK_STATE]: DeferBlockState|DeferBlockInternalState;\n\n  /**\n   * Block state that was requested when another state was rendered.\n   */\n  [NEXT_DEFER_BLOCK_STATE]: DeferBlockState|null;\n\n  /**\n   * Timestamp indicating when the current state can be switched to\n   * the next one, in case teh current state has `minimum` parameter.\n   */\n  [STATE_IS_FROZEN_UNTIL]: number|null;\n\n  /**\n   * Contains a reference to a cleanup function which cancels a timeout\n   * when Angular waits before rendering loading state. This is used when\n   * the loading block has the `after` parameter configured.\n   */\n  [LOADING_AFTER_CLEANUP_FN]: VoidFunction|null;\n}\n\n/**\n * Internal structure used for configuration of defer block behavior.\n * */\nexport interface DeferBlockConfig {\n  behavior: DeferBlockBehavior;\n}\n\n/**\n * Options for configuring defer blocks behavior.\n * @publicApi\n * @developerPreview\n */\nexport enum DeferBlockBehavior {\n  /**\n   * Manual triggering mode for defer blocks. Provides control over when defer blocks render\n   * and which state they render. This is the default behavior in test environments.\n   */\n  Manual,\n\n  /**\n   * Playthrough mode for defer blocks. This mode behaves like defer blocks would in a browser.\n   */\n  Playthrough,\n}\n\n/**\n * **INTERNAL**, avoid referencing it in application code.\n *\n * Describes a helper class that allows to intercept a call to retrieve current\n * dependency loading function and replace it with a different implementation.\n * This interceptor class is needed to allow testing blocks in different states\n * by simulating loading response.\n */\nexport interface DeferBlockDependencyInterceptor {\n  /**\n   * Invoked for each defer block when dependency loading function is accessed.\n   */\n  intercept(dependencyFn: DependencyResolverFn|null): DependencyResolverFn|null;\n\n  /**\n   * Allows to configure an interceptor function.\n   */\n  setInterceptor(interceptorFn: (current: DependencyResolverFn) => DependencyResolverFn): void;\n}\n"]}
|
|
@@ -0,0 +1,192 @@
|
|
|
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
|
+
import { ɵɵdefineInjectable } from '../di';
|
|
9
|
+
import { INJECTOR } from '../render3/interfaces/view';
|
|
10
|
+
import { arrayInsert2, arraySplice } from '../util/array_utils';
|
|
11
|
+
import { wrapWithLViewCleanup } from './utils';
|
|
12
|
+
/**
|
|
13
|
+
* Returns a function that captures a provided delay.
|
|
14
|
+
* Invoking the returned function schedules a trigger.
|
|
15
|
+
*/
|
|
16
|
+
export function onTimer(delay) {
|
|
17
|
+
return (callback, lView, withLViewCleanup) => scheduleTimerTrigger(delay, callback, lView, withLViewCleanup);
|
|
18
|
+
}
|
|
19
|
+
/**
|
|
20
|
+
* Schedules a callback to be invoked after a given timeout.
|
|
21
|
+
*
|
|
22
|
+
* @param delay A number of ms to wait until firing a callback.
|
|
23
|
+
* @param callback A function to be invoked after a timeout.
|
|
24
|
+
* @param lView LView that hosts an instance of a defer block.
|
|
25
|
+
* @param withLViewCleanup A flag that indicates whether a scheduled callback
|
|
26
|
+
* should be cancelled in case an LView is destroyed before a callback
|
|
27
|
+
* was invoked.
|
|
28
|
+
*/
|
|
29
|
+
export function scheduleTimerTrigger(delay, callback, lView, withLViewCleanup) {
|
|
30
|
+
const injector = lView[INJECTOR];
|
|
31
|
+
const scheduler = injector.get(TimerScheduler);
|
|
32
|
+
const cleanupFn = () => scheduler.remove(callback);
|
|
33
|
+
const wrappedCallback = withLViewCleanup ? wrapWithLViewCleanup(callback, lView, cleanupFn) : callback;
|
|
34
|
+
scheduler.add(delay, wrappedCallback);
|
|
35
|
+
return cleanupFn;
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Helper service to schedule `setTimeout`s for batches of defer blocks,
|
|
39
|
+
* to avoid calling `setTimeout` for each defer block (e.g. if defer blocks
|
|
40
|
+
* are created inside a for loop).
|
|
41
|
+
*/
|
|
42
|
+
export class TimerScheduler {
|
|
43
|
+
constructor() {
|
|
44
|
+
// Indicates whether current callbacks are being invoked.
|
|
45
|
+
this.executingCallbacks = false;
|
|
46
|
+
// Currently scheduled `setTimeout` id.
|
|
47
|
+
this.timeoutId = null;
|
|
48
|
+
// When currently scheduled timer would fire.
|
|
49
|
+
this.invokeTimerAt = null;
|
|
50
|
+
// List of callbacks to be invoked.
|
|
51
|
+
// For each callback we also store a timestamp on when the callback
|
|
52
|
+
// should be invoked. We store timestamps and callback functions
|
|
53
|
+
// in a flat array to avoid creating new objects for each entry.
|
|
54
|
+
// [timestamp1, callback1, timestamp2, callback2, ...]
|
|
55
|
+
this.current = [];
|
|
56
|
+
// List of callbacks collected while invoking current set of callbacks.
|
|
57
|
+
// Those callbacks are added to the "current" queue at the end of
|
|
58
|
+
// the current callback invocation. The shape of this list is the same
|
|
59
|
+
// as the shape of the `current` list.
|
|
60
|
+
this.deferred = [];
|
|
61
|
+
}
|
|
62
|
+
add(delay, callback) {
|
|
63
|
+
const target = this.executingCallbacks ? this.deferred : this.current;
|
|
64
|
+
this.addToQueue(target, Date.now() + delay, callback);
|
|
65
|
+
this.scheduleTimer();
|
|
66
|
+
}
|
|
67
|
+
remove(callback) {
|
|
68
|
+
const callbackIndex = this.removeFromQueue(this.current, callback);
|
|
69
|
+
if (callbackIndex === -1) {
|
|
70
|
+
// Try cleaning up deferred queue only in case
|
|
71
|
+
// we didn't find a callback in the "current" queue.
|
|
72
|
+
this.removeFromQueue(this.deferred, callback);
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
addToQueue(target, invokeAt, callback) {
|
|
76
|
+
let insertAtIndex = target.length;
|
|
77
|
+
for (let i = 0; i < target.length; i += 2) {
|
|
78
|
+
const invokeQueuedCallbackAt = target[i];
|
|
79
|
+
if (invokeQueuedCallbackAt > invokeAt) {
|
|
80
|
+
// We've reached a first timer that is scheduled
|
|
81
|
+
// for a later time than what we are trying to insert.
|
|
82
|
+
// This is the location at which we need to insert,
|
|
83
|
+
// no need to iterate further.
|
|
84
|
+
insertAtIndex = i;
|
|
85
|
+
break;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
arrayInsert2(target, insertAtIndex, invokeAt, callback);
|
|
89
|
+
}
|
|
90
|
+
removeFromQueue(target, callback) {
|
|
91
|
+
let index = -1;
|
|
92
|
+
for (let i = 0; i < target.length; i += 2) {
|
|
93
|
+
const queuedCallback = target[i + 1];
|
|
94
|
+
if (queuedCallback === callback) {
|
|
95
|
+
index = i;
|
|
96
|
+
break;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
if (index > -1) {
|
|
100
|
+
// Remove 2 elements: a timestamp slot and
|
|
101
|
+
// the following slot with a callback function.
|
|
102
|
+
arraySplice(target, index, 2);
|
|
103
|
+
}
|
|
104
|
+
return index;
|
|
105
|
+
}
|
|
106
|
+
scheduleTimer() {
|
|
107
|
+
const callback = () => {
|
|
108
|
+
clearTimeout(this.timeoutId);
|
|
109
|
+
this.timeoutId = null;
|
|
110
|
+
this.executingCallbacks = true;
|
|
111
|
+
// Invoke callbacks that were scheduled to run
|
|
112
|
+
// before the current time.
|
|
113
|
+
let now = Date.now();
|
|
114
|
+
let lastCallbackIndex = null;
|
|
115
|
+
for (let i = 0; i < this.current.length; i += 2) {
|
|
116
|
+
const invokeAt = this.current[i];
|
|
117
|
+
const callback = this.current[i + 1];
|
|
118
|
+
if (invokeAt <= now) {
|
|
119
|
+
callback();
|
|
120
|
+
// Point at the invoked callback function, which is located
|
|
121
|
+
// after the timestamp.
|
|
122
|
+
lastCallbackIndex = i + 1;
|
|
123
|
+
}
|
|
124
|
+
else {
|
|
125
|
+
// We've reached a timer that should not be invoked yet.
|
|
126
|
+
break;
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
if (lastCallbackIndex !== null) {
|
|
130
|
+
// If last callback index is `null` - no callbacks were invoked,
|
|
131
|
+
// so no cleanup is needed. Otherwise, remove invoked callbacks
|
|
132
|
+
// from the queue.
|
|
133
|
+
arraySplice(this.current, 0, lastCallbackIndex + 1);
|
|
134
|
+
}
|
|
135
|
+
this.executingCallbacks = false;
|
|
136
|
+
// If there are any callbacks added during an invocation
|
|
137
|
+
// of the current ones - move them over to the "current"
|
|
138
|
+
// queue.
|
|
139
|
+
if (this.deferred.length > 0) {
|
|
140
|
+
for (let i = 0; i < this.deferred.length; i += 2) {
|
|
141
|
+
const invokeAt = this.deferred[i];
|
|
142
|
+
const callback = this.deferred[i + 1];
|
|
143
|
+
this.addToQueue(this.current, invokeAt, callback);
|
|
144
|
+
}
|
|
145
|
+
this.deferred.length = 0;
|
|
146
|
+
}
|
|
147
|
+
this.scheduleTimer();
|
|
148
|
+
};
|
|
149
|
+
// Avoid running timer callbacks more than once per
|
|
150
|
+
// average frame duration. This is needed for better
|
|
151
|
+
// batching and to avoid kicking off excessive change
|
|
152
|
+
// detection cycles.
|
|
153
|
+
const FRAME_DURATION_MS = 16; // 1000ms / 60fps
|
|
154
|
+
if (this.current.length > 0) {
|
|
155
|
+
const now = Date.now();
|
|
156
|
+
// First element in the queue points at the timestamp
|
|
157
|
+
// of the first (earliest) event.
|
|
158
|
+
const invokeAt = this.current[0];
|
|
159
|
+
if (!this.timeoutId ||
|
|
160
|
+
// Reschedule a timer in case a queue contains an item with
|
|
161
|
+
// an earlier timestamp and the delta is more than an average
|
|
162
|
+
// frame duration.
|
|
163
|
+
(this.invokeTimerAt && (this.invokeTimerAt - invokeAt > FRAME_DURATION_MS))) {
|
|
164
|
+
if (this.timeoutId !== null) {
|
|
165
|
+
// There was a timeout already, but an earlier event was added
|
|
166
|
+
// into the queue. In this case we drop an old timer and setup
|
|
167
|
+
// a new one with an updated (smaller) timeout.
|
|
168
|
+
clearTimeout(this.timeoutId);
|
|
169
|
+
this.timeoutId = null;
|
|
170
|
+
}
|
|
171
|
+
const timeout = Math.max(invokeAt - now, FRAME_DURATION_MS);
|
|
172
|
+
this.invokeTimerAt = invokeAt;
|
|
173
|
+
this.timeoutId = setTimeout(callback, timeout);
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
}
|
|
177
|
+
ngOnDestroy() {
|
|
178
|
+
if (this.timeoutId !== null) {
|
|
179
|
+
clearTimeout(this.timeoutId);
|
|
180
|
+
this.timeoutId = null;
|
|
181
|
+
}
|
|
182
|
+
this.current.length = 0;
|
|
183
|
+
this.deferred.length = 0;
|
|
184
|
+
}
|
|
185
|
+
/** @nocollapse */
|
|
186
|
+
static { this.ɵprov = ɵɵdefineInjectable({
|
|
187
|
+
token: TimerScheduler,
|
|
188
|
+
providedIn: 'root',
|
|
189
|
+
factory: () => new TimerScheduler(),
|
|
190
|
+
}); }
|
|
191
|
+
}
|
|
192
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timer_scheduler.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/defer/timer_scheduler.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAC,kBAAkB,EAAC,MAAM,OAAO,CAAC;AACzC,OAAO,EAAC,QAAQ,EAAQ,MAAM,4BAA4B,CAAC;AAC3D,OAAO,EAAC,YAAY,EAAE,WAAW,EAAC,MAAM,qBAAqB,CAAC;AAE9D,OAAO,EAAC,oBAAoB,EAAC,MAAM,SAAS,CAAC;AAE7C;;;GAGG;AACH,MAAM,UAAU,OAAO,CAAC,KAAa;IACnC,OAAO,CAAC,QAAsB,EAAE,KAAY,EAAE,gBAAyB,EAAE,EAAE,CAChE,oBAAoB,CAAC,KAAK,EAAE,QAAQ,EAAE,KAAK,EAAE,gBAAgB,CAAC,CAAC;AAC5E,CAAC;AAED;;;;;;;;;GASG;AACH,MAAM,UAAU,oBAAoB,CAChC,KAAa,EAAE,QAAsB,EAAE,KAAY,EAAE,gBAAyB;IAChF,MAAM,QAAQ,GAAG,KAAK,CAAC,QAAQ,CAAE,CAAC;IAClC,MAAM,SAAS,GAAG,QAAQ,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;IAC/C,MAAM,SAAS,GAAG,GAAG,EAAE,CAAC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;IACnD,MAAM,eAAe,GACjB,gBAAgB,CAAC,CAAC,CAAC,oBAAoB,CAAC,QAAQ,EAAE,KAAK,EAAE,SAAS,CAAC,CAAC,CAAC,CAAC,QAAQ,CAAC;IACnF,SAAS,CAAC,GAAG,CAAC,KAAK,EAAE,eAAe,CAAC,CAAC;IACtC,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;;;GAIG;AACH,MAAM,OAAO,cAAc;IAA3B;QACE,yDAAyD;QACzD,uBAAkB,GAAG,KAAK,CAAC;QAE3B,uCAAuC;QACvC,cAAS,GAAgB,IAAI,CAAC;QAE9B,6CAA6C;QAC7C,kBAAa,GAAgB,IAAI,CAAC;QAElC,mCAAmC;QACnC,mEAAmE;QACnE,gEAAgE;QAChE,gEAAgE;QAChE,sDAAsD;QACtD,YAAO,GAA+B,EAAE,CAAC;QAEzC,uEAAuE;QACvE,iEAAiE;QACjE,sEAAsE;QACtE,sCAAsC;QACtC,aAAQ,GAA+B,EAAE,CAAC;IA8I5C,CAAC;IA5IC,GAAG,CAAC,KAAa,EAAE,QAAsB;QACvC,MAAM,MAAM,GAAG,IAAI,CAAC,kBAAkB,CAAC,CAAC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC;QACtE,IAAI,CAAC,UAAU,CAAC,MAAM,EAAE,IAAI,CAAC,GAAG,EAAE,GAAG,KAAK,EAAE,QAAQ,CAAC,CAAC;QACtD,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAED,MAAM,CAAC,QAAsB;QAC3B,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAC,CAAC;QACnE,IAAI,aAAa,KAAK,CAAC,CAAC,EAAE;YACxB,8CAA8C;YAC9C,oDAAoD;YACpD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,QAAQ,EAAE,QAAQ,CAAC,CAAC;SAC/C;IACH,CAAC;IAEO,UAAU,CAAC,MAAkC,EAAE,QAAgB,EAAE,QAAsB;QAC7F,IAAI,aAAa,GAAG,MAAM,CAAC,MAAM,CAAC;QAClC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,EAAE;YACzC,MAAM,sBAAsB,GAAG,MAAM,CAAC,CAAC,CAAW,CAAC;YACnD,IAAI,sBAAsB,GAAG,QAAQ,EAAE;gBACrC,gDAAgD;gBAChD,sDAAsD;gBACtD,mDAAmD;gBACnD,8BAA8B;gBAC9B,aAAa,GAAG,CAAC,CAAC;gBAClB,MAAM;aACP;SACF;QACD,YAAY,CAAC,MAAM,EAAE,aAAa,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC;IAC1D,CAAC;IAEO,eAAe,CAAC,MAAkC,EAAE,QAAsB;QAChF,IAAI,KAAK,GAAG,CAAC,CAAC,CAAC;QACf,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,MAAM,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,EAAE;YACzC,MAAM,cAAc,GAAG,MAAM,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC;YACrC,IAAI,cAAc,KAAK,QAAQ,EAAE;gBAC/B,KAAK,GAAG,CAAC,CAAC;gBACV,MAAM;aACP;SACF;QACD,IAAI,KAAK,GAAG,CAAC,CAAC,EAAE;YACd,0CAA0C;YAC1C,+CAA+C;YAC/C,WAAW,CAAC,MAAM,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;SAC/B;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAEO,aAAa;QACnB,MAAM,QAAQ,GAAG,GAAG,EAAE;YACpB,YAAY,CAAC,IAAI,CAAC,SAAU,CAAC,CAAC;YAC9B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;YAEtB,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;YAE/B,8CAA8C;YAC9C,2BAA2B;YAC3B,IAAI,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YACrB,IAAI,iBAAiB,GAAgB,IAAI,CAAC;YAC1C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,EAAE;gBAC/C,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,CAAW,CAAC;gBAC3C,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,GAAG,CAAC,CAAiB,CAAC;gBACrD,IAAI,QAAQ,IAAI,GAAG,EAAE;oBACnB,QAAQ,EAAE,CAAC;oBACX,2DAA2D;oBAC3D,uBAAuB;oBACvB,iBAAiB,GAAG,CAAC,GAAG,CAAC,CAAC;iBAC3B;qBAAM;oBACL,wDAAwD;oBACxD,MAAM;iBACP;aACF;YACD,IAAI,iBAAiB,KAAK,IAAI,EAAE;gBAC9B,gEAAgE;gBAChE,+DAA+D;gBAC/D,kBAAkB;gBAClB,WAAW,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,EAAE,iBAAiB,GAAG,CAAC,CAAC,CAAC;aACrD;YAED,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;YAEhC,wDAAwD;YACxD,wDAAwD;YACxD,SAAS;YACT,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE;gBAC5B,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,IAAI,CAAC,EAAE;oBAChD,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAW,CAAC;oBAC5C,MAAM,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,CAAiB,CAAC;oBACtD,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,OAAO,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC;iBACnD;gBACD,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC;aAC1B;YACD,IAAI,CAAC,aAAa,EAAE,CAAC;QACvB,CAAC,CAAC;QAEF,mDAAmD;QACnD,oDAAoD;QACpD,qDAAqD;QACrD,oBAAoB;QACpB,MAAM,iBAAiB,GAAG,EAAE,CAAC,CAAE,iBAAiB;QAEhD,IAAI,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE;YAC3B,MAAM,GAAG,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YACvB,qDAAqD;YACrD,iCAAiC;YACjC,MAAM,QAAQ,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,CAAW,CAAC;YAC3C,IAAI,CAAC,IAAI,CAAC,SAAS;gBACf,2DAA2D;gBAC3D,6DAA6D;gBAC7D,kBAAkB;gBAClB,CAAC,IAAI,CAAC,aAAa,IAAI,CAAC,IAAI,CAAC,aAAa,GAAG,QAAQ,GAAG,iBAAiB,CAAC,CAAC,EAAE;gBAC/E,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,EAAE;oBAC3B,8DAA8D;oBAC9D,8DAA8D;oBAC9D,+CAA+C;oBAC/C,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;oBAC7B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;iBACvB;gBACD,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,GAAG,GAAG,EAAE,iBAAiB,CAAC,CAAC;gBAC5D,IAAI,CAAC,aAAa,GAAG,QAAQ,CAAC;gBAC9B,IAAI,CAAC,SAAS,GAAG,UAAU,CAAC,QAAQ,EAAE,OAAO,CAAsB,CAAC;aACrE;SACF;IACH,CAAC;IAED,WAAW;QACT,IAAI,IAAI,CAAC,SAAS,KAAK,IAAI,EAAE;YAC3B,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YAC7B,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;SACvB;QACD,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC;QACxB,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,CAAC;IAC3B,CAAC;IAED,kBAAkB;aACX,UAAK,GAA6B,kBAAkB,CAAC;QAC1D,KAAK,EAAE,cAAc;QACrB,UAAU,EAAE,MAAM;QAClB,OAAO,EAAE,GAAG,EAAE,CAAC,IAAI,cAAc,EAAE;KACpC,CAAC,AAJU,CAIT","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 {ɵɵdefineInjectable} from '../di';\nimport {INJECTOR, LView} from '../render3/interfaces/view';\nimport {arrayInsert2, arraySplice} from '../util/array_utils';\n\nimport {wrapWithLViewCleanup} from './utils';\n\n/**\n * Returns a function that captures a provided delay.\n * Invoking the returned function schedules a trigger.\n */\nexport function onTimer(delay: number) {\n  return (callback: VoidFunction, lView: LView, withLViewCleanup: boolean) =>\n             scheduleTimerTrigger(delay, callback, lView, withLViewCleanup);\n}\n\n/**\n * Schedules a callback to be invoked after a given timeout.\n *\n * @param delay A number of ms to wait until firing a callback.\n * @param callback A function to be invoked after a timeout.\n * @param lView LView that hosts an instance of a defer block.\n * @param withLViewCleanup A flag that indicates whether a scheduled callback\n *           should be cancelled in case an LView is destroyed before a callback\n *           was invoked.\n */\nexport function scheduleTimerTrigger(\n    delay: number, callback: VoidFunction, lView: LView, withLViewCleanup: boolean) {\n  const injector = lView[INJECTOR]!;\n  const scheduler = injector.get(TimerScheduler);\n  const cleanupFn = () => scheduler.remove(callback);\n  const wrappedCallback =\n      withLViewCleanup ? wrapWithLViewCleanup(callback, lView, cleanupFn) : callback;\n  scheduler.add(delay, wrappedCallback);\n  return cleanupFn;\n}\n\n/**\n * Helper service to schedule `setTimeout`s for batches of defer blocks,\n * to avoid calling `setTimeout` for each defer block (e.g. if defer blocks\n * are created inside a for loop).\n */\nexport class TimerScheduler {\n  // Indicates whether current callbacks are being invoked.\n  executingCallbacks = false;\n\n  // Currently scheduled `setTimeout` id.\n  timeoutId: number|null = null;\n\n  // When currently scheduled timer would fire.\n  invokeTimerAt: number|null = null;\n\n  // List of callbacks to be invoked.\n  // For each callback we also store a timestamp on when the callback\n  // should be invoked. We store timestamps and callback functions\n  // in a flat array to avoid creating new objects for each entry.\n  // [timestamp1, callback1, timestamp2, callback2, ...]\n  current: Array<number|VoidFunction> = [];\n\n  // List of callbacks collected while invoking current set of callbacks.\n  // Those callbacks are added to the \"current\" queue at the end of\n  // the current callback invocation. The shape of this list is the same\n  // as the shape of the `current` list.\n  deferred: Array<number|VoidFunction> = [];\n\n  add(delay: number, callback: VoidFunction) {\n    const target = this.executingCallbacks ? this.deferred : this.current;\n    this.addToQueue(target, Date.now() + delay, callback);\n    this.scheduleTimer();\n  }\n\n  remove(callback: VoidFunction) {\n    const callbackIndex = this.removeFromQueue(this.current, callback);\n    if (callbackIndex === -1) {\n      // Try cleaning up deferred queue only in case\n      // we didn't find a callback in the \"current\" queue.\n      this.removeFromQueue(this.deferred, callback);\n    }\n  }\n\n  private addToQueue(target: Array<number|VoidFunction>, invokeAt: number, callback: VoidFunction) {\n    let insertAtIndex = target.length;\n    for (let i = 0; i < target.length; i += 2) {\n      const invokeQueuedCallbackAt = target[i] as number;\n      if (invokeQueuedCallbackAt > invokeAt) {\n        // We've reached a first timer that is scheduled\n        // for a later time than what we are trying to insert.\n        // This is the location at which we need to insert,\n        // no need to iterate further.\n        insertAtIndex = i;\n        break;\n      }\n    }\n    arrayInsert2(target, insertAtIndex, invokeAt, callback);\n  }\n\n  private removeFromQueue(target: Array<number|VoidFunction>, callback: VoidFunction) {\n    let index = -1;\n    for (let i = 0; i < target.length; i += 2) {\n      const queuedCallback = target[i + 1];\n      if (queuedCallback === callback) {\n        index = i;\n        break;\n      }\n    }\n    if (index > -1) {\n      // Remove 2 elements: a timestamp slot and\n      // the following slot with a callback function.\n      arraySplice(target, index, 2);\n    }\n    return index;\n  }\n\n  private scheduleTimer() {\n    const callback = () => {\n      clearTimeout(this.timeoutId!);\n      this.timeoutId = null;\n\n      this.executingCallbacks = true;\n\n      // Invoke callbacks that were scheduled to run\n      // before the current time.\n      let now = Date.now();\n      let lastCallbackIndex: number|null = null;\n      for (let i = 0; i < this.current.length; i += 2) {\n        const invokeAt = this.current[i] as number;\n        const callback = this.current[i + 1] as VoidFunction;\n        if (invokeAt <= now) {\n          callback();\n          // Point at the invoked callback function, which is located\n          // after the timestamp.\n          lastCallbackIndex = i + 1;\n        } else {\n          // We've reached a timer that should not be invoked yet.\n          break;\n        }\n      }\n      if (lastCallbackIndex !== null) {\n        // If last callback index is `null` - no callbacks were invoked,\n        // so no cleanup is needed. Otherwise, remove invoked callbacks\n        // from the queue.\n        arraySplice(this.current, 0, lastCallbackIndex + 1);\n      }\n\n      this.executingCallbacks = false;\n\n      // If there are any callbacks added during an invocation\n      // of the current ones - move them over to the \"current\"\n      // queue.\n      if (this.deferred.length > 0) {\n        for (let i = 0; i < this.deferred.length; i += 2) {\n          const invokeAt = this.deferred[i] as number;\n          const callback = this.deferred[i + 1] as VoidFunction;\n          this.addToQueue(this.current, invokeAt, callback);\n        }\n        this.deferred.length = 0;\n      }\n      this.scheduleTimer();\n    };\n\n    // Avoid running timer callbacks more than once per\n    // average frame duration. This is needed for better\n    // batching and to avoid kicking off excessive change\n    // detection cycles.\n    const FRAME_DURATION_MS = 16;  // 1000ms / 60fps\n\n    if (this.current.length > 0) {\n      const now = Date.now();\n      // First element in the queue points at the timestamp\n      // of the first (earliest) event.\n      const invokeAt = this.current[0] as number;\n      if (!this.timeoutId ||\n          // Reschedule a timer in case a queue contains an item with\n          // an earlier timestamp and the delta is more than an average\n          // frame duration.\n          (this.invokeTimerAt && (this.invokeTimerAt - invokeAt > FRAME_DURATION_MS))) {\n        if (this.timeoutId !== null) {\n          // There was a timeout already, but an earlier event was added\n          // into the queue. In this case we drop an old timer and setup\n          // a new one with an updated (smaller) timeout.\n          clearTimeout(this.timeoutId);\n          this.timeoutId = null;\n        }\n        const timeout = Math.max(invokeAt - now, FRAME_DURATION_MS);\n        this.invokeTimerAt = invokeAt;\n        this.timeoutId = setTimeout(callback, timeout) as unknown as number;\n      }\n    }\n  }\n\n  ngOnDestroy() {\n    if (this.timeoutId !== null) {\n      clearTimeout(this.timeoutId);\n      this.timeoutId = null;\n    }\n    this.current.length = 0;\n    this.deferred.length = 0;\n  }\n\n  /** @nocollapse */\n  static ɵprov = /** @pureOrBreakMyCode */ ɵɵdefineInjectable({\n    token: TimerScheduler,\n    providedIn: 'root',\n    factory: () => new TimerScheduler(),\n  });\n}\n"]}
|
|
@@ -0,0 +1,134 @@
|
|
|
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
|
+
import { assertIndexInDeclRange } from '../render3/assert';
|
|
9
|
+
import { HEADER_OFFSET, TVIEW } from '../render3/interfaces/view';
|
|
10
|
+
import { getTNode, removeLViewOnDestroy, storeLViewOnDestroy } from '../render3/util/view_utils';
|
|
11
|
+
import { assertEqual, throwError } from '../util/assert';
|
|
12
|
+
import { DeferBlockState, DeferDependenciesLoadingState, LOADING_AFTER_SLOT, MINIMUM_SLOT } from './interfaces';
|
|
13
|
+
/**
|
|
14
|
+
* Wraps a given callback into a logic that registers a cleanup function
|
|
15
|
+
* in the LView cleanup slot, to be invoked when an LView is destroyed.
|
|
16
|
+
*/
|
|
17
|
+
export function wrapWithLViewCleanup(callback, lView, cleanup) {
|
|
18
|
+
const wrappedCallback = () => {
|
|
19
|
+
callback();
|
|
20
|
+
removeLViewOnDestroy(lView, cleanup);
|
|
21
|
+
};
|
|
22
|
+
storeLViewOnDestroy(lView, cleanup);
|
|
23
|
+
return wrappedCallback;
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Calculates a data slot index for defer block info (either static or
|
|
27
|
+
* instance-specific), given an index of a defer instruction.
|
|
28
|
+
*/
|
|
29
|
+
export function getDeferBlockDataIndex(deferBlockIndex) {
|
|
30
|
+
// Instance state is located at the *next* position
|
|
31
|
+
// after the defer block slot in an LView or TView.data.
|
|
32
|
+
return deferBlockIndex + 1;
|
|
33
|
+
}
|
|
34
|
+
/** Retrieves a defer block state from an LView, given a TNode that represents a block. */
|
|
35
|
+
export function getLDeferBlockDetails(lView, tNode) {
|
|
36
|
+
const tView = lView[TVIEW];
|
|
37
|
+
const slotIndex = getDeferBlockDataIndex(tNode.index);
|
|
38
|
+
ngDevMode && assertIndexInDeclRange(tView, slotIndex);
|
|
39
|
+
return lView[slotIndex];
|
|
40
|
+
}
|
|
41
|
+
/** Stores a defer block instance state in LView. */
|
|
42
|
+
export function setLDeferBlockDetails(lView, deferBlockIndex, lDetails) {
|
|
43
|
+
const tView = lView[TVIEW];
|
|
44
|
+
const slotIndex = getDeferBlockDataIndex(deferBlockIndex);
|
|
45
|
+
ngDevMode && assertIndexInDeclRange(tView, slotIndex);
|
|
46
|
+
lView[slotIndex] = lDetails;
|
|
47
|
+
}
|
|
48
|
+
/** Retrieves static info about a defer block, given a TView and a TNode that represents a block. */
|
|
49
|
+
export function getTDeferBlockDetails(tView, tNode) {
|
|
50
|
+
const slotIndex = getDeferBlockDataIndex(tNode.index);
|
|
51
|
+
ngDevMode && assertIndexInDeclRange(tView, slotIndex);
|
|
52
|
+
return tView.data[slotIndex];
|
|
53
|
+
}
|
|
54
|
+
/** Stores a defer block static info in `TView.data`. */
|
|
55
|
+
export function setTDeferBlockDetails(tView, deferBlockIndex, deferBlockConfig) {
|
|
56
|
+
const slotIndex = getDeferBlockDataIndex(deferBlockIndex);
|
|
57
|
+
ngDevMode && assertIndexInDeclRange(tView, slotIndex);
|
|
58
|
+
tView.data[slotIndex] = deferBlockConfig;
|
|
59
|
+
}
|
|
60
|
+
export function getTemplateIndexForState(newState, hostLView, tNode) {
|
|
61
|
+
const tView = hostLView[TVIEW];
|
|
62
|
+
const tDetails = getTDeferBlockDetails(tView, tNode);
|
|
63
|
+
switch (newState) {
|
|
64
|
+
case DeferBlockState.Complete:
|
|
65
|
+
return tDetails.primaryTmplIndex;
|
|
66
|
+
case DeferBlockState.Loading:
|
|
67
|
+
return tDetails.loadingTmplIndex;
|
|
68
|
+
case DeferBlockState.Error:
|
|
69
|
+
return tDetails.errorTmplIndex;
|
|
70
|
+
case DeferBlockState.Placeholder:
|
|
71
|
+
return tDetails.placeholderTmplIndex;
|
|
72
|
+
default:
|
|
73
|
+
ngDevMode && throwError(`Unexpected defer block state: ${newState}`);
|
|
74
|
+
return null;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
/**
|
|
78
|
+
* Returns a minimum amount of time that a given state should be rendered for,
|
|
79
|
+
* taking into account `minimum` parameter value. If the `minimum` value is
|
|
80
|
+
* not specified - returns `null`.
|
|
81
|
+
*/
|
|
82
|
+
export function getMinimumDurationForState(tDetails, currentState) {
|
|
83
|
+
if (currentState === DeferBlockState.Placeholder) {
|
|
84
|
+
return tDetails.placeholderBlockConfig?.[MINIMUM_SLOT] ?? null;
|
|
85
|
+
}
|
|
86
|
+
else if (currentState === DeferBlockState.Loading) {
|
|
87
|
+
return tDetails.loadingBlockConfig?.[MINIMUM_SLOT] ?? null;
|
|
88
|
+
}
|
|
89
|
+
return null;
|
|
90
|
+
}
|
|
91
|
+
/** Retrieves the value of the `after` parameter on the @loading block. */
|
|
92
|
+
export function getLoadingBlockAfter(tDetails) {
|
|
93
|
+
return tDetails.loadingBlockConfig?.[LOADING_AFTER_SLOT] ?? null;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Adds downloaded dependencies into a directive or a pipe registry,
|
|
97
|
+
* making sure that a dependency doesn't yet exist in the registry.
|
|
98
|
+
*/
|
|
99
|
+
export function addDepsToRegistry(currentDeps, newDeps) {
|
|
100
|
+
if (!currentDeps || currentDeps.length === 0) {
|
|
101
|
+
return newDeps;
|
|
102
|
+
}
|
|
103
|
+
const currentDepSet = new Set(currentDeps);
|
|
104
|
+
for (const dep of newDeps) {
|
|
105
|
+
currentDepSet.add(dep);
|
|
106
|
+
}
|
|
107
|
+
// If `currentDeps` is the same length, there were no new deps and can
|
|
108
|
+
// return the original array.
|
|
109
|
+
return (currentDeps.length === currentDepSet.size) ? currentDeps : Array.from(currentDepSet);
|
|
110
|
+
}
|
|
111
|
+
/** Retrieves a TNode that represents main content of a defer block. */
|
|
112
|
+
export function getPrimaryBlockTNode(tView, tDetails) {
|
|
113
|
+
const adjustedIndex = tDetails.primaryTmplIndex + HEADER_OFFSET;
|
|
114
|
+
return getTNode(tView, adjustedIndex);
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* Asserts whether all dependencies for a defer block are loaded.
|
|
118
|
+
* Always run this function (in dev mode) before rendering a defer
|
|
119
|
+
* block in completed state.
|
|
120
|
+
*/
|
|
121
|
+
export function assertDeferredDependenciesLoaded(tDetails) {
|
|
122
|
+
assertEqual(tDetails.loadingState, DeferDependenciesLoadingState.COMPLETE, 'Expecting all deferred dependencies to be loaded.');
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Determines if a given value matches the expected structure of a defer block
|
|
126
|
+
*
|
|
127
|
+
* We can safely rely on the primaryTmplIndex because every defer block requires
|
|
128
|
+
* that a primary template exists. All the other template options are optional.
|
|
129
|
+
*/
|
|
130
|
+
export function isTDeferBlockDetails(value) {
|
|
131
|
+
return (typeof value === 'object') &&
|
|
132
|
+
(typeof value.primaryTmplIndex === 'number');
|
|
133
|
+
}
|
|
134
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/defer/utils.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 {assertIndexInDeclRange} from '../render3/assert';\nimport {DependencyDef} from '../render3/interfaces/definition';\nimport {TContainerNode, TNode} from '../render3/interfaces/node';\nimport {HEADER_OFFSET, LView, TVIEW, TView} from '../render3/interfaces/view';\nimport {getTNode, removeLViewOnDestroy, storeLViewOnDestroy} from '../render3/util/view_utils';\nimport {assertEqual, throwError} from '../util/assert';\n\nimport {DeferBlockState, DeferDependenciesLoadingState, LDeferBlockDetails, LOADING_AFTER_SLOT, MINIMUM_SLOT, TDeferBlockDetails} from './interfaces';\n\n/**\n * Wraps a given callback into a logic that registers a cleanup function\n * in the LView cleanup slot, to be invoked when an LView is destroyed.\n */\nexport function wrapWithLViewCleanup(\n    callback: VoidFunction, lView: LView, cleanup: VoidFunction): VoidFunction {\n  const wrappedCallback = () => {\n    callback();\n    removeLViewOnDestroy(lView, cleanup);\n  };\n  storeLViewOnDestroy(lView, cleanup);\n  return wrappedCallback;\n}\n\n/**\n * Calculates a data slot index for defer block info (either static or\n * instance-specific), given an index of a defer instruction.\n */\nexport function getDeferBlockDataIndex(deferBlockIndex: number) {\n  // Instance state is located at the *next* position\n  // after the defer block slot in an LView or TView.data.\n  return deferBlockIndex + 1;\n}\n\n/** Retrieves a defer block state from an LView, given a TNode that represents a block. */\nexport function getLDeferBlockDetails(lView: LView, tNode: TNode): LDeferBlockDetails {\n  const tView = lView[TVIEW];\n  const slotIndex = getDeferBlockDataIndex(tNode.index);\n  ngDevMode && assertIndexInDeclRange(tView, slotIndex);\n  return lView[slotIndex];\n}\n\n/** Stores a defer block instance state in LView. */\nexport function setLDeferBlockDetails(\n    lView: LView, deferBlockIndex: number, lDetails: LDeferBlockDetails) {\n  const tView = lView[TVIEW];\n  const slotIndex = getDeferBlockDataIndex(deferBlockIndex);\n  ngDevMode && assertIndexInDeclRange(tView, slotIndex);\n  lView[slotIndex] = lDetails;\n}\n\n/** Retrieves static info about a defer block, given a TView and a TNode that represents a block. */\nexport function getTDeferBlockDetails(tView: TView, tNode: TNode): TDeferBlockDetails {\n  const slotIndex = getDeferBlockDataIndex(tNode.index);\n  ngDevMode && assertIndexInDeclRange(tView, slotIndex);\n  return tView.data[slotIndex] as TDeferBlockDetails;\n}\n\n/** Stores a defer block static info in `TView.data`. */\nexport function setTDeferBlockDetails(\n    tView: TView, deferBlockIndex: number, deferBlockConfig: TDeferBlockDetails) {\n  const slotIndex = getDeferBlockDataIndex(deferBlockIndex);\n  ngDevMode && assertIndexInDeclRange(tView, slotIndex);\n  tView.data[slotIndex] = deferBlockConfig;\n}\n\nexport function getTemplateIndexForState(\n    newState: DeferBlockState, hostLView: LView, tNode: TNode): number|null {\n  const tView = hostLView[TVIEW];\n  const tDetails = getTDeferBlockDetails(tView, tNode);\n\n  switch (newState) {\n    case DeferBlockState.Complete:\n      return tDetails.primaryTmplIndex;\n    case DeferBlockState.Loading:\n      return tDetails.loadingTmplIndex;\n    case DeferBlockState.Error:\n      return tDetails.errorTmplIndex;\n    case DeferBlockState.Placeholder:\n      return tDetails.placeholderTmplIndex;\n    default:\n      ngDevMode && throwError(`Unexpected defer block state: ${newState}`);\n      return null;\n  }\n}\n\n/**\n * Returns a minimum amount of time that a given state should be rendered for,\n * taking into account `minimum` parameter value. If the `minimum` value is\n * not specified - returns `null`.\n */\nexport function getMinimumDurationForState(\n    tDetails: TDeferBlockDetails, currentState: DeferBlockState): number|null {\n  if (currentState === DeferBlockState.Placeholder) {\n    return tDetails.placeholderBlockConfig?.[MINIMUM_SLOT] ?? null;\n  } else if (currentState === DeferBlockState.Loading) {\n    return tDetails.loadingBlockConfig?.[MINIMUM_SLOT] ?? null;\n  }\n  return null;\n}\n\n/** Retrieves the value of the `after` parameter on the @loading block. */\nexport function getLoadingBlockAfter(tDetails: TDeferBlockDetails): number|null {\n  return tDetails.loadingBlockConfig?.[LOADING_AFTER_SLOT] ?? null;\n}\n\n/**\n * Adds downloaded dependencies into a directive or a pipe registry,\n * making sure that a dependency doesn't yet exist in the registry.\n */\nexport function addDepsToRegistry<T extends DependencyDef[]>(currentDeps: T|null, newDeps: T): T {\n  if (!currentDeps || currentDeps.length === 0) {\n    return newDeps;\n  }\n\n  const currentDepSet = new Set(currentDeps);\n  for (const dep of newDeps) {\n    currentDepSet.add(dep);\n  }\n\n  // If `currentDeps` is the same length, there were no new deps and can\n  // return the original array.\n  return (currentDeps.length === currentDepSet.size) ? currentDeps : Array.from(currentDepSet) as T;\n}\n\n/** Retrieves a TNode that represents main content of a defer block. */\nexport function getPrimaryBlockTNode(tView: TView, tDetails: TDeferBlockDetails): TContainerNode {\n  const adjustedIndex = tDetails.primaryTmplIndex + HEADER_OFFSET;\n  return getTNode(tView, adjustedIndex) as TContainerNode;\n}\n\n/**\n * Asserts whether all dependencies for a defer block are loaded.\n * Always run this function (in dev mode) before rendering a defer\n * block in completed state.\n */\nexport function assertDeferredDependenciesLoaded(tDetails: TDeferBlockDetails) {\n  assertEqual(\n      tDetails.loadingState, DeferDependenciesLoadingState.COMPLETE,\n      'Expecting all deferred dependencies to be loaded.');\n}\n\n/**\n * Determines if a given value matches the expected structure of a defer block\n *\n * We can safely rely on the primaryTmplIndex because every defer block requires\n * that a primary template exists. All the other template options are optional.\n */\nexport function isTDeferBlockDetails(value: unknown): value is TDeferBlockDetails {\n  return (typeof value === 'object') &&\n      (typeof (value as TDeferBlockDetails).primaryTmplIndex === 'number');\n}\n"]}
|
package/esm2022/src/errors.mjs
CHANGED
|
@@ -46,4 +46,4 @@ export function formatRuntimeError(code, message) {
|
|
|
46
46
|
}
|
|
47
47
|
return errorMessage;
|
|
48
48
|
}
|
|
49
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
49
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"errors.js","sourceRoot":"","sources":["../../../../../../packages/core/src/errors.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAC,2BAA2B,EAAC,MAAM,0BAA0B,CAAC;AAiHrE;;;;;;;;;;;;;;;GAeG;AACH,MAAM,OAAO,YAAkD,SAAQ,KAAK;IAC1E,YAAmB,IAAO,EAAE,OAA0B;QACpD,KAAK,CAAC,kBAAkB,CAAI,IAAI,EAAE,OAAO,CAAC,CAAC,CAAC;QAD3B,SAAI,GAAJ,IAAI,CAAG;IAE1B,CAAC;CACF;AAED;;;GAGG;AACH,MAAM,UAAU,kBAAkB,CAC9B,IAAO,EAAE,OAA0B;IACrC,+FAA+F;IAC/F,2DAA2D;IAC3D,kDAAkD;IAClD,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CAAC;IAExC,IAAI,YAAY,GAAG,GAAG,QAAQ,GAAG,OAAO,CAAC,CAAC,CAAC,IAAI,GAAG,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC;IAEjE,IAAI,SAAS,IAAI,IAAI,GAAG,CAAC,EAAE;QACzB,MAAM,kBAAkB,GAAG,CAAC,YAAY,CAAC,KAAK,CAAC,YAAY,CAAC,CAAC;QAC7D,MAAM,SAAS,GAAG,kBAAkB,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC;QAChD,YAAY;YACR,GAAG,YAAY,GAAG,SAAS,iBAAiB,2BAA2B,IAAI,QAAQ,EAAE,CAAC;KAC3F;IACD,OAAO,YAAY,CAAC;AACtB,CAAC","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 {ERROR_DETAILS_PAGE_BASE_URL} from './error_details_base_url';\n\n/**\n * The list of error codes used in runtime code of the `core` package.\n * Reserved error code range: 100-999.\n *\n * Note: the minus sign denotes the fact that a particular code has a detailed guide on\n * angular.io. This extra annotation is needed to avoid introducing a separate set to store\n * error codes which have guides, which might leak into runtime code.\n *\n * Full list of available error guides can be found at https://angular.io/errors.\n *\n * Error code ranges per package:\n *  - core (this package): 100-999\n *  - forms: 1000-1999\n *  - common: 2000-2999\n *  - animations: 3000-3999\n *  - router: 4000-4999\n *  - platform-browser: 5000-5500\n */\nexport const enum RuntimeErrorCode {\n  // Change Detection Errors\n  EXPRESSION_CHANGED_AFTER_CHECKED = -100,\n  RECURSIVE_APPLICATION_REF_TICK = 101,\n  RECURSIVE_APPLICATION_RENDER = 102,\n\n  // Dependency Injection Errors\n  CYCLIC_DI_DEPENDENCY = -200,\n  PROVIDER_NOT_FOUND = -201,\n  INVALID_FACTORY_DEPENDENCY = 202,\n  MISSING_INJECTION_CONTEXT = -203,\n  INVALID_INJECTION_TOKEN = 204,\n  INJECTOR_ALREADY_DESTROYED = 205,\n  PROVIDER_IN_WRONG_CONTEXT = 207,\n  MISSING_INJECTION_TOKEN = 208,\n  INVALID_MULTI_PROVIDER = -209,\n  MISSING_DOCUMENT = 210,\n\n  // Template Errors\n  MULTIPLE_COMPONENTS_MATCH = -300,\n  EXPORT_NOT_FOUND = -301,\n  PIPE_NOT_FOUND = -302,\n  UNKNOWN_BINDING = 303,\n  UNKNOWN_ELEMENT = 304,\n  TEMPLATE_STRUCTURE_ERROR = 305,\n  INVALID_EVENT_BINDING = 306,\n  HOST_DIRECTIVE_UNRESOLVABLE = 307,\n  HOST_DIRECTIVE_NOT_STANDALONE = 308,\n  DUPLICATE_DIRECTITVE = 309,\n  HOST_DIRECTIVE_COMPONENT = 310,\n  HOST_DIRECTIVE_UNDEFINED_BINDING = 311,\n  HOST_DIRECTIVE_CONFLICTING_ALIAS = 312,\n  MULTIPLE_MATCHING_PIPES = 313,\n\n  // Bootstrap Errors\n  MULTIPLE_PLATFORMS = 400,\n  PLATFORM_NOT_FOUND = 401,\n  MISSING_REQUIRED_INJECTABLE_IN_BOOTSTRAP = 402,\n  BOOTSTRAP_COMPONENTS_NOT_FOUND = -403,\n  PLATFORM_ALREADY_DESTROYED = 404,\n  ASYNC_INITIALIZERS_STILL_RUNNING = 405,\n  APPLICATION_REF_ALREADY_DESTROYED = 406,\n  RENDERER_NOT_FOUND = 407,\n\n  // Hydration Errors\n  HYDRATION_NODE_MISMATCH = -500,\n  HYDRATION_MISSING_SIBLINGS = -501,\n  HYDRATION_MISSING_NODE = -502,\n  UNSUPPORTED_PROJECTION_DOM_NODES = -503,\n  INVALID_SKIP_HYDRATION_HOST = -504,\n  MISSING_HYDRATION_ANNOTATIONS = -505,\n  HYDRATION_STABLE_TIMEDOUT = -506,\n  MISSING_SSR_CONTENT_INTEGRITY_MARKER = -507,\n\n  // Signal Errors\n  SIGNAL_WRITE_FROM_ILLEGAL_CONTEXT = 600,\n  REQUIRE_SYNC_WITHOUT_SYNC_EMIT = 601,\n  ASSERTION_NOT_INSIDE_REACTIVE_CONTEXT = -602,\n\n  // Styling Errors\n\n  // Declarations Errors\n\n  // i18n Errors\n  INVALID_I18N_STRUCTURE = 700,\n  MISSING_LOCALE_DATA = 701,\n\n  // standalone errors\n  IMPORT_PROVIDERS_FROM_STANDALONE = 800,\n\n  // JIT Compilation Errors\n  // Other\n  INVALID_DIFFER_INPUT = 900,\n  NO_SUPPORTING_DIFFER_FACTORY = 901,\n  VIEW_ALREADY_ATTACHED = 902,\n  INVALID_INHERITANCE = 903,\n  UNSAFE_VALUE_IN_RESOURCE_URL = 904,\n  UNSAFE_VALUE_IN_SCRIPT = 905,\n  MISSING_GENERATED_DEF = 906,\n  TYPE_IS_NOT_STANDALONE = 907,\n  MISSING_ZONEJS = 908,\n  UNEXPECTED_ZONE_STATE = 909,\n  UNSAFE_IFRAME_ATTRS = -910,\n  VIEW_ALREADY_DESTROYED = 911,\n  COMPONENT_ID_COLLISION = -912,\n  IMAGE_PERFORMANCE_WARNING = -913,\n\n  // Runtime dependency tracker errors\n  RUNTIME_DEPS_INVALID_IMPORTED_TYPE = 1000,\n  RUNTIME_DEPS_ORPHAN_COMPONENT = 1001,\n}\n\n\n/**\n * Class that represents a runtime error.\n * Formats and outputs the error message in a consistent way.\n *\n * Example:\n * ```\n *  throw new RuntimeError(\n *    RuntimeErrorCode.INJECTOR_ALREADY_DESTROYED,\n *    ngDevMode && 'Injector has already been destroyed.');\n * ```\n *\n * Note: the `message` argument contains a descriptive error message as a string in development\n * mode (when the `ngDevMode` is defined). In production mode (after tree-shaking pass), the\n * `message` argument becomes `false`, thus we account for it in the typings and the runtime\n * logic.\n */\nexport class RuntimeError<T extends number = RuntimeErrorCode> extends Error {\n  constructor(public code: T, message: null|false|string) {\n    super(formatRuntimeError<T>(code, message));\n  }\n}\n\n/**\n * Called to format a runtime error.\n * See additional info on the `message` argument type in the `RuntimeError` class description.\n */\nexport function formatRuntimeError<T extends number = RuntimeErrorCode>(\n    code: T, message: null|false|string): string {\n  // Error code might be a negative number, which is a special marker that instructs the logic to\n  // generate a link to the error details page on angular.io.\n  // We also prepend `0` to non-compile-time errors.\n  const fullCode = `NG0${Math.abs(code)}`;\n\n  let errorMessage = `${fullCode}${message ? ': ' + message : ''}`;\n\n  if (ngDevMode && code < 0) {\n    const addPeriodSeparator = !errorMessage.match(/[.,;!?\\n]$/);\n    const separator = addPeriodSeparator ? '.' : '';\n    errorMessage =\n        `${errorMessage}${separator} Find more at ${ERROR_DETAILS_PAGE_BASE_URL}/${fullCode}`;\n  }\n  return errorMessage;\n}\n"]}
|