@angular/core 17.0.0-next.8 → 17.0.0-rc.1

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 (68) hide show
  1. package/esm2022/src/application_init.mjs +3 -2
  2. package/esm2022/src/application_ref.mjs +4 -3
  3. package/esm2022/src/application_tokens.mjs +1 -11
  4. package/esm2022/src/core.mjs +2 -2
  5. package/esm2022/src/core_private_export.mjs +5 -4
  6. package/esm2022/src/core_render3_private_export.mjs +2 -2
  7. package/esm2022/src/debug/debug_node.mjs +5 -9
  8. package/esm2022/src/defer/cleanup.mjs +40 -0
  9. package/esm2022/src/defer/discovery.mjs +47 -0
  10. package/esm2022/src/defer/dom_triggers.mjs +256 -0
  11. package/esm2022/src/defer/idle_scheduler.mjs +112 -0
  12. package/esm2022/src/defer/instructions.mjs +662 -0
  13. package/esm2022/src/defer/interfaces.mjs +81 -0
  14. package/esm2022/src/defer/timer_scheduler.mjs +201 -0
  15. package/esm2022/src/defer/utils.mjs +122 -0
  16. package/esm2022/src/di/r3_injector.mjs +6 -1
  17. package/esm2022/src/errors.mjs +1 -1
  18. package/esm2022/src/hydration/api.mjs +2 -4
  19. package/esm2022/src/hydration/skip_hydration.mjs +7 -7
  20. package/esm2022/src/hydration/utils.mjs +2 -2
  21. package/esm2022/src/linker/view_container_ref.mjs +4 -6
  22. package/esm2022/src/metadata/directives.mjs +1 -1
  23. package/esm2022/src/render/api.mjs +1 -1
  24. package/esm2022/src/render3/after_render_hooks.mjs +55 -21
  25. package/esm2022/src/render3/debug/injector_profiler.mjs +26 -8
  26. package/esm2022/src/render3/index.mjs +3 -2
  27. package/esm2022/src/render3/instructions/all.mjs +2 -2
  28. package/esm2022/src/render3/instructions/change_detection.mjs +24 -3
  29. package/esm2022/src/render3/instructions/control_flow.mjs +25 -15
  30. package/esm2022/src/render3/interfaces/container.mjs +1 -4
  31. package/esm2022/src/render3/interfaces/definition.mjs +2 -4
  32. package/esm2022/src/render3/interfaces/i18n.mjs +1 -4
  33. package/esm2022/src/render3/interfaces/injector.mjs +1 -4
  34. package/esm2022/src/render3/interfaces/node.mjs +1 -4
  35. package/esm2022/src/render3/interfaces/projection.mjs +2 -4
  36. package/esm2022/src/render3/interfaces/query.mjs +2 -4
  37. package/esm2022/src/render3/interfaces/renderer.mjs +2 -4
  38. package/esm2022/src/render3/interfaces/renderer_dom.mjs +2 -4
  39. package/esm2022/src/render3/interfaces/view.mjs +1 -4
  40. package/esm2022/src/render3/list_reconciliation.mjs +58 -34
  41. package/esm2022/src/render3/reactive_lview_consumer.mjs +2 -7
  42. package/esm2022/src/render3/util/injector_discovery_utils.mjs +26 -2
  43. package/esm2022/src/render3/util/view_utils.mjs +1 -4
  44. package/esm2022/src/version.mjs +1 -1
  45. package/esm2022/testing/src/defer.mjs +6 -3
  46. package/esm2022/testing/src/logger.mjs +3 -3
  47. package/fesm2022/core.mjs +1015 -932
  48. package/fesm2022/core.mjs.map +1 -1
  49. package/fesm2022/primitives/signals.mjs +1 -1
  50. package/fesm2022/rxjs-interop.mjs +1 -1
  51. package/fesm2022/testing.mjs +6 -3
  52. package/fesm2022/testing.mjs.map +1 -1
  53. package/index.d.ts +54 -18
  54. package/package.json +1 -1
  55. package/primitives/signals/index.d.ts +1 -1
  56. package/rxjs-interop/index.d.ts +1 -1
  57. package/schematics/migrations/block-template-entities/bundle.js +1051 -467
  58. package/schematics/migrations/block-template-entities/bundle.js.map +4 -4
  59. package/schematics/migrations/compiler-options/bundle.js +13 -13
  60. package/schematics/migrations/transfer-state/bundle.js +13 -13
  61. package/schematics/ng-generate/control-flow-migration/bundle.js +1284 -612
  62. package/schematics/ng-generate/control-flow-migration/bundle.js.map +4 -4
  63. package/schematics/ng-generate/standalone-migration/bundle.js +1300 -708
  64. package/schematics/ng-generate/standalone-migration/bundle.js.map +4 -4
  65. package/testing/index.d.ts +1 -1
  66. package/esm2022/src/render3/instructions/defer.mjs +0 -1225
  67. package/esm2022/src/render3/instructions/defer_events.mjs +0 -174
  68. package/esm2022/src/render3/interfaces/defer.mjs +0 -79
@@ -0,0 +1,81 @@
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
+ export const TRIGGER_CLEANUP_FNS = 4;
63
+ export const PREFETCH_TRIGGER_CLEANUP_FNS = 5;
64
+ /**
65
+ * Options for configuring defer blocks behavior.
66
+ * @publicApi
67
+ * @developerPreview
68
+ */
69
+ export var DeferBlockBehavior;
70
+ (function (DeferBlockBehavior) {
71
+ /**
72
+ * Manual triggering mode for defer blocks. Provides control over when defer blocks render
73
+ * and which state they render. This is the default behavior in test environments.
74
+ */
75
+ DeferBlockBehavior[DeferBlockBehavior["Manual"] = 0] = "Manual";
76
+ /**
77
+ * Playthrough mode for defer blocks. This mode behaves like defer blocks would in a browser.
78
+ */
79
+ DeferBlockBehavior[DeferBlockBehavior["Playthrough"] = 1] = "Playthrough";
80
+ })(DeferBlockBehavior || (DeferBlockBehavior = {}));
81
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"interfaces.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/defer/interfaces.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAyBH;;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;AAC1C,MAAM,CAAC,MAAM,mBAAmB,GAAG,CAAC,CAAC;AACrC,MAAM,CAAC,MAAM,4BAA4B,GAAG,CAAC,CAAC;AAmD9C;;;;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 * Defines types of defer block triggers.\n */\nexport const enum TriggerType {\n  /**\n   * Represents regular triggers (e.g. `@defer (on idle) { ... }`).\n   */\n  Regular,\n\n  /**\n   * Represents prefetch triggers (e.g. `@defer (prefetch on idle) { ... }`).\n   */\n  Prefetch,\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;\nexport const TRIGGER_CLEANUP_FNS = 4;\nexport const PREFETCH_TRIGGER_CLEANUP_FNS = 5;\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   * List of cleanup functions for regular triggers.\n   */\n  [TRIGGER_CLEANUP_FNS]: VoidFunction[]|null;\n\n  /**\n   * List of cleanup functions for prefetch triggers.\n   */\n  [PREFETCH_TRIGGER_CLEANUP_FNS]: 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,201 @@
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
+ /**
12
+ * Returns a function that captures a provided delay.
13
+ * Invoking the returned function schedules a trigger.
14
+ */
15
+ export function onTimer(delay) {
16
+ return (callback, lView) => scheduleTimerTrigger(delay, callback, lView);
17
+ }
18
+ /**
19
+ * Schedules a callback to be invoked after a given timeout.
20
+ *
21
+ * @param delay A number of ms to wait until firing a callback.
22
+ * @param callback A function to be invoked after a timeout.
23
+ * @param lView LView that hosts an instance of a defer block.
24
+ */
25
+ export function scheduleTimerTrigger(delay, callback, lView) {
26
+ const injector = lView[INJECTOR];
27
+ const scheduler = injector.get(TimerScheduler);
28
+ const cleanupFn = () => scheduler.remove(callback);
29
+ scheduler.add(delay, callback);
30
+ return cleanupFn;
31
+ }
32
+ /**
33
+ * Helper service to schedule `setTimeout`s for batches of defer blocks,
34
+ * to avoid calling `setTimeout` for each defer block (e.g. if defer blocks
35
+ * are created inside a for loop).
36
+ */
37
+ export class TimerScheduler {
38
+ constructor() {
39
+ // Indicates whether current callbacks are being invoked.
40
+ this.executingCallbacks = false;
41
+ // Currently scheduled `setTimeout` id.
42
+ this.timeoutId = null;
43
+ // When currently scheduled timer would fire.
44
+ this.invokeTimerAt = null;
45
+ // List of callbacks to be invoked.
46
+ // For each callback we also store a timestamp on when the callback
47
+ // should be invoked. We store timestamps and callback functions
48
+ // in a flat array to avoid creating new objects for each entry.
49
+ // [timestamp1, callback1, timestamp2, callback2, ...]
50
+ this.current = [];
51
+ // List of callbacks collected while invoking current set of callbacks.
52
+ // Those callbacks are added to the "current" queue at the end of
53
+ // the current callback invocation. The shape of this list is the same
54
+ // as the shape of the `current` list.
55
+ this.deferred = [];
56
+ }
57
+ add(delay, callback) {
58
+ const target = this.executingCallbacks ? this.deferred : this.current;
59
+ this.addToQueue(target, Date.now() + delay, callback);
60
+ this.scheduleTimer();
61
+ }
62
+ remove(callback) {
63
+ const { current, deferred } = this;
64
+ const callbackIndex = this.removeFromQueue(current, callback);
65
+ if (callbackIndex === -1) {
66
+ // Try cleaning up deferred queue only in case
67
+ // we didn't find a callback in the "current" queue.
68
+ this.removeFromQueue(deferred, callback);
69
+ }
70
+ // If the last callback was removed and there is a pending timeout - cancel it.
71
+ if (current.length === 0 && deferred.length === 0) {
72
+ this.clearTimeout();
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
+ this.clearTimeout();
109
+ this.executingCallbacks = true;
110
+ // Clone the current state of the queue, since it might be altered
111
+ // as we invoke callbacks.
112
+ const current = [...this.current];
113
+ // Invoke callbacks that were scheduled to run before the current time.
114
+ const now = Date.now();
115
+ for (let i = 0; i < current.length; i += 2) {
116
+ const invokeAt = current[i];
117
+ const callback = current[i + 1];
118
+ if (invokeAt <= now) {
119
+ callback();
120
+ }
121
+ else {
122
+ // We've reached a timer that should not be invoked yet.
123
+ break;
124
+ }
125
+ }
126
+ // The state of the queue might've changed after callbacks invocation,
127
+ // run the cleanup logic based on the *current* state of the queue.
128
+ let lastCallbackIndex = -1;
129
+ for (let i = 0; i < this.current.length; i += 2) {
130
+ const invokeAt = this.current[i];
131
+ if (invokeAt <= now) {
132
+ // Add +1 to account for a callback function that
133
+ // goes after the timestamp in events array.
134
+ lastCallbackIndex = i + 1;
135
+ }
136
+ else {
137
+ // We've reached a timer that should not be invoked yet.
138
+ break;
139
+ }
140
+ }
141
+ if (lastCallbackIndex >= 0) {
142
+ arraySplice(this.current, 0, lastCallbackIndex + 1);
143
+ }
144
+ this.executingCallbacks = false;
145
+ // If there are any callbacks added during an invocation
146
+ // of the current ones - move them over to the "current"
147
+ // queue.
148
+ if (this.deferred.length > 0) {
149
+ for (let i = 0; i < this.deferred.length; i += 2) {
150
+ const invokeAt = this.deferred[i];
151
+ const callback = this.deferred[i + 1];
152
+ this.addToQueue(this.current, invokeAt, callback);
153
+ }
154
+ this.deferred.length = 0;
155
+ }
156
+ this.scheduleTimer();
157
+ };
158
+ // Avoid running timer callbacks more than once per
159
+ // average frame duration. This is needed for better
160
+ // batching and to avoid kicking off excessive change
161
+ // detection cycles.
162
+ const FRAME_DURATION_MS = 16; // 1000ms / 60fps
163
+ if (this.current.length > 0) {
164
+ const now = Date.now();
165
+ // First element in the queue points at the timestamp
166
+ // of the first (earliest) event.
167
+ const invokeAt = this.current[0];
168
+ if (this.timeoutId === null ||
169
+ // Reschedule a timer in case a queue contains an item with
170
+ // an earlier timestamp and the delta is more than an average
171
+ // frame duration.
172
+ (this.invokeTimerAt && (this.invokeTimerAt - invokeAt > FRAME_DURATION_MS))) {
173
+ // There was a timeout already, but an earlier event was added
174
+ // into the queue. In this case we drop an old timer and setup
175
+ // a new one with an updated (smaller) timeout.
176
+ this.clearTimeout();
177
+ const timeout = Math.max(invokeAt - now, FRAME_DURATION_MS);
178
+ this.invokeTimerAt = invokeAt;
179
+ this.timeoutId = setTimeout(callback, timeout);
180
+ }
181
+ }
182
+ }
183
+ clearTimeout() {
184
+ if (this.timeoutId !== null) {
185
+ clearTimeout(this.timeoutId);
186
+ this.timeoutId = null;
187
+ }
188
+ }
189
+ ngOnDestroy() {
190
+ this.clearTimeout();
191
+ this.current.length = 0;
192
+ this.deferred.length = 0;
193
+ }
194
+ /** @nocollapse */
195
+ static { this.ɵprov = ɵɵdefineInjectable({
196
+ token: TimerScheduler,
197
+ providedIn: 'root',
198
+ factory: () => new TimerScheduler(),
199
+ }); }
200
+ }
201
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"timer_scheduler.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/defer/timer_scheduler.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 {ɵɵdefineInjectable} from '../di';\nimport {INJECTOR, LView} from '../render3/interfaces/view';\nimport {arrayInsert2, arraySplice} from '../util/array_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) => scheduleTimerTrigger(delay, callback, lView);\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 */\nexport function scheduleTimerTrigger(delay: number, callback: VoidFunction, lView: LView) {\n  const injector = lView[INJECTOR]!;\n  const scheduler = injector.get(TimerScheduler);\n  const cleanupFn = () => scheduler.remove(callback);\n  scheduler.add(delay, callback);\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 {current, deferred} = this;\n    const callbackIndex = this.removeFromQueue(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(deferred, callback);\n    }\n    // If the last callback was removed and there is a pending timeout - cancel it.\n    if (current.length === 0 && deferred.length === 0) {\n      this.clearTimeout();\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      this.clearTimeout();\n\n      this.executingCallbacks = true;\n\n      // Clone the current state of the queue, since it might be altered\n      // as we invoke callbacks.\n      const current = [...this.current];\n\n      // Invoke callbacks that were scheduled to run before the current time.\n      const now = Date.now();\n      for (let i = 0; i < current.length; i += 2) {\n        const invokeAt = current[i] as number;\n        const callback = current[i + 1] as VoidFunction;\n        if (invokeAt <= now) {\n          callback();\n        } else {\n          // We've reached a timer that should not be invoked yet.\n          break;\n        }\n      }\n      // The state of the queue might've changed after callbacks invocation,\n      // run the cleanup logic based on the *current* state of the queue.\n      let lastCallbackIndex = -1;\n      for (let i = 0; i < this.current.length; i += 2) {\n        const invokeAt = this.current[i] as number;\n        if (invokeAt <= now) {\n          // Add +1 to account for a callback function that\n          // goes after the timestamp in events array.\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 >= 0) {\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 === null ||\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        // 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        this.clearTimeout();\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  private clearTimeout() {\n    if (this.timeoutId !== null) {\n      clearTimeout(this.timeoutId);\n      this.timeoutId = null;\n    }\n  }\n\n  ngOnDestroy() {\n    this.clearTimeout();\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,122 @@
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 } 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
+ * Calculates a data slot index for defer block info (either static or
15
+ * instance-specific), given an index of a defer instruction.
16
+ */
17
+ export function getDeferBlockDataIndex(deferBlockIndex) {
18
+ // Instance state is located at the *next* position
19
+ // after the defer block slot in an LView or TView.data.
20
+ return deferBlockIndex + 1;
21
+ }
22
+ /** Retrieves a defer block state from an LView, given a TNode that represents a block. */
23
+ export function getLDeferBlockDetails(lView, tNode) {
24
+ const tView = lView[TVIEW];
25
+ const slotIndex = getDeferBlockDataIndex(tNode.index);
26
+ ngDevMode && assertIndexInDeclRange(tView, slotIndex);
27
+ return lView[slotIndex];
28
+ }
29
+ /** Stores a defer block instance state in LView. */
30
+ export function setLDeferBlockDetails(lView, deferBlockIndex, lDetails) {
31
+ const tView = lView[TVIEW];
32
+ const slotIndex = getDeferBlockDataIndex(deferBlockIndex);
33
+ ngDevMode && assertIndexInDeclRange(tView, slotIndex);
34
+ lView[slotIndex] = lDetails;
35
+ }
36
+ /** Retrieves static info about a defer block, given a TView and a TNode that represents a block. */
37
+ export function getTDeferBlockDetails(tView, tNode) {
38
+ const slotIndex = getDeferBlockDataIndex(tNode.index);
39
+ ngDevMode && assertIndexInDeclRange(tView, slotIndex);
40
+ return tView.data[slotIndex];
41
+ }
42
+ /** Stores a defer block static info in `TView.data`. */
43
+ export function setTDeferBlockDetails(tView, deferBlockIndex, deferBlockConfig) {
44
+ const slotIndex = getDeferBlockDataIndex(deferBlockIndex);
45
+ ngDevMode && assertIndexInDeclRange(tView, slotIndex);
46
+ tView.data[slotIndex] = deferBlockConfig;
47
+ }
48
+ export function getTemplateIndexForState(newState, hostLView, tNode) {
49
+ const tView = hostLView[TVIEW];
50
+ const tDetails = getTDeferBlockDetails(tView, tNode);
51
+ switch (newState) {
52
+ case DeferBlockState.Complete:
53
+ return tDetails.primaryTmplIndex;
54
+ case DeferBlockState.Loading:
55
+ return tDetails.loadingTmplIndex;
56
+ case DeferBlockState.Error:
57
+ return tDetails.errorTmplIndex;
58
+ case DeferBlockState.Placeholder:
59
+ return tDetails.placeholderTmplIndex;
60
+ default:
61
+ ngDevMode && throwError(`Unexpected defer block state: ${newState}`);
62
+ return null;
63
+ }
64
+ }
65
+ /**
66
+ * Returns a minimum amount of time that a given state should be rendered for,
67
+ * taking into account `minimum` parameter value. If the `minimum` value is
68
+ * not specified - returns `null`.
69
+ */
70
+ export function getMinimumDurationForState(tDetails, currentState) {
71
+ if (currentState === DeferBlockState.Placeholder) {
72
+ return tDetails.placeholderBlockConfig?.[MINIMUM_SLOT] ?? null;
73
+ }
74
+ else if (currentState === DeferBlockState.Loading) {
75
+ return tDetails.loadingBlockConfig?.[MINIMUM_SLOT] ?? null;
76
+ }
77
+ return null;
78
+ }
79
+ /** Retrieves the value of the `after` parameter on the @loading block. */
80
+ export function getLoadingBlockAfter(tDetails) {
81
+ return tDetails.loadingBlockConfig?.[LOADING_AFTER_SLOT] ?? null;
82
+ }
83
+ /**
84
+ * Adds downloaded dependencies into a directive or a pipe registry,
85
+ * making sure that a dependency doesn't yet exist in the registry.
86
+ */
87
+ export function addDepsToRegistry(currentDeps, newDeps) {
88
+ if (!currentDeps || currentDeps.length === 0) {
89
+ return newDeps;
90
+ }
91
+ const currentDepSet = new Set(currentDeps);
92
+ for (const dep of newDeps) {
93
+ currentDepSet.add(dep);
94
+ }
95
+ // If `currentDeps` is the same length, there were no new deps and can
96
+ // return the original array.
97
+ return (currentDeps.length === currentDepSet.size) ? currentDeps : Array.from(currentDepSet);
98
+ }
99
+ /** Retrieves a TNode that represents main content of a defer block. */
100
+ export function getPrimaryBlockTNode(tView, tDetails) {
101
+ const adjustedIndex = tDetails.primaryTmplIndex + HEADER_OFFSET;
102
+ return getTNode(tView, adjustedIndex);
103
+ }
104
+ /**
105
+ * Asserts whether all dependencies for a defer block are loaded.
106
+ * Always run this function (in dev mode) before rendering a defer
107
+ * block in completed state.
108
+ */
109
+ export function assertDeferredDependenciesLoaded(tDetails) {
110
+ assertEqual(tDetails.loadingState, DeferDependenciesLoadingState.COMPLETE, 'Expecting all deferred dependencies to be loaded.');
111
+ }
112
+ /**
113
+ * Determines if a given value matches the expected structure of a defer block
114
+ *
115
+ * We can safely rely on the primaryTmplIndex because every defer block requires
116
+ * that a primary template exists. All the other template options are optional.
117
+ */
118
+ export function isTDeferBlockDetails(value) {
119
+ return (typeof value === 'object') &&
120
+ (typeof value.primaryTmplIndex === 'number');
121
+ }
122
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"utils.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/defer/utils.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EAAC,sBAAsB,EAAC,MAAM,mBAAmB,CAAC;AAGzD,OAAO,EAAC,aAAa,EAAS,KAAK,EAAQ,MAAM,4BAA4B,CAAC;AAC9E,OAAO,EAAC,QAAQ,EAAC,MAAM,4BAA4B,CAAC;AACpD,OAAO,EAAC,WAAW,EAAE,UAAU,EAAC,MAAM,gBAAgB,CAAC;AAEvD,OAAO,EAAC,eAAe,EAAE,6BAA6B,EAAsB,kBAAkB,EAAE,YAAY,EAAqB,MAAM,cAAc,CAAC;AAGtJ;;;GAGG;AACH,MAAM,UAAU,sBAAsB,CAAC,eAAuB;IAC5D,mDAAmD;IACnD,wDAAwD;IACxD,OAAO,eAAe,GAAG,CAAC,CAAC;AAC7B,CAAC;AAED,0FAA0F;AAC1F,MAAM,UAAU,qBAAqB,CAAC,KAAY,EAAE,KAAY;IAC9D,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC;IAC3B,MAAM,SAAS,GAAG,sBAAsB,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IACtD,SAAS,IAAI,sBAAsB,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;IACtD,OAAO,KAAK,CAAC,SAAS,CAAC,CAAC;AAC1B,CAAC;AAED,oDAAoD;AACpD,MAAM,UAAU,qBAAqB,CACjC,KAAY,EAAE,eAAuB,EAAE,QAA4B;IACrE,MAAM,KAAK,GAAG,KAAK,CAAC,KAAK,CAAC,CAAC;IAC3B,MAAM,SAAS,GAAG,sBAAsB,CAAC,eAAe,CAAC,CAAC;IAC1D,SAAS,IAAI,sBAAsB,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;IACtD,KAAK,CAAC,SAAS,CAAC,GAAG,QAAQ,CAAC;AAC9B,CAAC;AAED,oGAAoG;AACpG,MAAM,UAAU,qBAAqB,CAAC,KAAY,EAAE,KAAY;IAC9D,MAAM,SAAS,GAAG,sBAAsB,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC;IACtD,SAAS,IAAI,sBAAsB,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;IACtD,OAAO,KAAK,CAAC,IAAI,CAAC,SAAS,CAAuB,CAAC;AACrD,CAAC;AAED,wDAAwD;AACxD,MAAM,UAAU,qBAAqB,CACjC,KAAY,EAAE,eAAuB,EAAE,gBAAoC;IAC7E,MAAM,SAAS,GAAG,sBAAsB,CAAC,eAAe,CAAC,CAAC;IAC1D,SAAS,IAAI,sBAAsB,CAAC,KAAK,EAAE,SAAS,CAAC,CAAC;IACtD,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,GAAG,gBAAgB,CAAC;AAC3C,CAAC;AAED,MAAM,UAAU,wBAAwB,CACpC,QAAyB,EAAE,SAAgB,EAAE,KAAY;IAC3D,MAAM,KAAK,GAAG,SAAS,CAAC,KAAK,CAAC,CAAC;IAC/B,MAAM,QAAQ,GAAG,qBAAqB,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;IAErD,QAAQ,QAAQ,EAAE;QAChB,KAAK,eAAe,CAAC,QAAQ;YAC3B,OAAO,QAAQ,CAAC,gBAAgB,CAAC;QACnC,KAAK,eAAe,CAAC,OAAO;YAC1B,OAAO,QAAQ,CAAC,gBAAgB,CAAC;QACnC,KAAK,eAAe,CAAC,KAAK;YACxB,OAAO,QAAQ,CAAC,cAAc,CAAC;QACjC,KAAK,eAAe,CAAC,WAAW;YAC9B,OAAO,QAAQ,CAAC,oBAAoB,CAAC;QACvC;YACE,SAAS,IAAI,UAAU,CAAC,iCAAiC,QAAQ,EAAE,CAAC,CAAC;YACrE,OAAO,IAAI,CAAC;KACf;AACH,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,0BAA0B,CACtC,QAA4B,EAAE,YAA6B;IAC7D,IAAI,YAAY,KAAK,eAAe,CAAC,WAAW,EAAE;QAChD,OAAO,QAAQ,CAAC,sBAAsB,EAAE,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC;KAChE;SAAM,IAAI,YAAY,KAAK,eAAe,CAAC,OAAO,EAAE;QACnD,OAAO,QAAQ,CAAC,kBAAkB,EAAE,CAAC,YAAY,CAAC,IAAI,IAAI,CAAC;KAC5D;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED,0EAA0E;AAC1E,MAAM,UAAU,oBAAoB,CAAC,QAA4B;IAC/D,OAAO,QAAQ,CAAC,kBAAkB,EAAE,CAAC,kBAAkB,CAAC,IAAI,IAAI,CAAC;AACnE,CAAC;AAED;;;GAGG;AACH,MAAM,UAAU,iBAAiB,CAA4B,WAAmB,EAAE,OAAU;IAC1F,IAAI,CAAC,WAAW,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE;QAC5C,OAAO,OAAO,CAAC;KAChB;IAED,MAAM,aAAa,GAAG,IAAI,GAAG,CAAC,WAAW,CAAC,CAAC;IAC3C,KAAK,MAAM,GAAG,IAAI,OAAO,EAAE;QACzB,aAAa,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;KACxB;IAED,sEAAsE;IACtE,6BAA6B;IAC7B,OAAO,CAAC,WAAW,CAAC,MAAM,KAAK,aAAa,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,aAAa,CAAM,CAAC;AACpG,CAAC;AAED,uEAAuE;AACvE,MAAM,UAAU,oBAAoB,CAAC,KAAY,EAAE,QAA4B;IAC7E,MAAM,aAAa,GAAG,QAAQ,CAAC,gBAAgB,GAAG,aAAa,CAAC;IAChE,OAAO,QAAQ,CAAC,KAAK,EAAE,aAAa,CAAmB,CAAC;AAC1D,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,gCAAgC,CAAC,QAA4B;IAC3E,WAAW,CACP,QAAQ,CAAC,YAAY,EAAE,6BAA6B,CAAC,QAAQ,EAC7D,mDAAmD,CAAC,CAAC;AAC3D,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,oBAAoB,CAAC,KAAc;IACjD,OAAO,CAAC,OAAO,KAAK,KAAK,QAAQ,CAAC;QAC9B,CAAC,OAAQ,KAA4B,CAAC,gBAAgB,KAAK,QAAQ,CAAC,CAAC;AAC3E,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 {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} 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/**\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"]}