@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,40 @@
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 { PREFETCH_TRIGGER_CLEANUP_FNS, TRIGGER_CLEANUP_FNS } from './interfaces';
9
+ /**
10
+ * Registers a cleanup function associated with a prefetching trigger
11
+ * or a regular trigger of a defer block.
12
+ */
13
+ export function storeTriggerCleanupFn(type, lDetails, cleanupFn) {
14
+ const key = type === 1 /* TriggerType.Prefetch */ ? PREFETCH_TRIGGER_CLEANUP_FNS : TRIGGER_CLEANUP_FNS;
15
+ if (lDetails[key] === null) {
16
+ lDetails[key] = [];
17
+ }
18
+ lDetails[key].push(cleanupFn);
19
+ }
20
+ /**
21
+ * Invokes registered cleanup functions either for prefetch or for regular triggers.
22
+ */
23
+ export function invokeTriggerCleanupFns(type, lDetails) {
24
+ const key = type === 1 /* TriggerType.Prefetch */ ? PREFETCH_TRIGGER_CLEANUP_FNS : TRIGGER_CLEANUP_FNS;
25
+ const cleanupFns = lDetails[key];
26
+ if (cleanupFns !== null) {
27
+ for (const cleanupFn of cleanupFns) {
28
+ cleanupFn();
29
+ }
30
+ lDetails[key] = null;
31
+ }
32
+ }
33
+ /**
34
+ * Invokes registered cleanup functions for both prefetch and regular triggers.
35
+ */
36
+ export function invokeAllTriggerCleanupFns(lDetails) {
37
+ invokeTriggerCleanupFns(1 /* TriggerType.Prefetch */, lDetails);
38
+ invokeTriggerCleanupFns(0 /* TriggerType.Regular */, lDetails);
39
+ }
40
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2xlYW51cC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2NvcmUvc3JjL2RlZmVyL2NsZWFudXAudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7OztHQU1HO0FBRUgsT0FBTyxFQUFxQiw0QkFBNEIsRUFBRSxtQkFBbUIsRUFBYyxNQUFNLGNBQWMsQ0FBQztBQUVoSDs7O0dBR0c7QUFDSCxNQUFNLFVBQVUscUJBQXFCLENBQ2pDLElBQWlCLEVBQUUsUUFBNEIsRUFBRSxTQUF1QjtJQUMxRSxNQUFNLEdBQUcsR0FBRyxJQUFJLGlDQUF5QixDQUFDLENBQUMsQ0FBQyw0QkFBNEIsQ0FBQyxDQUFDLENBQUMsbUJBQW1CLENBQUM7SUFDL0YsSUFBSSxRQUFRLENBQUMsR0FBRyxDQUFDLEtBQUssSUFBSSxFQUFFO1FBQzFCLFFBQVEsQ0FBQyxHQUFHLENBQUMsR0FBRyxFQUFFLENBQUM7S0FDcEI7SUFDRCxRQUFRLENBQUMsR0FBRyxDQUFFLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO0FBQ2pDLENBQUM7QUFFRDs7R0FFRztBQUNILE1BQU0sVUFBVSx1QkFBdUIsQ0FBQyxJQUFpQixFQUFFLFFBQTRCO0lBQ3JGLE1BQU0sR0FBRyxHQUFHLElBQUksaUNBQXlCLENBQUMsQ0FBQyxDQUFDLDRCQUE0QixDQUFDLENBQUMsQ0FBQyxtQkFBbUIsQ0FBQztJQUMvRixNQUFNLFVBQVUsR0FBRyxRQUFRLENBQUMsR0FBRyxDQUFDLENBQUM7SUFDakMsSUFBSSxVQUFVLEtBQUssSUFBSSxFQUFFO1FBQ3ZCLEtBQUssTUFBTSxTQUFTLElBQUksVUFBVSxFQUFFO1lBQ2xDLFNBQVMsRUFBRSxDQUFDO1NBQ2I7UUFDRCxRQUFRLENBQUMsR0FBRyxDQUFDLEdBQUcsSUFBSSxDQUFDO0tBQ3RCO0FBQ0gsQ0FBQztBQUVEOztHQUVHO0FBQ0gsTUFBTSxVQUFVLDBCQUEwQixDQUFDLFFBQTRCO0lBQ3JFLHVCQUF1QiwrQkFBdUIsUUFBUSxDQUFDLENBQUM7SUFDeEQsdUJBQXVCLDhCQUFzQixRQUFRLENBQUMsQ0FBQztBQUN6RCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyohXG4gKiBAbGljZW5zZVxuICogQ29weXJpZ2h0IEdvb2dsZSBMTEMgQWxsIFJpZ2h0cyBSZXNlcnZlZC5cbiAqXG4gKiBVc2Ugb2YgdGhpcyBzb3VyY2UgY29kZSBpcyBnb3Zlcm5lZCBieSBhbiBNSVQtc3R5bGUgbGljZW5zZSB0aGF0IGNhbiBiZVxuICogZm91bmQgaW4gdGhlIExJQ0VOU0UgZmlsZSBhdCBodHRwczovL2FuZ3VsYXIuaW8vbGljZW5zZVxuICovXG5cbmltcG9ydCB7TERlZmVyQmxvY2tEZXRhaWxzLCBQUkVGRVRDSF9UUklHR0VSX0NMRUFOVVBfRk5TLCBUUklHR0VSX0NMRUFOVVBfRk5TLCBUcmlnZ2VyVHlwZX0gZnJvbSAnLi9pbnRlcmZhY2VzJztcblxuLyoqXG4gKiBSZWdpc3RlcnMgYSBjbGVhbnVwIGZ1bmN0aW9uIGFzc29jaWF0ZWQgd2l0aCBhIHByZWZldGNoaW5nIHRyaWdnZXJcbiAqIG9yIGEgcmVndWxhciB0cmlnZ2VyIG9mIGEgZGVmZXIgYmxvY2suXG4gKi9cbmV4cG9ydCBmdW5jdGlvbiBzdG9yZVRyaWdnZXJDbGVhbnVwRm4oXG4gICAgdHlwZTogVHJpZ2dlclR5cGUsIGxEZXRhaWxzOiBMRGVmZXJCbG9ja0RldGFpbHMsIGNsZWFudXBGbjogVm9pZEZ1bmN0aW9uKSB7XG4gIGNvbnN0IGtleSA9IHR5cGUgPT09IFRyaWdnZXJUeXBlLlByZWZldGNoID8gUFJFRkVUQ0hfVFJJR0dFUl9DTEVBTlVQX0ZOUyA6IFRSSUdHRVJfQ0xFQU5VUF9GTlM7XG4gIGlmIChsRGV0YWlsc1trZXldID09PSBudWxsKSB7XG4gICAgbERldGFpbHNba2V5XSA9IFtdO1xuICB9XG4gIGxEZXRhaWxzW2tleV0hLnB1c2goY2xlYW51cEZuKTtcbn1cblxuLyoqXG4gKiBJbnZva2VzIHJlZ2lzdGVyZWQgY2xlYW51cCBmdW5jdGlvbnMgZWl0aGVyIGZvciBwcmVmZXRjaCBvciBmb3IgcmVndWxhciB0cmlnZ2Vycy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGludm9rZVRyaWdnZXJDbGVhbnVwRm5zKHR5cGU6IFRyaWdnZXJUeXBlLCBsRGV0YWlsczogTERlZmVyQmxvY2tEZXRhaWxzKSB7XG4gIGNvbnN0IGtleSA9IHR5cGUgPT09IFRyaWdnZXJUeXBlLlByZWZldGNoID8gUFJFRkVUQ0hfVFJJR0dFUl9DTEVBTlVQX0ZOUyA6IFRSSUdHRVJfQ0xFQU5VUF9GTlM7XG4gIGNvbnN0IGNsZWFudXBGbnMgPSBsRGV0YWlsc1trZXldO1xuICBpZiAoY2xlYW51cEZucyAhPT0gbnVsbCkge1xuICAgIGZvciAoY29uc3QgY2xlYW51cEZuIG9mIGNsZWFudXBGbnMpIHtcbiAgICAgIGNsZWFudXBGbigpO1xuICAgIH1cbiAgICBsRGV0YWlsc1trZXldID0gbnVsbDtcbiAgfVxufVxuXG4vKipcbiAqIEludm9rZXMgcmVnaXN0ZXJlZCBjbGVhbnVwIGZ1bmN0aW9ucyBmb3IgYm90aCBwcmVmZXRjaCBhbmQgcmVndWxhciB0cmlnZ2Vycy5cbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGludm9rZUFsbFRyaWdnZXJDbGVhbnVwRm5zKGxEZXRhaWxzOiBMRGVmZXJCbG9ja0RldGFpbHMpIHtcbiAgaW52b2tlVHJpZ2dlckNsZWFudXBGbnMoVHJpZ2dlclR5cGUuUHJlZmV0Y2gsIGxEZXRhaWxzKTtcbiAgaW52b2tlVHJpZ2dlckNsZWFudXBGbnMoVHJpZ2dlclR5cGUuUmVndWxhciwgbERldGFpbHMpO1xufVxuIl19
@@ -0,0 +1,47 @@
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 { CONTAINER_HEADER_OFFSET } from '../render3/interfaces/container';
9
+ import { isLContainer, isLView } from '../render3/interfaces/type_checks';
10
+ import { HEADER_OFFSET, TVIEW } from '../render3/interfaces/view';
11
+ import { getTDeferBlockDetails, isTDeferBlockDetails } from './utils';
12
+ /**
13
+ * Retrieves all defer blocks in a given LView.
14
+ *
15
+ * @param lView lView with defer blocks
16
+ * @param deferBlocks defer block aggregator array
17
+ */
18
+ export function getDeferBlocks(lView, deferBlocks) {
19
+ const tView = lView[TVIEW];
20
+ for (let i = HEADER_OFFSET; i < tView.bindingStartIndex; i++) {
21
+ if (isLContainer(lView[i])) {
22
+ const lContainer = lView[i];
23
+ // An LContainer may represent an instance of a defer block, in which case
24
+ // we store it as a result. Otherwise, keep iterating over LContainer views and
25
+ // look for defer blocks.
26
+ const isLast = i === tView.bindingStartIndex - 1;
27
+ if (!isLast) {
28
+ const tNode = tView.data[i];
29
+ const tDetails = getTDeferBlockDetails(tView, tNode);
30
+ if (isTDeferBlockDetails(tDetails)) {
31
+ deferBlocks.push({ lContainer, lView, tNode, tDetails });
32
+ // This LContainer represents a defer block, so we exit
33
+ // this iteration and don't inspect views in this LContainer.
34
+ continue;
35
+ }
36
+ }
37
+ for (let i = CONTAINER_HEADER_OFFSET; i < lContainer.length; i++) {
38
+ getDeferBlocks(lContainer[i], deferBlocks);
39
+ }
40
+ }
41
+ else if (isLView(lView[i])) {
42
+ // This is a component, enter the `getDeferBlocks` recursively.
43
+ getDeferBlocks(lView[i], deferBlocks);
44
+ }
45
+ }
46
+ }
47
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGlzY292ZXJ5LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvY29yZS9zcmMvZGVmZXIvZGlzY292ZXJ5LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7R0FNRztBQUVILE9BQU8sRUFBQyx1QkFBdUIsRUFBYSxNQUFNLGlDQUFpQyxDQUFDO0FBRXBGLE9BQU8sRUFBQyxZQUFZLEVBQUUsT0FBTyxFQUFDLE1BQU0sbUNBQW1DLENBQUM7QUFDeEUsT0FBTyxFQUFDLGFBQWEsRUFBUyxLQUFLLEVBQUMsTUFBTSw0QkFBNEIsQ0FBQztBQUd2RSxPQUFPLEVBQUMscUJBQXFCLEVBQUUsb0JBQW9CLEVBQUMsTUFBTSxTQUFTLENBQUM7QUFZcEU7Ozs7O0dBS0c7QUFDSCxNQUFNLFVBQVUsY0FBYyxDQUFDLEtBQVksRUFBRSxXQUFnQztJQUMzRSxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUM7SUFDM0IsS0FBSyxJQUFJLENBQUMsR0FBRyxhQUFhLEVBQUUsQ0FBQyxHQUFHLEtBQUssQ0FBQyxpQkFBaUIsRUFBRSxDQUFDLEVBQUUsRUFBRTtRQUM1RCxJQUFJLFlBQVksQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsRUFBRTtZQUMxQixNQUFNLFVBQVUsR0FBRyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUM7WUFDNUIsMEVBQTBFO1lBQzFFLCtFQUErRTtZQUMvRSx5QkFBeUI7WUFDekIsTUFBTSxNQUFNLEdBQUcsQ0FBQyxLQUFLLEtBQUssQ0FBQyxpQkFBaUIsR0FBRyxDQUFDLENBQUM7WUFDakQsSUFBSSxDQUFDLE1BQU0sRUFBRTtnQkFDWCxNQUFNLEtBQUssR0FBRyxLQUFLLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBVSxDQUFDO2dCQUNyQyxNQUFNLFFBQVEsR0FBRyxxQkFBcUIsQ0FBQyxLQUFLLEVBQUUsS0FBSyxDQUFDLENBQUM7Z0JBQ3JELElBQUksb0JBQW9CLENBQUMsUUFBUSxDQUFDLEVBQUU7b0JBQ2xDLFdBQVcsQ0FBQyxJQUFJLENBQUMsRUFBQyxVQUFVLEVBQUUsS0FBSyxFQUFFLEtBQUssRUFBRSxRQUFRLEVBQUMsQ0FBQyxDQUFDO29CQUN2RCx1REFBdUQ7b0JBQ3ZELDZEQUE2RDtvQkFDN0QsU0FBUztpQkFDVjthQUNGO1lBQ0QsS0FBSyxJQUFJLENBQUMsR0FBRyx1QkFBdUIsRUFBRSxDQUFDLEdBQUcsVUFBVSxDQUFDLE1BQU0sRUFBRSxDQUFDLEVBQUUsRUFBRTtnQkFDaEUsY0FBYyxDQUFDLFVBQVUsQ0FBQyxDQUFDLENBQVUsRUFBRSxXQUFXLENBQUMsQ0FBQzthQUNyRDtTQUNGO2FBQU0sSUFBSSxPQUFPLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDLEVBQUU7WUFDNUIsK0RBQStEO1lBQy9ELGNBQWMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLEVBQUUsV0FBVyxDQUFDLENBQUM7U0FDdkM7S0FDRjtBQUNILENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEBsaWNlbnNlXG4gKiBDb3B5cmlnaHQgR29vZ2xlIExMQyBBbGwgUmlnaHRzIFJlc2VydmVkLlxuICpcbiAqIFVzZSBvZiB0aGlzIHNvdXJjZSBjb2RlIGlzIGdvdmVybmVkIGJ5IGFuIE1JVC1zdHlsZSBsaWNlbnNlIHRoYXQgY2FuIGJlXG4gKiBmb3VuZCBpbiB0aGUgTElDRU5TRSBmaWxlIGF0IGh0dHBzOi8vYW5ndWxhci5pby9saWNlbnNlXG4gKi9cblxuaW1wb3J0IHtDT05UQUlORVJfSEVBREVSX09GRlNFVCwgTENvbnRhaW5lcn0gZnJvbSAnLi4vcmVuZGVyMy9pbnRlcmZhY2VzL2NvbnRhaW5lcic7XG5pbXBvcnQge1ROb2RlfSBmcm9tICcuLi9yZW5kZXIzL2ludGVyZmFjZXMvbm9kZSc7XG5pbXBvcnQge2lzTENvbnRhaW5lciwgaXNMVmlld30gZnJvbSAnLi4vcmVuZGVyMy9pbnRlcmZhY2VzL3R5cGVfY2hlY2tzJztcbmltcG9ydCB7SEVBREVSX09GRlNFVCwgTFZpZXcsIFRWSUVXfSBmcm9tICcuLi9yZW5kZXIzL2ludGVyZmFjZXMvdmlldyc7XG5cbmltcG9ydCB7VERlZmVyQmxvY2tEZXRhaWxzfSBmcm9tICcuL2ludGVyZmFjZXMnO1xuaW1wb3J0IHtnZXRURGVmZXJCbG9ja0RldGFpbHMsIGlzVERlZmVyQmxvY2tEZXRhaWxzfSBmcm9tICcuL3V0aWxzJztcblxuLyoqXG4gKiBEZWZlciBibG9jayBpbnN0YW5jZSBmb3IgdGVzdGluZy5cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBEZWZlckJsb2NrRGV0YWlscyB7XG4gIGxDb250YWluZXI6IExDb250YWluZXI7XG4gIGxWaWV3OiBMVmlldztcbiAgdE5vZGU6IFROb2RlO1xuICB0RGV0YWlsczogVERlZmVyQmxvY2tEZXRhaWxzO1xufVxuXG4vKipcbiAqIFJldHJpZXZlcyBhbGwgZGVmZXIgYmxvY2tzIGluIGEgZ2l2ZW4gTFZpZXcuXG4gKlxuICogQHBhcmFtIGxWaWV3IGxWaWV3IHdpdGggZGVmZXIgYmxvY2tzXG4gKiBAcGFyYW0gZGVmZXJCbG9ja3MgZGVmZXIgYmxvY2sgYWdncmVnYXRvciBhcnJheVxuICovXG5leHBvcnQgZnVuY3Rpb24gZ2V0RGVmZXJCbG9ja3MobFZpZXc6IExWaWV3LCBkZWZlckJsb2NrczogRGVmZXJCbG9ja0RldGFpbHNbXSkge1xuICBjb25zdCB0VmlldyA9IGxWaWV3W1RWSUVXXTtcbiAgZm9yIChsZXQgaSA9IEhFQURFUl9PRkZTRVQ7IGkgPCB0Vmlldy5iaW5kaW5nU3RhcnRJbmRleDsgaSsrKSB7XG4gICAgaWYgKGlzTENvbnRhaW5lcihsVmlld1tpXSkpIHtcbiAgICAgIGNvbnN0IGxDb250YWluZXIgPSBsVmlld1tpXTtcbiAgICAgIC8vIEFuIExDb250YWluZXIgbWF5IHJlcHJlc2VudCBhbiBpbnN0YW5jZSBvZiBhIGRlZmVyIGJsb2NrLCBpbiB3aGljaCBjYXNlXG4gICAgICAvLyB3ZSBzdG9yZSBpdCBhcyBhIHJlc3VsdC4gT3RoZXJ3aXNlLCBrZWVwIGl0ZXJhdGluZyBvdmVyIExDb250YWluZXIgdmlld3MgYW5kXG4gICAgICAvLyBsb29rIGZvciBkZWZlciBibG9ja3MuXG4gICAgICBjb25zdCBpc0xhc3QgPSBpID09PSB0Vmlldy5iaW5kaW5nU3RhcnRJbmRleCAtIDE7XG4gICAgICBpZiAoIWlzTGFzdCkge1xuICAgICAgICBjb25zdCB0Tm9kZSA9IHRWaWV3LmRhdGFbaV0gYXMgVE5vZGU7XG4gICAgICAgIGNvbnN0IHREZXRhaWxzID0gZ2V0VERlZmVyQmxvY2tEZXRhaWxzKHRWaWV3LCB0Tm9kZSk7XG4gICAgICAgIGlmIChpc1REZWZlckJsb2NrRGV0YWlscyh0RGV0YWlscykpIHtcbiAgICAgICAgICBkZWZlckJsb2Nrcy5wdXNoKHtsQ29udGFpbmVyLCBsVmlldywgdE5vZGUsIHREZXRhaWxzfSk7XG4gICAgICAgICAgLy8gVGhpcyBMQ29udGFpbmVyIHJlcHJlc2VudHMgYSBkZWZlciBibG9jaywgc28gd2UgZXhpdFxuICAgICAgICAgIC8vIHRoaXMgaXRlcmF0aW9uIGFuZCBkb24ndCBpbnNwZWN0IHZpZXdzIGluIHRoaXMgTENvbnRhaW5lci5cbiAgICAgICAgICBjb250aW51ZTtcbiAgICAgICAgfVxuICAgICAgfVxuICAgICAgZm9yIChsZXQgaSA9IENPTlRBSU5FUl9IRUFERVJfT0ZGU0VUOyBpIDwgbENvbnRhaW5lci5sZW5ndGg7IGkrKykge1xuICAgICAgICBnZXREZWZlckJsb2NrcyhsQ29udGFpbmVyW2ldIGFzIExWaWV3LCBkZWZlckJsb2Nrcyk7XG4gICAgICB9XG4gICAgfSBlbHNlIGlmIChpc0xWaWV3KGxWaWV3W2ldKSkge1xuICAgICAgLy8gVGhpcyBpcyBhIGNvbXBvbmVudCwgZW50ZXIgdGhlIGBnZXREZWZlckJsb2Nrc2AgcmVjdXJzaXZlbHkuXG4gICAgICBnZXREZWZlckJsb2NrcyhsVmlld1tpXSwgZGVmZXJCbG9ja3MpO1xuICAgIH1cbiAgfVxufVxuIl19
@@ -0,0 +1,256 @@
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 { internalAfterNextRender } from '../render3/after_render_hooks';
9
+ import { assertLContainer, assertLView } from '../render3/assert';
10
+ import { CONTAINER_HEADER_OFFSET } from '../render3/interfaces/container';
11
+ import { isDestroyed } from '../render3/interfaces/type_checks';
12
+ import { HEADER_OFFSET, INJECTOR } from '../render3/interfaces/view';
13
+ import { getNativeByIndex, removeLViewOnDestroy, storeLViewOnDestroy, walkUpViews } from '../render3/util/view_utils';
14
+ import { assertElement, assertEqual } from '../util/assert';
15
+ import { NgZone } from '../zone';
16
+ import { storeTriggerCleanupFn } from './cleanup';
17
+ import { DEFER_BLOCK_STATE, DeferBlockInternalState, DeferBlockState } from './interfaces';
18
+ import { getLDeferBlockDetails } from './utils';
19
+ /** Configuration object used to register passive and capturing events. */
20
+ const eventListenerOptions = {
21
+ passive: true,
22
+ capture: true
23
+ };
24
+ /** Keeps track of the currently-registered `on hover` triggers. */
25
+ const hoverTriggers = new WeakMap();
26
+ /** Keeps track of the currently-registered `on interaction` triggers. */
27
+ const interactionTriggers = new WeakMap();
28
+ /** Currently-registered `viewport` triggers. */
29
+ const viewportTriggers = new WeakMap();
30
+ /** Names of the events considered as interaction events. */
31
+ const interactionEventNames = ['click', 'keydown'];
32
+ /** Names of the events considered as hover events. */
33
+ const hoverEventNames = ['mouseenter', 'focusin'];
34
+ /** `IntersectionObserver` used to observe `viewport` triggers. */
35
+ let intersectionObserver = null;
36
+ /** Number of elements currently observed with `viewport` triggers. */
37
+ let observedViewportElements = 0;
38
+ /** Object keeping track of registered callbacks for a deferred block trigger. */
39
+ class DeferEventEntry {
40
+ constructor() {
41
+ this.callbacks = new Set();
42
+ this.listener = () => {
43
+ for (const callback of this.callbacks) {
44
+ callback();
45
+ }
46
+ };
47
+ }
48
+ }
49
+ /**
50
+ * Registers an interaction trigger.
51
+ * @param trigger Element that is the trigger.
52
+ * @param callback Callback to be invoked when the trigger is interacted with.
53
+ */
54
+ export function onInteraction(trigger, callback) {
55
+ let entry = interactionTriggers.get(trigger);
56
+ // If this is the first entry for this element, add the listeners.
57
+ if (!entry) {
58
+ // Note that managing events centrally like this lends itself well to using global
59
+ // event delegation. It currently does delegation at the element level, rather than the
60
+ // document level, because:
61
+ // 1. Global delegation is the most effective when there are a lot of events being registered
62
+ // at the same time. Deferred blocks are unlikely to be used in such a way.
63
+ // 2. Matching events to their target isn't free. For each `click` and `keydown` event we
64
+ // would have look through all the triggers and check if the target either is the element
65
+ // itself or it's contained within the element. Given that `click` and `keydown` are some
66
+ // of the most common events, this may end up introducing a lot of runtime overhead.
67
+ // 3. We're still registering only two events per element, no matter how many deferred blocks
68
+ // are referencing it.
69
+ entry = new DeferEventEntry();
70
+ interactionTriggers.set(trigger, entry);
71
+ // Ensure that the handler runs in the NgZone
72
+ ngDevMode && NgZone.assertInAngularZone();
73
+ for (const name of interactionEventNames) {
74
+ trigger.addEventListener(name, entry.listener, eventListenerOptions);
75
+ }
76
+ }
77
+ entry.callbacks.add(callback);
78
+ return () => {
79
+ const { callbacks, listener } = entry;
80
+ callbacks.delete(callback);
81
+ if (callbacks.size === 0) {
82
+ interactionTriggers.delete(trigger);
83
+ for (const name of interactionEventNames) {
84
+ trigger.removeEventListener(name, listener, eventListenerOptions);
85
+ }
86
+ }
87
+ };
88
+ }
89
+ /**
90
+ * Registers a hover trigger.
91
+ * @param trigger Element that is the trigger.
92
+ * @param callback Callback to be invoked when the trigger is hovered over.
93
+ */
94
+ export function onHover(trigger, callback) {
95
+ let entry = hoverTriggers.get(trigger);
96
+ // If this is the first entry for this element, add the listener.
97
+ if (!entry) {
98
+ entry = new DeferEventEntry();
99
+ hoverTriggers.set(trigger, entry);
100
+ // Ensure that the handler runs in the NgZone
101
+ ngDevMode && NgZone.assertInAngularZone();
102
+ for (const name of hoverEventNames) {
103
+ trigger.addEventListener(name, entry.listener, eventListenerOptions);
104
+ }
105
+ }
106
+ entry.callbacks.add(callback);
107
+ return () => {
108
+ const { callbacks, listener } = entry;
109
+ callbacks.delete(callback);
110
+ if (callbacks.size === 0) {
111
+ for (const name of hoverEventNames) {
112
+ trigger.removeEventListener(name, listener, eventListenerOptions);
113
+ }
114
+ hoverTriggers.delete(trigger);
115
+ }
116
+ };
117
+ }
118
+ /**
119
+ * Registers a viewport trigger.
120
+ * @param trigger Element that is the trigger.
121
+ * @param callback Callback to be invoked when the trigger comes into the viewport.
122
+ * @param injector Injector that can be used by the trigger to resolve DI tokens.
123
+ */
124
+ export function onViewport(trigger, callback, injector) {
125
+ const ngZone = injector.get(NgZone);
126
+ let entry = viewportTriggers.get(trigger);
127
+ intersectionObserver = intersectionObserver || ngZone.runOutsideAngular(() => {
128
+ return new IntersectionObserver(entries => {
129
+ for (const current of entries) {
130
+ // Only invoke the callbacks if the specific element is intersecting.
131
+ if (current.isIntersecting && viewportTriggers.has(current.target)) {
132
+ ngZone.run(viewportTriggers.get(current.target).listener);
133
+ }
134
+ }
135
+ });
136
+ });
137
+ if (!entry) {
138
+ entry = new DeferEventEntry();
139
+ ngZone.runOutsideAngular(() => intersectionObserver.observe(trigger));
140
+ viewportTriggers.set(trigger, entry);
141
+ observedViewportElements++;
142
+ }
143
+ entry.callbacks.add(callback);
144
+ return () => {
145
+ // It's possible that a different cleanup callback fully removed this element already.
146
+ if (!viewportTriggers.has(trigger)) {
147
+ return;
148
+ }
149
+ entry.callbacks.delete(callback);
150
+ if (entry.callbacks.size === 0) {
151
+ intersectionObserver?.unobserve(trigger);
152
+ viewportTriggers.delete(trigger);
153
+ observedViewportElements--;
154
+ }
155
+ if (observedViewportElements === 0) {
156
+ intersectionObserver?.disconnect();
157
+ intersectionObserver = null;
158
+ }
159
+ };
160
+ }
161
+ /**
162
+ * Helper function to get the LView in which a deferred block's trigger is rendered.
163
+ * @param deferredHostLView LView in which the deferred block is defined.
164
+ * @param deferredTNode TNode defining the deferred block.
165
+ * @param walkUpTimes Number of times to go up in the view hierarchy to find the trigger's view.
166
+ * A negative value means that the trigger is inside the block's placeholder, while an undefined
167
+ * value means that the trigger is in the same LView as the deferred block.
168
+ */
169
+ export function getTriggerLView(deferredHostLView, deferredTNode, walkUpTimes) {
170
+ // The trigger is in the same view, we don't need to traverse.
171
+ if (walkUpTimes == null) {
172
+ return deferredHostLView;
173
+ }
174
+ // A positive value or zero means that the trigger is in a parent view.
175
+ if (walkUpTimes >= 0) {
176
+ return walkUpViews(walkUpTimes, deferredHostLView);
177
+ }
178
+ // If the value is negative, it means that the trigger is inside the placeholder.
179
+ const deferredContainer = deferredHostLView[deferredTNode.index];
180
+ ngDevMode && assertLContainer(deferredContainer);
181
+ const triggerLView = deferredContainer[CONTAINER_HEADER_OFFSET] ?? null;
182
+ // We need to null check, because the placeholder might not have been rendered yet.
183
+ if (ngDevMode && triggerLView !== null) {
184
+ const lDetails = getLDeferBlockDetails(deferredHostLView, deferredTNode);
185
+ const renderedState = lDetails[DEFER_BLOCK_STATE];
186
+ assertEqual(renderedState, DeferBlockState.Placeholder, 'Expected a placeholder to be rendered in this defer block.');
187
+ assertLView(triggerLView);
188
+ }
189
+ return triggerLView;
190
+ }
191
+ /**
192
+ * Gets the element that a deferred block's trigger is pointing to.
193
+ * @param triggerLView LView in which the trigger is defined.
194
+ * @param triggerIndex Index at which the trigger element should've been rendered.
195
+ */
196
+ export function getTriggerElement(triggerLView, triggerIndex) {
197
+ const element = getNativeByIndex(HEADER_OFFSET + triggerIndex, triggerLView);
198
+ ngDevMode && assertElement(element);
199
+ return element;
200
+ }
201
+ /**
202
+ * Registers a DOM-node based trigger.
203
+ * @param initialLView LView in which the defer block is rendered.
204
+ * @param tNode TNode representing the defer block.
205
+ * @param triggerIndex Index at which to find the trigger element.
206
+ * @param walkUpTimes Number of times to go up/down in the view hierarchy to find the trigger.
207
+ * @param registerFn Function that will register the DOM events.
208
+ * @param callback Callback to be invoked when the trigger receives the event that should render
209
+ * the deferred block.
210
+ * @param type Trigger type to distinguish between regular and prefetch triggers.
211
+ */
212
+ export function registerDomTrigger(initialLView, tNode, triggerIndex, walkUpTimes, registerFn, callback, type) {
213
+ const injector = initialLView[INJECTOR];
214
+ function pollDomTrigger() {
215
+ // If the initial view was destroyed, we don't need to do anything.
216
+ if (isDestroyed(initialLView)) {
217
+ return;
218
+ }
219
+ const lDetails = getLDeferBlockDetails(initialLView, tNode);
220
+ const renderedState = lDetails[DEFER_BLOCK_STATE];
221
+ // If the block was loaded before the trigger was resolved, we don't need to do anything.
222
+ if (renderedState !== DeferBlockInternalState.Initial &&
223
+ renderedState !== DeferBlockState.Placeholder) {
224
+ return;
225
+ }
226
+ const triggerLView = getTriggerLView(initialLView, tNode, walkUpTimes);
227
+ // Keep polling until we resolve the trigger's LView.
228
+ if (!triggerLView) {
229
+ internalAfterNextRender(pollDomTrigger, { injector });
230
+ return;
231
+ }
232
+ // It's possible that the trigger's view was destroyed before we resolved the trigger element.
233
+ if (isDestroyed(triggerLView)) {
234
+ return;
235
+ }
236
+ const element = getTriggerElement(triggerLView, triggerIndex);
237
+ const cleanup = registerFn(element, () => {
238
+ if (initialLView !== triggerLView) {
239
+ removeLViewOnDestroy(triggerLView, cleanup);
240
+ }
241
+ callback();
242
+ }, injector);
243
+ // The trigger and deferred block might be in different LViews.
244
+ // For the main LView the cleanup would happen as a part of
245
+ // `storeTriggerCleanupFn` logic. For trigger LView we register
246
+ // a cleanup function there to remove event handlers in case an
247
+ // LView gets destroyed before a trigger is invoked.
248
+ if (initialLView !== triggerLView) {
249
+ storeLViewOnDestroy(triggerLView, cleanup);
250
+ }
251
+ storeTriggerCleanupFn(type, lDetails, cleanup);
252
+ }
253
+ // Begin polling for the trigger.
254
+ internalAfterNextRender(pollDomTrigger, { injector });
255
+ }
256
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dom_triggers.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/defer/dom_triggers.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAGH,OAAO,EAAC,uBAAuB,EAAC,MAAM,+BAA+B,CAAC;AACtE,OAAO,EAAC,gBAAgB,EAAE,WAAW,EAAC,MAAM,mBAAmB,CAAC;AAChE,OAAO,EAAC,uBAAuB,EAAC,MAAM,iCAAiC,CAAC;AAExE,OAAO,EAAC,WAAW,EAAC,MAAM,mCAAmC,CAAC;AAC9D,OAAO,EAAC,aAAa,EAAE,QAAQ,EAAQ,MAAM,4BAA4B,CAAC;AAC1E,OAAO,EAAC,gBAAgB,EAAE,oBAAoB,EAAE,mBAAmB,EAAE,WAAW,EAAC,MAAM,4BAA4B,CAAC;AACpH,OAAO,EAAC,aAAa,EAAE,WAAW,EAAC,MAAM,gBAAgB,CAAC;AAC1D,OAAO,EAAC,MAAM,EAAC,MAAM,SAAS,CAAC;AAC/B,OAAO,EAAC,qBAAqB,EAAC,MAAM,WAAW,CAAC;AAEhD,OAAO,EAAC,iBAAiB,EAAE,uBAAuB,EAAE,eAAe,EAAc,MAAM,cAAc,CAAC;AACtG,OAAO,EAAC,qBAAqB,EAAC,MAAM,SAAS,CAAC;AAE9C,0EAA0E;AAC1E,MAAM,oBAAoB,GAA4B;IACpD,OAAO,EAAE,IAAI;IACb,OAAO,EAAE,IAAI;CACd,CAAC;AAEF,mEAAmE;AACnE,MAAM,aAAa,GAAG,IAAI,OAAO,EAA4B,CAAC;AAE9D,yEAAyE;AACzE,MAAM,mBAAmB,GAAG,IAAI,OAAO,EAA4B,CAAC;AAEpE,gDAAgD;AAChD,MAAM,gBAAgB,GAAG,IAAI,OAAO,EAA4B,CAAC;AAEjE,4DAA4D;AAC5D,MAAM,qBAAqB,GAAG,CAAC,OAAO,EAAE,SAAS,CAAU,CAAC;AAE5D,sDAAsD;AACtD,MAAM,eAAe,GAAG,CAAC,YAAY,EAAE,SAAS,CAAU,CAAC;AAE3D,kEAAkE;AAClE,IAAI,oBAAoB,GAA8B,IAAI,CAAC;AAE3D,sEAAsE;AACtE,IAAI,wBAAwB,GAAG,CAAC,CAAC;AAEjC,iFAAiF;AACjF,MAAM,eAAe;IAArB;QACE,cAAS,GAAG,IAAI,GAAG,EAAgB,CAAC;QAEpC,aAAQ,GAAG,GAAG,EAAE;YACd,KAAK,MAAM,QAAQ,IAAI,IAAI,CAAC,SAAS,EAAE;gBACrC,QAAQ,EAAE,CAAC;aACZ;QACH,CAAC,CAAA;IACH,CAAC;CAAA;AAED;;;;GAIG;AACH,MAAM,UAAU,aAAa,CAAC,OAAgB,EAAE,QAAsB;IACpE,IAAI,KAAK,GAAG,mBAAmB,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAE7C,kEAAkE;IAClE,IAAI,CAAC,KAAK,EAAE;QACV,kFAAkF;QAClF,uFAAuF;QACvF,2BAA2B;QAC3B,6FAA6F;QAC7F,2EAA2E;QAC3E,yFAAyF;QACzF,yFAAyF;QACzF,yFAAyF;QACzF,oFAAoF;QACpF,6FAA6F;QAC7F,sBAAsB;QACtB,KAAK,GAAG,IAAI,eAAe,EAAE,CAAC;QAC9B,mBAAmB,CAAC,GAAG,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAExC,6CAA6C;QAC7C,SAAS,IAAI,MAAM,CAAC,mBAAmB,EAAE,CAAC;QAE1C,KAAK,MAAM,IAAI,IAAI,qBAAqB,EAAE;YACxC,OAAO,CAAC,gBAAgB,CAAC,IAAI,EAAE,KAAM,CAAC,QAAQ,EAAE,oBAAoB,CAAC,CAAC;SACvE;KACF;IAED,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAE9B,OAAO,GAAG,EAAE;QACV,MAAM,EAAC,SAAS,EAAE,QAAQ,EAAC,GAAG,KAAM,CAAC;QACrC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QAE3B,IAAI,SAAS,CAAC,IAAI,KAAK,CAAC,EAAE;YACxB,mBAAmB,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YAEpC,KAAK,MAAM,IAAI,IAAI,qBAAqB,EAAE;gBACxC,OAAO,CAAC,mBAAmB,CAAC,IAAI,EAAE,QAAQ,EAAE,oBAAoB,CAAC,CAAC;aACnE;SACF;IACH,CAAC,CAAC;AACJ,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,OAAO,CAAC,OAAgB,EAAE,QAAsB;IAC9D,IAAI,KAAK,GAAG,aAAa,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAEvC,iEAAiE;IACjE,IAAI,CAAC,KAAK,EAAE;QACV,KAAK,GAAG,IAAI,eAAe,EAAE,CAAC;QAC9B,aAAa,CAAC,GAAG,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAElC,6CAA6C;QAC7C,SAAS,IAAI,MAAM,CAAC,mBAAmB,EAAE,CAAC;QAE1C,KAAK,MAAM,IAAI,IAAI,eAAe,EAAE;YAClC,OAAO,CAAC,gBAAgB,CAAC,IAAI,EAAE,KAAM,CAAC,QAAQ,EAAE,oBAAoB,CAAC,CAAC;SACvE;KACF;IAED,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAE9B,OAAO,GAAG,EAAE;QACV,MAAM,EAAC,SAAS,EAAE,QAAQ,EAAC,GAAG,KAAM,CAAC;QACrC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QAE3B,IAAI,SAAS,CAAC,IAAI,KAAK,CAAC,EAAE;YACxB,KAAK,MAAM,IAAI,IAAI,eAAe,EAAE;gBAClC,OAAO,CAAC,mBAAmB,CAAC,IAAI,EAAE,QAAQ,EAAE,oBAAoB,CAAC,CAAC;aACnE;YACD,aAAa,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;SAC/B;IACH,CAAC,CAAC;AACJ,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,UAAU,CACtB,OAAgB,EAAE,QAAsB,EAAE,QAAkB;IAC9D,MAAM,MAAM,GAAG,QAAQ,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;IACpC,IAAI,KAAK,GAAG,gBAAgB,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAE1C,oBAAoB,GAAG,oBAAoB,IAAI,MAAM,CAAC,iBAAiB,CAAC,GAAG,EAAE;QAC3E,OAAO,IAAI,oBAAoB,CAAC,OAAO,CAAC,EAAE;YACxC,KAAK,MAAM,OAAO,IAAI,OAAO,EAAE;gBAC7B,qEAAqE;gBACrE,IAAI,OAAO,CAAC,cAAc,IAAI,gBAAgB,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;oBAClE,MAAM,CAAC,GAAG,CAAC,gBAAgB,CAAC,GAAG,CAAC,OAAO,CAAC,MAAM,CAAE,CAAC,QAAQ,CAAC,CAAC;iBAC5D;aACF;QACH,CAAC,CAAC,CAAC;IACL,CAAC,CAAC,CAAC;IAEH,IAAI,CAAC,KAAK,EAAE;QACV,KAAK,GAAG,IAAI,eAAe,EAAE,CAAC;QAC9B,MAAM,CAAC,iBAAiB,CAAC,GAAG,EAAE,CAAC,oBAAqB,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC;QACvE,gBAAgB,CAAC,GAAG,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QACrC,wBAAwB,EAAE,CAAC;KAC5B;IAED,KAAK,CAAC,SAAS,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IAE9B,OAAO,GAAG,EAAE;QACV,sFAAsF;QACtF,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE;YAClC,OAAO;SACR;QAED,KAAM,CAAC,SAAS,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;QAElC,IAAI,KAAM,CAAC,SAAS,CAAC,IAAI,KAAK,CAAC,EAAE;YAC/B,oBAAoB,EAAE,SAAS,CAAC,OAAO,CAAC,CAAC;YACzC,gBAAgB,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;YACjC,wBAAwB,EAAE,CAAC;SAC5B;QAED,IAAI,wBAAwB,KAAK,CAAC,EAAE;YAClC,oBAAoB,EAAE,UAAU,EAAE,CAAC;YACnC,oBAAoB,GAAG,IAAI,CAAC;SAC7B;IACH,CAAC,CAAC;AACJ,CAAC;AAED;;;;;;;GAOG;AACH,MAAM,UAAU,eAAe,CAC3B,iBAAwB,EAAE,aAAoB,EAAE,WAA6B;IAC/E,8DAA8D;IAC9D,IAAI,WAAW,IAAI,IAAI,EAAE;QACvB,OAAO,iBAAiB,CAAC;KAC1B;IAED,uEAAuE;IACvE,IAAI,WAAW,IAAI,CAAC,EAAE;QACpB,OAAO,WAAW,CAAC,WAAW,EAAE,iBAAiB,CAAC,CAAC;KACpD;IAED,iFAAiF;IACjF,MAAM,iBAAiB,GAAG,iBAAiB,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;IACjE,SAAS,IAAI,gBAAgB,CAAC,iBAAiB,CAAC,CAAC;IACjD,MAAM,YAAY,GAAG,iBAAiB,CAAC,uBAAuB,CAAC,IAAI,IAAI,CAAC;IAExE,mFAAmF;IACnF,IAAI,SAAS,IAAI,YAAY,KAAK,IAAI,EAAE;QACtC,MAAM,QAAQ,GAAG,qBAAqB,CAAC,iBAAiB,EAAE,aAAa,CAAC,CAAC;QACzE,MAAM,aAAa,GAAG,QAAQ,CAAC,iBAAiB,CAAC,CAAC;QAClD,WAAW,CACP,aAAa,EAAE,eAAe,CAAC,WAAW,EAC1C,4DAA4D,CAAC,CAAC;QAClE,WAAW,CAAC,YAAY,CAAC,CAAC;KAC3B;IAED,OAAO,YAAY,CAAC;AACtB,CAAC;AAED;;;;GAIG;AACH,MAAM,UAAU,iBAAiB,CAAC,YAAmB,EAAE,YAAoB;IACzE,MAAM,OAAO,GAAG,gBAAgB,CAAC,aAAa,GAAG,YAAY,EAAE,YAAY,CAAC,CAAC;IAC7E,SAAS,IAAI,aAAa,CAAC,OAAO,CAAC,CAAC;IACpC,OAAO,OAAkB,CAAC;AAC5B,CAAC;AAED;;;;;;;;;;GAUG;AACH,MAAM,UAAU,kBAAkB,CAC9B,YAAmB,EAAE,KAAY,EAAE,YAAoB,EAAE,WAA6B,EACtF,UAA0F,EAC1F,QAAsB,EAAE,IAAiB;IAC3C,MAAM,QAAQ,GAAG,YAAY,CAAC,QAAQ,CAAE,CAAC;IACzC,SAAS,cAAc;QACrB,mEAAmE;QACnE,IAAI,WAAW,CAAC,YAAY,CAAC,EAAE;YAC7B,OAAO;SACR;QAED,MAAM,QAAQ,GAAG,qBAAqB,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;QAC5D,MAAM,aAAa,GAAG,QAAQ,CAAC,iBAAiB,CAAC,CAAC;QAElD,yFAAyF;QACzF,IAAI,aAAa,KAAK,uBAAuB,CAAC,OAAO;YACjD,aAAa,KAAK,eAAe,CAAC,WAAW,EAAE;YACjD,OAAO;SACR;QAED,MAAM,YAAY,GAAG,eAAe,CAAC,YAAY,EAAE,KAAK,EAAE,WAAW,CAAC,CAAC;QAEvE,qDAAqD;QACrD,IAAI,CAAC,YAAY,EAAE;YACjB,uBAAuB,CAAC,cAAc,EAAE,EAAC,QAAQ,EAAC,CAAC,CAAC;YACpD,OAAO;SACR;QAED,8FAA8F;QAC9F,IAAI,WAAW,CAAC,YAAY,CAAC,EAAE;YAC7B,OAAO;SACR;QAED,MAAM,OAAO,GAAG,iBAAiB,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC;QAC9D,MAAM,OAAO,GAAG,UAAU,CAAC,OAAO,EAAE,GAAG,EAAE;YACvC,IAAI,YAAY,KAAK,YAAY,EAAE;gBACjC,oBAAoB,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;aAC7C;YACD,QAAQ,EAAE,CAAC;QACb,CAAC,EAAE,QAAQ,CAAC,CAAC;QAEb,+DAA+D;QAC/D,2DAA2D;QAC3D,+DAA+D;QAC/D,+DAA+D;QAC/D,oDAAoD;QACpD,IAAI,YAAY,KAAK,YAAY,EAAE;YACjC,mBAAmB,CAAC,YAAY,EAAE,OAAO,CAAC,CAAC;SAC5C;QAED,qBAAqB,CAAC,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,CAAC;IACjD,CAAC;IAED,iCAAiC;IACjC,uBAAuB,CAAC,cAAc,EAAE,EAAC,QAAQ,EAAC,CAAC,CAAC;AACtD,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 type {Injector} from '../di';\nimport {internalAfterNextRender} from '../render3/after_render_hooks';\nimport {assertLContainer, assertLView} from '../render3/assert';\nimport {CONTAINER_HEADER_OFFSET} from '../render3/interfaces/container';\nimport {TNode} from '../render3/interfaces/node';\nimport {isDestroyed} from '../render3/interfaces/type_checks';\nimport {HEADER_OFFSET, INJECTOR, LView} from '../render3/interfaces/view';\nimport {getNativeByIndex, removeLViewOnDestroy, storeLViewOnDestroy, walkUpViews} from '../render3/util/view_utils';\nimport {assertElement, assertEqual} from '../util/assert';\nimport {NgZone} from '../zone';\nimport {storeTriggerCleanupFn} from './cleanup';\n\nimport {DEFER_BLOCK_STATE, DeferBlockInternalState, DeferBlockState, TriggerType} from './interfaces';\nimport {getLDeferBlockDetails} from './utils';\n\n/** Configuration object used to register passive and capturing events. */\nconst eventListenerOptions: AddEventListenerOptions = {\n  passive: true,\n  capture: true\n};\n\n/** Keeps track of the currently-registered `on hover` triggers. */\nconst hoverTriggers = new WeakMap<Element, DeferEventEntry>();\n\n/** Keeps track of the currently-registered `on interaction` triggers. */\nconst interactionTriggers = new WeakMap<Element, DeferEventEntry>();\n\n/** Currently-registered `viewport` triggers. */\nconst viewportTriggers = new WeakMap<Element, DeferEventEntry>();\n\n/** Names of the events considered as interaction events. */\nconst interactionEventNames = ['click', 'keydown'] as const;\n\n/** Names of the events considered as hover events. */\nconst hoverEventNames = ['mouseenter', 'focusin'] as const;\n\n/** `IntersectionObserver` used to observe `viewport` triggers. */\nlet intersectionObserver: IntersectionObserver|null = null;\n\n/** Number of elements currently observed with `viewport` triggers. */\nlet observedViewportElements = 0;\n\n/** Object keeping track of registered callbacks for a deferred block trigger. */\nclass DeferEventEntry {\n  callbacks = new Set<VoidFunction>();\n\n  listener = () => {\n    for (const callback of this.callbacks) {\n      callback();\n    }\n  }\n}\n\n/**\n * Registers an interaction trigger.\n * @param trigger Element that is the trigger.\n * @param callback Callback to be invoked when the trigger is interacted with.\n */\nexport function onInteraction(trigger: Element, callback: VoidFunction): VoidFunction {\n  let entry = interactionTriggers.get(trigger);\n\n  // If this is the first entry for this element, add the listeners.\n  if (!entry) {\n    // Note that managing events centrally like this lends itself well to using global\n    // event delegation. It currently does delegation at the element level, rather than the\n    // document level, because:\n    // 1. Global delegation is the most effective when there are a lot of events being registered\n    // at the same time. Deferred blocks are unlikely to be used in such a way.\n    // 2. Matching events to their target isn't free. For each `click` and `keydown` event we\n    // would have look through all the triggers and check if the target either is the element\n    // itself or it's contained within the element. Given that `click` and `keydown` are some\n    // of the most common events, this may end up introducing a lot of runtime overhead.\n    // 3. We're still registering only two events per element, no matter how many deferred blocks\n    // are referencing it.\n    entry = new DeferEventEntry();\n    interactionTriggers.set(trigger, entry);\n\n    // Ensure that the handler runs in the NgZone\n    ngDevMode && NgZone.assertInAngularZone();\n\n    for (const name of interactionEventNames) {\n      trigger.addEventListener(name, entry!.listener, eventListenerOptions);\n    }\n  }\n\n  entry.callbacks.add(callback);\n\n  return () => {\n    const {callbacks, listener} = entry!;\n    callbacks.delete(callback);\n\n    if (callbacks.size === 0) {\n      interactionTriggers.delete(trigger);\n\n      for (const name of interactionEventNames) {\n        trigger.removeEventListener(name, listener, eventListenerOptions);\n      }\n    }\n  };\n}\n\n/**\n * Registers a hover trigger.\n * @param trigger Element that is the trigger.\n * @param callback Callback to be invoked when the trigger is hovered over.\n */\nexport function onHover(trigger: Element, callback: VoidFunction): VoidFunction {\n  let entry = hoverTriggers.get(trigger);\n\n  // If this is the first entry for this element, add the listener.\n  if (!entry) {\n    entry = new DeferEventEntry();\n    hoverTriggers.set(trigger, entry);\n\n    // Ensure that the handler runs in the NgZone\n    ngDevMode && NgZone.assertInAngularZone();\n\n    for (const name of hoverEventNames) {\n      trigger.addEventListener(name, entry!.listener, eventListenerOptions);\n    }\n  }\n\n  entry.callbacks.add(callback);\n\n  return () => {\n    const {callbacks, listener} = entry!;\n    callbacks.delete(callback);\n\n    if (callbacks.size === 0) {\n      for (const name of hoverEventNames) {\n        trigger.removeEventListener(name, listener, eventListenerOptions);\n      }\n      hoverTriggers.delete(trigger);\n    }\n  };\n}\n\n/**\n * Registers a viewport trigger.\n * @param trigger Element that is the trigger.\n * @param callback Callback to be invoked when the trigger comes into the viewport.\n * @param injector Injector that can be used by the trigger to resolve DI tokens.\n */\nexport function onViewport(\n    trigger: Element, callback: VoidFunction, injector: Injector): VoidFunction {\n  const ngZone = injector.get(NgZone);\n  let entry = viewportTriggers.get(trigger);\n\n  intersectionObserver = intersectionObserver || ngZone.runOutsideAngular(() => {\n    return new IntersectionObserver(entries => {\n      for (const current of entries) {\n        // Only invoke the callbacks if the specific element is intersecting.\n        if (current.isIntersecting && viewportTriggers.has(current.target)) {\n          ngZone.run(viewportTriggers.get(current.target)!.listener);\n        }\n      }\n    });\n  });\n\n  if (!entry) {\n    entry = new DeferEventEntry();\n    ngZone.runOutsideAngular(() => intersectionObserver!.observe(trigger));\n    viewportTriggers.set(trigger, entry);\n    observedViewportElements++;\n  }\n\n  entry.callbacks.add(callback);\n\n  return () => {\n    // It's possible that a different cleanup callback fully removed this element already.\n    if (!viewportTriggers.has(trigger)) {\n      return;\n    }\n\n    entry!.callbacks.delete(callback);\n\n    if (entry!.callbacks.size === 0) {\n      intersectionObserver?.unobserve(trigger);\n      viewportTriggers.delete(trigger);\n      observedViewportElements--;\n    }\n\n    if (observedViewportElements === 0) {\n      intersectionObserver?.disconnect();\n      intersectionObserver = null;\n    }\n  };\n}\n\n/**\n * Helper function to get the LView in which a deferred block's trigger is rendered.\n * @param deferredHostLView LView in which the deferred block is defined.\n * @param deferredTNode TNode defining the deferred block.\n * @param walkUpTimes Number of times to go up in the view hierarchy to find the trigger's view.\n *   A negative value means that the trigger is inside the block's placeholder, while an undefined\n *   value means that the trigger is in the same LView as the deferred block.\n */\nexport function getTriggerLView(\n    deferredHostLView: LView, deferredTNode: TNode, walkUpTimes: number|undefined): LView|null {\n  // The trigger is in the same view, we don't need to traverse.\n  if (walkUpTimes == null) {\n    return deferredHostLView;\n  }\n\n  // A positive value or zero means that the trigger is in a parent view.\n  if (walkUpTimes >= 0) {\n    return walkUpViews(walkUpTimes, deferredHostLView);\n  }\n\n  // If the value is negative, it means that the trigger is inside the placeholder.\n  const deferredContainer = deferredHostLView[deferredTNode.index];\n  ngDevMode && assertLContainer(deferredContainer);\n  const triggerLView = deferredContainer[CONTAINER_HEADER_OFFSET] ?? null;\n\n  // We need to null check, because the placeholder might not have been rendered yet.\n  if (ngDevMode && triggerLView !== null) {\n    const lDetails = getLDeferBlockDetails(deferredHostLView, deferredTNode);\n    const renderedState = lDetails[DEFER_BLOCK_STATE];\n    assertEqual(\n        renderedState, DeferBlockState.Placeholder,\n        'Expected a placeholder to be rendered in this defer block.');\n    assertLView(triggerLView);\n  }\n\n  return triggerLView;\n}\n\n/**\n * Gets the element that a deferred block's trigger is pointing to.\n * @param triggerLView LView in which the trigger is defined.\n * @param triggerIndex Index at which the trigger element should've been rendered.\n */\nexport function getTriggerElement(triggerLView: LView, triggerIndex: number): Element {\n  const element = getNativeByIndex(HEADER_OFFSET + triggerIndex, triggerLView);\n  ngDevMode && assertElement(element);\n  return element as Element;\n}\n\n/**\n * Registers a DOM-node based trigger.\n * @param initialLView LView in which the defer block is rendered.\n * @param tNode TNode representing the defer block.\n * @param triggerIndex Index at which to find the trigger element.\n * @param walkUpTimes Number of times to go up/down in the view hierarchy to find the trigger.\n * @param registerFn Function that will register the DOM events.\n * @param callback Callback to be invoked when the trigger receives the event that should render\n *     the deferred block.\n * @param type Trigger type to distinguish between regular and prefetch triggers.\n */\nexport function registerDomTrigger(\n    initialLView: LView, tNode: TNode, triggerIndex: number, walkUpTimes: number|undefined,\n    registerFn: (element: Element, callback: VoidFunction, injector: Injector) => VoidFunction,\n    callback: VoidFunction, type: TriggerType) {\n  const injector = initialLView[INJECTOR]!;\n  function pollDomTrigger() {\n    // If the initial view was destroyed, we don't need to do anything.\n    if (isDestroyed(initialLView)) {\n      return;\n    }\n\n    const lDetails = getLDeferBlockDetails(initialLView, tNode);\n    const renderedState = lDetails[DEFER_BLOCK_STATE];\n\n    // If the block was loaded before the trigger was resolved, we don't need to do anything.\n    if (renderedState !== DeferBlockInternalState.Initial &&\n        renderedState !== DeferBlockState.Placeholder) {\n      return;\n    }\n\n    const triggerLView = getTriggerLView(initialLView, tNode, walkUpTimes);\n\n    // Keep polling until we resolve the trigger's LView.\n    if (!triggerLView) {\n      internalAfterNextRender(pollDomTrigger, {injector});\n      return;\n    }\n\n    // It's possible that the trigger's view was destroyed before we resolved the trigger element.\n    if (isDestroyed(triggerLView)) {\n      return;\n    }\n\n    const element = getTriggerElement(triggerLView, triggerIndex);\n    const cleanup = registerFn(element, () => {\n      if (initialLView !== triggerLView) {\n        removeLViewOnDestroy(triggerLView, cleanup);\n      }\n      callback();\n    }, injector);\n\n    // The trigger and deferred block might be in different LViews.\n    // For the main LView the cleanup would happen as a part of\n    // `storeTriggerCleanupFn` logic. For trigger LView we register\n    // a cleanup function there to remove event handlers in case an\n    // LView gets destroyed before a trigger is invoked.\n    if (initialLView !== triggerLView) {\n      storeLViewOnDestroy(triggerLView, cleanup);\n    }\n\n    storeTriggerCleanupFn(type, lDetails, cleanup);\n  }\n\n  // Begin polling for the trigger.\n  internalAfterNextRender(pollDomTrigger, {injector});\n}\n"]}
@@ -0,0 +1,112 @@
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 { inject, ɵɵdefineInjectable } from '../di';
9
+ import { INJECTOR } from '../render3/interfaces/view';
10
+ import { NgZone } from '../zone';
11
+ /**
12
+ * Helper function to schedule a callback to be invoked when a browser becomes idle.
13
+ *
14
+ * @param callback A function to be invoked when a browser becomes idle.
15
+ * @param lView LView that hosts an instance of a defer block.
16
+ */
17
+ export function onIdle(callback, lView) {
18
+ const injector = lView[INJECTOR];
19
+ const scheduler = injector.get(IdleScheduler);
20
+ const cleanupFn = () => scheduler.remove(callback);
21
+ scheduler.add(callback);
22
+ return cleanupFn;
23
+ }
24
+ /**
25
+ * Use shims for the `requestIdleCallback` and `cancelIdleCallback` functions for
26
+ * environments where those functions are not available (e.g. Node.js and Safari).
27
+ *
28
+ * Note: we wrap the `requestIdleCallback` call into a function, so that it can be
29
+ * overridden/mocked in test environment and picked up by the runtime code.
30
+ */
31
+ const _requestIdleCallback = () => typeof requestIdleCallback !== 'undefined' ? requestIdleCallback : setTimeout;
32
+ const _cancelIdleCallback = () => typeof requestIdleCallback !== 'undefined' ? cancelIdleCallback : clearTimeout;
33
+ /**
34
+ * Helper service to schedule `requestIdleCallback`s for batches of defer blocks,
35
+ * to avoid calling `requestIdleCallback` for each defer block (e.g. if
36
+ * defer blocks are defined inside a for loop).
37
+ */
38
+ export class IdleScheduler {
39
+ constructor() {
40
+ // Indicates whether current callbacks are being invoked.
41
+ this.executingCallbacks = false;
42
+ // Currently scheduled idle callback id.
43
+ this.idleId = null;
44
+ // Set of callbacks to be invoked next.
45
+ this.current = new Set();
46
+ // Set of callbacks collected while invoking current set of callbacks.
47
+ // Those callbacks are scheduled for the next idle period.
48
+ this.deferred = new Set();
49
+ this.ngZone = inject(NgZone);
50
+ this.requestIdleCallbackFn = _requestIdleCallback().bind(globalThis);
51
+ this.cancelIdleCallbackFn = _cancelIdleCallback().bind(globalThis);
52
+ }
53
+ add(callback) {
54
+ const target = this.executingCallbacks ? this.deferred : this.current;
55
+ target.add(callback);
56
+ if (this.idleId === null) {
57
+ this.scheduleIdleCallback();
58
+ }
59
+ }
60
+ remove(callback) {
61
+ const { current, deferred } = this;
62
+ current.delete(callback);
63
+ deferred.delete(callback);
64
+ // If the last callback was removed and there is a pending
65
+ // idle callback - cancel it.
66
+ if (current.size === 0 && deferred.size === 0) {
67
+ this.cancelIdleCallback();
68
+ }
69
+ }
70
+ scheduleIdleCallback() {
71
+ const callback = () => {
72
+ this.cancelIdleCallback();
73
+ this.executingCallbacks = true;
74
+ for (const callback of this.current) {
75
+ callback();
76
+ }
77
+ this.current.clear();
78
+ this.executingCallbacks = false;
79
+ // If there are any callbacks added during an invocation
80
+ // of the current ones - make them "current" and schedule
81
+ // a new idle callback.
82
+ if (this.deferred.size > 0) {
83
+ for (const callback of this.deferred) {
84
+ this.current.add(callback);
85
+ }
86
+ this.deferred.clear();
87
+ this.scheduleIdleCallback();
88
+ }
89
+ };
90
+ // Ensure that the callback runs in the NgZone since
91
+ // the `requestIdleCallback` is not currently patched by Zone.js.
92
+ this.idleId = this.requestIdleCallbackFn(() => this.ngZone.run(callback));
93
+ }
94
+ cancelIdleCallback() {
95
+ if (this.idleId !== null) {
96
+ this.cancelIdleCallbackFn(this.idleId);
97
+ this.idleId = null;
98
+ }
99
+ }
100
+ ngOnDestroy() {
101
+ this.cancelIdleCallback();
102
+ this.current.clear();
103
+ this.deferred.clear();
104
+ }
105
+ /** @nocollapse */
106
+ static { this.ɵprov = ɵɵdefineInjectable({
107
+ token: IdleScheduler,
108
+ providedIn: 'root',
109
+ factory: () => new IdleScheduler(),
110
+ }); }
111
+ }
112
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"idle_scheduler.js","sourceRoot":"","sources":["../../../../../../../packages/core/src/defer/idle_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 {inject, ɵɵdefineInjectable} from '../di';\nimport {INJECTOR, LView} from '../render3/interfaces/view';\nimport {NgZone} from '../zone';\n\n/**\n * Helper function to schedule a callback to be invoked when a browser becomes idle.\n *\n * @param callback A function to be invoked when a browser becomes idle.\n * @param lView LView that hosts an instance of a defer block.\n */\nexport function onIdle(callback: VoidFunction, lView: LView) {\n  const injector = lView[INJECTOR]!;\n  const scheduler = injector.get(IdleScheduler);\n  const cleanupFn = () => scheduler.remove(callback);\n  scheduler.add(callback);\n  return cleanupFn;\n}\n\n/**\n * Use shims for the `requestIdleCallback` and `cancelIdleCallback` functions for\n * environments where those functions are not available (e.g. Node.js and Safari).\n *\n * Note: we wrap the `requestIdleCallback` call into a function, so that it can be\n * overridden/mocked in test environment and picked up by the runtime code.\n */\nconst _requestIdleCallback = () =>\n    typeof requestIdleCallback !== 'undefined' ? requestIdleCallback : setTimeout;\nconst _cancelIdleCallback = () =>\n    typeof requestIdleCallback !== 'undefined' ? cancelIdleCallback : clearTimeout;\n\n/**\n * Helper service to schedule `requestIdleCallback`s for batches of defer blocks,\n * to avoid calling `requestIdleCallback` for each defer block (e.g. if\n * defer blocks are defined inside a for loop).\n */\nexport class IdleScheduler {\n  // Indicates whether current callbacks are being invoked.\n  executingCallbacks = false;\n\n  // Currently scheduled idle callback id.\n  idleId: number|null = null;\n\n  // Set of callbacks to be invoked next.\n  current = new Set<VoidFunction>();\n\n  // Set of callbacks collected while invoking current set of callbacks.\n  // Those callbacks are scheduled for the next idle period.\n  deferred = new Set<VoidFunction>();\n\n  ngZone = inject(NgZone);\n\n  requestIdleCallbackFn = _requestIdleCallback().bind(globalThis);\n  cancelIdleCallbackFn = _cancelIdleCallback().bind(globalThis);\n\n  add(callback: VoidFunction) {\n    const target = this.executingCallbacks ? this.deferred : this.current;\n    target.add(callback);\n    if (this.idleId === null) {\n      this.scheduleIdleCallback();\n    }\n  }\n\n  remove(callback: VoidFunction) {\n    const {current, deferred} = this;\n\n    current.delete(callback);\n    deferred.delete(callback);\n\n    // If the last callback was removed and there is a pending\n    // idle callback - cancel it.\n    if (current.size === 0 && deferred.size === 0) {\n      this.cancelIdleCallback();\n    }\n  }\n\n  private scheduleIdleCallback() {\n    const callback = () => {\n      this.cancelIdleCallback();\n\n      this.executingCallbacks = true;\n\n      for (const callback of this.current) {\n        callback();\n      }\n      this.current.clear();\n\n      this.executingCallbacks = false;\n\n      // If there are any callbacks added during an invocation\n      // of the current ones - make them \"current\" and schedule\n      // a new idle callback.\n      if (this.deferred.size > 0) {\n        for (const callback of this.deferred) {\n          this.current.add(callback);\n        }\n        this.deferred.clear();\n        this.scheduleIdleCallback();\n      }\n    };\n    // Ensure that the callback runs in the NgZone since\n    // the `requestIdleCallback` is not currently patched by Zone.js.\n    this.idleId = this.requestIdleCallbackFn(() => this.ngZone.run(callback)) as number;\n  }\n\n  private cancelIdleCallback() {\n    if (this.idleId !== null) {\n      this.cancelIdleCallbackFn(this.idleId);\n      this.idleId = null;\n    }\n  }\n\n  ngOnDestroy() {\n    this.cancelIdleCallback();\n    this.current.clear();\n    this.deferred.clear();\n  }\n\n  /** @nocollapse */\n  static ɵprov = /** @pureOrBreakMyCode */ ɵɵdefineInjectable({\n    token: IdleScheduler,\n    providedIn: 'root',\n    factory: () => new IdleScheduler(),\n  });\n}\n"]}