@angular/core 17.0.0-next.5 → 17.0.0-next.7
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/rxjs-interop/src/to_signal.mjs +13 -11
- package/esm2022/src/application_init.mjs +3 -3
- package/esm2022/src/application_module.mjs +3 -3
- package/esm2022/src/application_ref.mjs +23 -6
- package/esm2022/src/change_detection/differs/default_iterable_differ.mjs +1 -2
- package/esm2022/src/console.mjs +2 -2
- package/esm2022/src/core_private_export.mjs +2 -2
- package/esm2022/src/hydration/api.mjs +7 -8
- package/esm2022/src/hydration/views.mjs +3 -3
- package/esm2022/src/initial_render_pending_tasks.mjs +2 -2
- package/esm2022/src/linker/compiler.mjs +2 -2
- package/esm2022/src/linker/query_list.mjs +7 -10
- package/esm2022/src/linker/view_container_ref.mjs +12 -10
- package/esm2022/src/render3/deps_tracker/deps_tracker.mjs +4 -8
- package/esm2022/src/render3/instructions/control_flow.mjs +69 -64
- package/esm2022/src/render3/instructions/defer.mjs +615 -97
- package/esm2022/src/render3/instructions/defer_events.mjs +164 -0
- package/esm2022/src/render3/instructions/shared.mjs +1 -1
- package/esm2022/src/render3/instructions/template.mjs +9 -2
- package/esm2022/src/render3/interfaces/defer.mjs +9 -11
- package/esm2022/src/render3/interfaces/definition.mjs +1 -1
- package/esm2022/src/render3/interfaces/view.mjs +1 -1
- package/esm2022/src/render3/list_reconciliation.mjs +257 -0
- package/esm2022/src/render3/metadata.mjs +2 -2
- package/esm2022/src/render3/pipe.mjs +4 -4
- package/esm2022/src/render3/state.mjs +2 -11
- package/esm2022/src/render3/util/global_utils.mjs +3 -2
- package/esm2022/src/render3/util/injector_discovery_utils.mjs +35 -3
- package/esm2022/src/render3/util/view_utils.mjs +17 -3
- package/esm2022/src/render3/view_ref.mjs +11 -1
- package/esm2022/src/testability/testability.mjs +5 -5
- package/esm2022/src/util/assert.mjs +6 -1
- package/esm2022/src/version.mjs +1 -1
- package/esm2022/testing/src/component_fixture.mjs +4 -2
- package/esm2022/testing/src/defer.mjs +3 -3
- package/esm2022/testing/src/logger.mjs +4 -4
- package/fesm2022/core.mjs +13603 -12606
- package/fesm2022/core.mjs.map +1 -1
- package/fesm2022/rxjs-interop.mjs +13 -11
- package/fesm2022/rxjs-interop.mjs.map +1 -1
- package/fesm2022/testing.mjs +6 -4
- package/fesm2022/testing.mjs.map +1 -1
- package/index.d.ts +94 -53
- package/package.json +1 -1
- package/rxjs-interop/index.d.ts +1 -1
- package/schematics/migrations/block-template-entities/bundle.js +23643 -0
- package/schematics/migrations/block-template-entities/bundle.js.map +7 -0
- package/schematics/migrations.json +4 -9
- package/schematics/ng-generate/standalone-migration/bundle.js +3250 -2695
- package/schematics/ng-generate/standalone-migration/bundle.js.map +4 -4
- package/testing/index.d.ts +5 -3
- package/schematics/migrations/guard-and-resolve-interfaces/bundle.js +0 -694
- package/schematics/migrations/guard-and-resolve-interfaces/bundle.js.map +0 -7
- package/schematics/migrations/remove-module-id/bundle.js +0 -368
- package/schematics/migrations/remove-module-id/bundle.js.map +0 -7
|
@@ -0,0 +1,164 @@
|
|
|
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 { NgZone } from '../../zone';
|
|
10
|
+
/** Configuration object used to register passive and capturing events. */
|
|
11
|
+
const eventListenerOptions = {
|
|
12
|
+
passive: true,
|
|
13
|
+
capture: true
|
|
14
|
+
};
|
|
15
|
+
/** Keeps track of the currently-registered `on hover` triggers. */
|
|
16
|
+
const hoverTriggers = new WeakMap();
|
|
17
|
+
/** Keeps track of the currently-registered `on interaction` triggers. */
|
|
18
|
+
const interactionTriggers = new WeakMap();
|
|
19
|
+
/** Names of the events considered as interaction events. */
|
|
20
|
+
const interactionEventNames = ['click', 'keydown'];
|
|
21
|
+
/** Object keeping track of registered callbacks for a deferred block trigger. */
|
|
22
|
+
class DeferEventEntry {
|
|
23
|
+
constructor() {
|
|
24
|
+
this.callbacks = new Set();
|
|
25
|
+
this.listener = () => {
|
|
26
|
+
for (const callback of this.callbacks) {
|
|
27
|
+
callback();
|
|
28
|
+
}
|
|
29
|
+
};
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Registers an interaction trigger.
|
|
34
|
+
* @param trigger Element that is the trigger.
|
|
35
|
+
* @param callback Callback to be invoked when the trigger is interacted with.
|
|
36
|
+
* @param injector Injector that can be used by the trigger to resolve DI tokens.
|
|
37
|
+
*/
|
|
38
|
+
export function onInteraction(trigger, callback, injector) {
|
|
39
|
+
let entry = interactionTriggers.get(trigger);
|
|
40
|
+
// If this is the first entry for this element, add the listeners.
|
|
41
|
+
if (!entry) {
|
|
42
|
+
// Note that using managing events centrally like this lends itself well to using global
|
|
43
|
+
// event delegation. It currently does delegation at the element level, rather than the
|
|
44
|
+
// document level, because:
|
|
45
|
+
// 1. Global delegation is the most effective when there are a lot of events being registered
|
|
46
|
+
// at the same time. Deferred blocks are unlikely to be used in such a way.
|
|
47
|
+
// 2. Matching events to their target isn't free. For each `click` and `keydown` event we
|
|
48
|
+
// would have look through all the triggers and check if the target either is the element
|
|
49
|
+
// itself or it's contained within the element. Given that `click` and `keydown` are some
|
|
50
|
+
// of the most common events, this may end up introducing a lot of runtime overhead.
|
|
51
|
+
// 3. We're still registering only two events per element, no matter how many deferred blocks
|
|
52
|
+
// are referencing it.
|
|
53
|
+
entry = new DeferEventEntry();
|
|
54
|
+
interactionTriggers.set(trigger, entry);
|
|
55
|
+
// Ensure that the handler runs in the NgZone since it gets
|
|
56
|
+
// registered in `afterRender` which runs outside.
|
|
57
|
+
injector.get(NgZone).run(() => {
|
|
58
|
+
for (const name of interactionEventNames) {
|
|
59
|
+
trigger.addEventListener(name, entry.listener, eventListenerOptions);
|
|
60
|
+
}
|
|
61
|
+
});
|
|
62
|
+
}
|
|
63
|
+
entry.callbacks.add(callback);
|
|
64
|
+
return () => {
|
|
65
|
+
const { callbacks, listener } = entry;
|
|
66
|
+
callbacks.delete(callback);
|
|
67
|
+
if (callbacks.size === 0) {
|
|
68
|
+
interactionTriggers.delete(trigger);
|
|
69
|
+
for (const name of interactionEventNames) {
|
|
70
|
+
trigger.removeEventListener(name, listener, eventListenerOptions);
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
};
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Registers a hover trigger.
|
|
77
|
+
* @param trigger Element that is the trigger.
|
|
78
|
+
* @param callback Callback to be invoked when the trigger is hovered over.
|
|
79
|
+
* @param injector Injector that can be used by the trigger to resolve DI tokens.
|
|
80
|
+
*/
|
|
81
|
+
export function onHover(trigger, callback, injector) {
|
|
82
|
+
let entry = hoverTriggers.get(trigger);
|
|
83
|
+
// If this is the first entry for this element, add the listener.
|
|
84
|
+
if (!entry) {
|
|
85
|
+
entry = new DeferEventEntry();
|
|
86
|
+
hoverTriggers.set(trigger, entry);
|
|
87
|
+
// Ensure that the handler runs in the NgZone since it gets
|
|
88
|
+
// registered in `afterRender` which runs outside.
|
|
89
|
+
injector.get(NgZone).run(() => {
|
|
90
|
+
trigger.addEventListener('mouseenter', entry.listener, eventListenerOptions);
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
entry.callbacks.add(callback);
|
|
94
|
+
return () => {
|
|
95
|
+
const { callbacks, listener } = entry;
|
|
96
|
+
callbacks.delete(callback);
|
|
97
|
+
if (callbacks.size === 0) {
|
|
98
|
+
trigger.removeEventListener('mouseenter', listener, eventListenerOptions);
|
|
99
|
+
hoverTriggers.delete(trigger);
|
|
100
|
+
}
|
|
101
|
+
};
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Registers a viewport trigger.
|
|
105
|
+
* @param trigger Element that is the trigger.
|
|
106
|
+
* @param callback Callback to be invoked when the trigger comes into the viewport.
|
|
107
|
+
* @param injector Injector that can be used by the trigger to resolve DI tokens.
|
|
108
|
+
*/
|
|
109
|
+
export function onViewport(trigger, callback, injector) {
|
|
110
|
+
return injector.get(DeferIntersectionManager).register(trigger, callback);
|
|
111
|
+
}
|
|
112
|
+
/** Keeps track of the registered `viewport` triggers. */
|
|
113
|
+
class DeferIntersectionManager {
|
|
114
|
+
/** @nocollapse */
|
|
115
|
+
static { this.ɵprov = ɵɵdefineInjectable({
|
|
116
|
+
token: DeferIntersectionManager,
|
|
117
|
+
providedIn: 'root',
|
|
118
|
+
factory: () => new DeferIntersectionManager(inject(NgZone)),
|
|
119
|
+
}); }
|
|
120
|
+
constructor(ngZone) {
|
|
121
|
+
this.ngZone = ngZone;
|
|
122
|
+
/** `IntersectionObserver` used to observe `viewport` triggers. */
|
|
123
|
+
this.intersectionObserver = null;
|
|
124
|
+
/** Number of elements currently observed with `viewport` triggers. */
|
|
125
|
+
this.observedViewportElements = 0;
|
|
126
|
+
/** Currently-registered `viewport` triggers. */
|
|
127
|
+
this.viewportTriggers = new WeakMap();
|
|
128
|
+
this.intersectionCallback = entries => {
|
|
129
|
+
for (const current of entries) {
|
|
130
|
+
// Only invoke the callbacks if the specific element is intersecting.
|
|
131
|
+
if (current.isIntersecting && this.viewportTriggers.has(current.target)) {
|
|
132
|
+
this.ngZone.run(this.viewportTriggers.get(current.target).listener);
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
};
|
|
136
|
+
}
|
|
137
|
+
register(trigger, callback) {
|
|
138
|
+
let entry = this.viewportTriggers.get(trigger);
|
|
139
|
+
if (!this.intersectionObserver) {
|
|
140
|
+
this.intersectionObserver =
|
|
141
|
+
this.ngZone.runOutsideAngular(() => new IntersectionObserver(this.intersectionCallback));
|
|
142
|
+
}
|
|
143
|
+
if (!entry) {
|
|
144
|
+
entry = new DeferEventEntry();
|
|
145
|
+
this.ngZone.runOutsideAngular(() => this.intersectionObserver.observe(trigger));
|
|
146
|
+
this.viewportTriggers.set(trigger, entry);
|
|
147
|
+
this.observedViewportElements++;
|
|
148
|
+
}
|
|
149
|
+
entry.callbacks.add(callback);
|
|
150
|
+
return () => {
|
|
151
|
+
entry.callbacks.delete(callback);
|
|
152
|
+
if (entry.callbacks.size === 0) {
|
|
153
|
+
this.intersectionObserver?.unobserve(trigger);
|
|
154
|
+
this.viewportTriggers.delete(trigger);
|
|
155
|
+
this.observedViewportElements--;
|
|
156
|
+
}
|
|
157
|
+
if (this.observedViewportElements === 0) {
|
|
158
|
+
this.intersectionObserver?.disconnect();
|
|
159
|
+
this.intersectionObserver = null;
|
|
160
|
+
}
|
|
161
|
+
};
|
|
162
|
+
}
|
|
163
|
+
}
|
|
164
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"defer_events.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/instructions/defer_events.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, Injector, ɵɵdefineInjectable} from '../../di';\nimport {NgZone} from '../../zone';\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/** Names of the events considered as interaction events. */\nconst interactionEventNames = ['click', 'keydown'] as const;\n\n/** Object keeping track of registered callbacks for a deferred block trigger. */\nclass DeferEventEntry {\n  callbacks = new Set<() => void>();\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 * @param injector Injector that can be used by the trigger to resolve DI tokens.\n */\nexport function onInteraction(\n    trigger: Element, callback: VoidFunction, injector: Injector): 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 using 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 since it gets\n    // registered in `afterRender` which runs outside.\n    injector.get(NgZone).run(() => {\n      for (const name of interactionEventNames) {\n        trigger.addEventListener(name, entry!.listener, eventListenerOptions);\n      }\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 * @param injector Injector that can be used by the trigger to resolve DI tokens.\n */\nexport function onHover(\n    trigger: Element, callback: VoidFunction, injector: Injector): 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    // Ensure that the handler runs in the NgZone since it gets\n    // registered in `afterRender` which runs outside.\n    injector.get(NgZone).run(() => {\n      trigger.addEventListener('mouseenter', 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      trigger.removeEventListener('mouseenter', listener, eventListenerOptions);\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  return injector.get(DeferIntersectionManager).register(trigger, callback);\n}\n\n/** Keeps track of the registered `viewport` triggers. */\nclass DeferIntersectionManager {\n  /** @nocollapse */\n  static ɵprov = /** @pureOrBreakMyCode */ ɵɵdefineInjectable({\n    token: DeferIntersectionManager,\n    providedIn: 'root',\n    factory: () => new DeferIntersectionManager(inject(NgZone)),\n  });\n\n  /** `IntersectionObserver` used to observe `viewport` triggers. */\n  private intersectionObserver: IntersectionObserver|null = null;\n\n  /** Number of elements currently observed with `viewport` triggers. */\n  private observedViewportElements = 0;\n\n  /** Currently-registered `viewport` triggers. */\n  private viewportTriggers = new WeakMap<Element, DeferEventEntry>();\n\n  constructor(private ngZone: NgZone) {}\n\n  register(trigger: Element, callback: VoidFunction): VoidFunction {\n    let entry = this.viewportTriggers.get(trigger);\n\n    if (!this.intersectionObserver) {\n      this.intersectionObserver =\n          this.ngZone.runOutsideAngular(() => new IntersectionObserver(this.intersectionCallback));\n    }\n\n    if (!entry) {\n      entry = new DeferEventEntry();\n      this.ngZone.runOutsideAngular(() => this.intersectionObserver!.observe(trigger));\n      this.viewportTriggers.set(trigger, entry);\n      this.observedViewportElements++;\n    }\n\n    entry.callbacks.add(callback);\n\n    return () => {\n      entry!.callbacks.delete(callback);\n\n      if (entry!.callbacks.size === 0) {\n        this.intersectionObserver?.unobserve(trigger);\n        this.viewportTriggers.delete(trigger);\n        this.observedViewportElements--;\n      }\n\n      if (this.observedViewportElements === 0) {\n        this.intersectionObserver?.disconnect();\n        this.intersectionObserver = null;\n      }\n    };\n  }\n\n  private intersectionCallback: IntersectionObserverCallback = entries => {\n    for (const current of entries) {\n      // Only invoke the callbacks if the specific element is intersecting.\n      if (current.isIntersecting && this.viewportTriggers.has(current.target)) {\n        this.ngZone.run(this.viewportTriggers.get(current.target)!.listener);\n      }\n    }\n  }\n}\n"]}
|