@angular/core 17.0.0-next.1 → 17.0.0-next.3
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 +4 -6
- package/esm2022/src/core_private_export.mjs +2 -2
- package/esm2022/src/core_render3_private_export.mjs +5 -2
- package/esm2022/src/di/injection_token.mjs +12 -7
- package/esm2022/src/hydration/annotate.mjs +61 -30
- package/esm2022/src/hydration/cleanup.mjs +2 -2
- package/esm2022/src/linker/template_ref.mjs +3 -3
- package/esm2022/src/linker/view_container_ref.mjs +80 -25
- package/esm2022/src/render3/after_render_hooks.mjs +74 -43
- package/esm2022/src/render3/assert.mjs +2 -3
- package/esm2022/src/render3/collect_native_nodes.mjs +30 -23
- package/esm2022/src/render3/component.mjs +4 -3
- package/esm2022/src/render3/di.mjs +1 -1
- package/esm2022/src/render3/index.mjs +2 -2
- package/esm2022/src/render3/instructions/advance.mjs +3 -3
- package/esm2022/src/render3/instructions/all.mjs +2 -1
- package/esm2022/src/render3/instructions/change_detection.mjs +5 -6
- package/esm2022/src/render3/instructions/component_instance.mjs +23 -0
- package/esm2022/src/render3/instructions/control_flow.mjs +175 -3
- package/esm2022/src/render3/instructions/defer.mjs +409 -18
- package/esm2022/src/render3/instructions/shared.mjs +20 -14
- package/esm2022/src/render3/instructions/template.mjs +2 -1
- package/esm2022/src/render3/interfaces/defer.mjs +9 -0
- package/esm2022/src/render3/interfaces/definition.mjs +1 -1
- package/esm2022/src/render3/interfaces/injector.mjs +1 -1
- package/esm2022/src/render3/interfaces/node.mjs +16 -1
- package/esm2022/src/render3/interfaces/styling.mjs +4 -7
- package/esm2022/src/render3/interfaces/type_checks.mjs +4 -1
- package/esm2022/src/render3/interfaces/view.mjs +1 -1
- package/esm2022/src/render3/jit/environment.mjs +6 -1
- package/esm2022/src/render3/node_manipulation.mjs +4 -3
- package/esm2022/src/render3/reactive_lview_consumer.mjs +25 -45
- package/esm2022/src/render3/reactivity/effect.mjs +8 -8
- package/esm2022/src/render3/util/injector_utils.mjs +1 -1
- package/esm2022/src/render3/view_manipulation.mjs +13 -2
- package/esm2022/src/signals/index.mjs +4 -4
- package/esm2022/src/signals/src/api.mjs +2 -11
- package/esm2022/src/signals/src/computed.mjs +43 -93
- package/esm2022/src/signals/src/graph.mjs +238 -162
- package/esm2022/src/signals/src/signal.mjs +59 -79
- package/esm2022/src/signals/src/watch.mjs +38 -52
- package/esm2022/src/signals/src/weak_ref.mjs +2 -29
- package/esm2022/src/util/dom.mjs +2 -2
- package/esm2022/src/util/security/trusted_type_defs.mjs +1 -1
- package/esm2022/src/util/security/trusted_types.mjs +1 -1
- package/esm2022/src/version.mjs +1 -1
- package/esm2022/src/zone/ng_zone.mjs +16 -1
- package/esm2022/testing/src/logger.mjs +3 -3
- package/esm2022/testing/src/test_bed.mjs +2 -3
- package/esm2022/testing/src/test_bed_compiler.mjs +9 -13
- package/fesm2022/core.mjs +18796 -18114
- package/fesm2022/core.mjs.map +1 -1
- package/fesm2022/rxjs-interop.mjs +9 -708
- package/fesm2022/rxjs-interop.mjs.map +1 -1
- package/fesm2022/testing.mjs +35 -25672
- package/fesm2022/testing.mjs.map +1 -1
- package/index.d.ts +549 -232
- package/package.json +1 -1
- package/rxjs-interop/index.d.ts +1 -1
- package/schematics/ng-generate/standalone-migration/bundle.js +7307 -6572
- package/schematics/ng-generate/standalone-migration/bundle.js.map +4 -4
- package/testing/index.d.ts +1 -1
|
@@ -5,50 +5,168 @@
|
|
|
5
5
|
* Use of this source code is governed by an MIT-style license that can be
|
|
6
6
|
* found in the LICENSE file at https://angular.io/license
|
|
7
7
|
*/
|
|
8
|
+
import { InjectionToken } from '../../di';
|
|
9
|
+
import { findMatchingDehydratedView } from '../../hydration/views';
|
|
10
|
+
import { populateDehydratedViewsInContainer } from '../../linker/view_container_ref';
|
|
11
|
+
import { assertDefined, assertEqual, throwError } from '../../util/assert';
|
|
12
|
+
import { assertIndexInDeclRange, assertLContainer, assertTNodeForLView } from '../assert';
|
|
13
|
+
import { bindingUpdated } from '../bindings';
|
|
14
|
+
import { getComponentDef, getDirectiveDef, getPipeDef } from '../definition';
|
|
15
|
+
import { DEFER_BLOCK_STATE } from '../interfaces/defer';
|
|
16
|
+
import { isDestroyed } from '../interfaces/type_checks';
|
|
17
|
+
import { HEADER_OFFSET, INJECTOR, PARENT, TVIEW } from '../interfaces/view';
|
|
18
|
+
import { getCurrentTNode, getLView, getSelectedTNode, getTView, nextBindingIndex } from '../state';
|
|
19
|
+
import { isPlatformBrowser } from '../util/misc_utils';
|
|
20
|
+
import { getConstant, getTNode, removeLViewOnDestroy, storeLViewOnDestroy } from '../util/view_utils';
|
|
21
|
+
import { addLViewToLContainer, createAndRenderEmbeddedLView, removeLViewFromLContainer, shouldAddViewToDom } from '../view_manipulation';
|
|
22
|
+
import { ɵɵtemplate } from './template';
|
|
23
|
+
/**
|
|
24
|
+
* Returns whether defer blocks should be triggered.
|
|
25
|
+
*
|
|
26
|
+
* Currently, defer blocks are not triggered on the server,
|
|
27
|
+
* only placeholder content is rendered (if provided).
|
|
28
|
+
*/
|
|
29
|
+
function shouldTriggerDeferBlock(injector) {
|
|
30
|
+
return isPlatformBrowser(injector);
|
|
31
|
+
}
|
|
32
|
+
/**
|
|
33
|
+
* Shims for the `requestIdleCallback` and `cancelIdleCallback` functions for environments
|
|
34
|
+
* where those functions are not available (e.g. Node.js).
|
|
35
|
+
*/
|
|
36
|
+
const _requestIdleCallback = typeof requestIdleCallback !== 'undefined' ? requestIdleCallback : setTimeout;
|
|
37
|
+
const _cancelIdleCallback = typeof requestIdleCallback !== 'undefined' ? cancelIdleCallback : clearTimeout;
|
|
8
38
|
/**
|
|
9
39
|
* Creates runtime data structures for `{#defer}` blocks.
|
|
10
40
|
*
|
|
11
|
-
* @param
|
|
12
|
-
* @param
|
|
13
|
-
* @param
|
|
14
|
-
* @param
|
|
15
|
-
* @param
|
|
16
|
-
* @param
|
|
17
|
-
* @param loadingConfigIndex Index in the constants array of the configuration of the `{:loading}
|
|
41
|
+
* @param index Index of the `defer` instruction.
|
|
42
|
+
* @param primaryTmplIndex Index of the template with the primary block content.
|
|
43
|
+
* @param dependencyResolverFn Function that contains dependencies for this defer block.
|
|
44
|
+
* @param loadingTmplIndex Index of the template with the `{:loading}` block content.
|
|
45
|
+
* @param placeholderTmplIndex Index of the template with the `{:placeholder}` block content.
|
|
46
|
+
* @param errorTmplIndex Index of the template with the `{:error}` block content.
|
|
47
|
+
* @param loadingConfigIndex Index in the constants array of the configuration of the `{:loading}`.
|
|
18
48
|
* block.
|
|
19
49
|
* @param placeholderConfigIndexIndex in the constants array of the configuration of the
|
|
20
50
|
* `{:placeholder}` block.
|
|
21
51
|
*
|
|
22
52
|
* @codeGenApi
|
|
23
53
|
*/
|
|
24
|
-
export function ɵɵdefer(
|
|
54
|
+
export function ɵɵdefer(index, primaryTmplIndex, dependencyResolverFn, loadingTmplIndex, placeholderTmplIndex, errorTmplIndex, loadingConfigIndex, placeholderConfigIndex) {
|
|
55
|
+
const lView = getLView();
|
|
56
|
+
const tView = getTView();
|
|
57
|
+
const tViewConsts = tView.consts;
|
|
58
|
+
const adjustedIndex = index + HEADER_OFFSET;
|
|
59
|
+
ɵɵtemplate(index, null, 0, 0);
|
|
60
|
+
if (tView.firstCreatePass) {
|
|
61
|
+
const deferBlockConfig = {
|
|
62
|
+
primaryTmplIndex,
|
|
63
|
+
loadingTmplIndex: loadingTmplIndex ?? null,
|
|
64
|
+
placeholderTmplIndex: placeholderTmplIndex ?? null,
|
|
65
|
+
errorTmplIndex: errorTmplIndex ?? null,
|
|
66
|
+
placeholderBlockConfig: placeholderConfigIndex != null ?
|
|
67
|
+
getConstant(tViewConsts, placeholderConfigIndex) :
|
|
68
|
+
null,
|
|
69
|
+
loadingBlockConfig: loadingConfigIndex != null ?
|
|
70
|
+
getConstant(tViewConsts, loadingConfigIndex) :
|
|
71
|
+
null,
|
|
72
|
+
dependencyResolverFn: dependencyResolverFn ?? null,
|
|
73
|
+
loadingState: 0 /* DeferDependenciesLoadingState.NOT_STARTED */,
|
|
74
|
+
loadingPromise: null,
|
|
75
|
+
};
|
|
76
|
+
setTDeferBlockDetails(tView, adjustedIndex, deferBlockConfig);
|
|
77
|
+
}
|
|
78
|
+
// Lookup dehydrated views that belong to this LContainer.
|
|
79
|
+
// In client-only mode, this operation is noop.
|
|
80
|
+
const lContainer = lView[adjustedIndex];
|
|
81
|
+
populateDehydratedViewsInContainer(lContainer);
|
|
82
|
+
// Init instance-specific defer details and store it.
|
|
83
|
+
const lDetails = [];
|
|
84
|
+
lDetails[DEFER_BLOCK_STATE] = 0 /* DeferBlockInstanceState.INITIAL */;
|
|
85
|
+
setLDeferBlockDetails(lView, adjustedIndex, lDetails);
|
|
86
|
+
}
|
|
25
87
|
/**
|
|
26
|
-
* Loads
|
|
88
|
+
* Loads defer block dependencies when a trigger value becomes truthy.
|
|
27
89
|
* @codeGenApi
|
|
28
90
|
*/
|
|
29
|
-
export function ɵɵdeferWhen(
|
|
91
|
+
export function ɵɵdeferWhen(rawValue) {
|
|
92
|
+
const lView = getLView();
|
|
93
|
+
const bindingIndex = nextBindingIndex();
|
|
94
|
+
if (bindingUpdated(lView, bindingIndex, rawValue)) {
|
|
95
|
+
const value = Boolean(rawValue); // handle truthy or falsy values
|
|
96
|
+
const tNode = getSelectedTNode();
|
|
97
|
+
const lDetails = getLDeferBlockDetails(lView, tNode);
|
|
98
|
+
const renderedState = lDetails[DEFER_BLOCK_STATE];
|
|
99
|
+
if (value === false && renderedState === 0 /* DeferBlockInstanceState.INITIAL */) {
|
|
100
|
+
// If nothing is rendered yet, render a placeholder (if defined).
|
|
101
|
+
renderPlaceholder(lView, tNode);
|
|
102
|
+
}
|
|
103
|
+
else if (value === true &&
|
|
104
|
+
(renderedState === 0 /* DeferBlockInstanceState.INITIAL */ ||
|
|
105
|
+
renderedState === 1 /* DeferBlockInstanceState.PLACEHOLDER */)) {
|
|
106
|
+
// The `when` condition has changed to `true`, trigger defer block loading
|
|
107
|
+
// if the block is either in initial (nothing is rendered) or a placeholder
|
|
108
|
+
// state.
|
|
109
|
+
triggerDeferBlock(lView, tNode);
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
30
113
|
/**
|
|
31
114
|
* Prefetches the deferred content when a value becomes truthy.
|
|
32
115
|
* @codeGenApi
|
|
33
116
|
*/
|
|
34
|
-
export function ɵɵdeferPrefetchWhen(
|
|
117
|
+
export function ɵɵdeferPrefetchWhen(rawValue) {
|
|
118
|
+
const lView = getLView();
|
|
119
|
+
const bindingIndex = nextBindingIndex();
|
|
120
|
+
if (bindingUpdated(lView, bindingIndex, rawValue)) {
|
|
121
|
+
const value = Boolean(rawValue); // handle truthy or falsy values
|
|
122
|
+
const tView = lView[TVIEW];
|
|
123
|
+
const tNode = getSelectedTNode();
|
|
124
|
+
const tDetails = getTDeferBlockDetails(tView, tNode);
|
|
125
|
+
if (value === true && tDetails.loadingState === 0 /* DeferDependenciesLoadingState.NOT_STARTED */) {
|
|
126
|
+
// If loading has not been started yet, trigger it now.
|
|
127
|
+
triggerResourceLoading(tDetails, tView, lView);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
}
|
|
35
131
|
/**
|
|
36
|
-
*
|
|
132
|
+
* Sets up handlers that represent `on idle` deferred trigger.
|
|
37
133
|
* @codeGenApi
|
|
38
134
|
*/
|
|
39
|
-
export function ɵɵdeferOnIdle() {
|
|
135
|
+
export function ɵɵdeferOnIdle() {
|
|
136
|
+
const lView = getLView();
|
|
137
|
+
const tNode = getCurrentTNode();
|
|
138
|
+
renderPlaceholder(lView, tNode);
|
|
139
|
+
// Note: we pass an `lView` as a second argument to cancel an `idle`
|
|
140
|
+
// callback in case an LView got destroyed before an `idle` callback
|
|
141
|
+
// is invoked.
|
|
142
|
+
onIdle(() => triggerDeferBlock(lView, tNode), lView);
|
|
143
|
+
}
|
|
40
144
|
/**
|
|
41
|
-
* Creates runtime data structures for the `
|
|
145
|
+
* Creates runtime data structures for the `prefetch on idle` deferred trigger.
|
|
42
146
|
* @codeGenApi
|
|
43
147
|
*/
|
|
44
|
-
export function ɵɵdeferPrefetchOnIdle() {
|
|
148
|
+
export function ɵɵdeferPrefetchOnIdle() {
|
|
149
|
+
const lView = getLView();
|
|
150
|
+
const tNode = getCurrentTNode();
|
|
151
|
+
const tView = lView[TVIEW];
|
|
152
|
+
const tDetails = getTDeferBlockDetails(tView, tNode);
|
|
153
|
+
if (tDetails.loadingState === 0 /* DeferDependenciesLoadingState.NOT_STARTED */) {
|
|
154
|
+
// Set loading to the scheduled state, so that we don't register it again.
|
|
155
|
+
tDetails.loadingState = 1 /* DeferDependenciesLoadingState.SCHEDULED */;
|
|
156
|
+
// In case of prefetching, we intentionally avoid cancelling prefetching if
|
|
157
|
+
// an underlying LView get destroyed (thus passing `null` as a second argument),
|
|
158
|
+
// because there might be other LViews (that represent embedded views) that
|
|
159
|
+
// depend on resource loading.
|
|
160
|
+
onIdle(() => triggerResourceLoading(tDetails, tView, lView), null /* LView */);
|
|
161
|
+
}
|
|
162
|
+
}
|
|
45
163
|
/**
|
|
46
164
|
* Creates runtime data structures for the `on immediate` deferred trigger.
|
|
47
165
|
* @codeGenApi
|
|
48
166
|
*/
|
|
49
167
|
export function ɵɵdeferOnImmediate() { } // TODO: implement runtime logic.
|
|
50
168
|
/**
|
|
51
|
-
* Creates runtime data structures for the `
|
|
169
|
+
* Creates runtime data structures for the `prefetch on immediate` deferred trigger.
|
|
52
170
|
* @codeGenApi
|
|
53
171
|
*/
|
|
54
172
|
export function ɵɵdeferPrefetchOnImmediate() { } // TODO: implement runtime logic.
|
|
@@ -70,7 +188,7 @@ export function ɵɵdeferPrefetchOnTimer(delay) { } // TODO: implement runtime l
|
|
|
70
188
|
*/
|
|
71
189
|
export function ɵɵdeferOnHover() { } // TODO: implement runtime logic.
|
|
72
190
|
/**
|
|
73
|
-
* Creates runtime data structures for the `
|
|
191
|
+
* Creates runtime data structures for the `prefetch on hover` deferred trigger.
|
|
74
192
|
* @codeGenApi
|
|
75
193
|
*/
|
|
76
194
|
export function ɵɵdeferPrefetchOnHover() { } // TODO: implement runtime logic.
|
|
@@ -98,4 +216,277 @@ export function ɵɵdeferOnViewport(target) { } // TODO: implement runtime logic
|
|
|
98
216
|
* @codeGenApi
|
|
99
217
|
*/
|
|
100
218
|
export function ɵɵdeferPrefetchOnViewport(target) { } // TODO: implement runtime logic.
|
|
101
|
-
|
|
219
|
+
/********** Helper functions **********/
|
|
220
|
+
/**
|
|
221
|
+
* Helper function to schedule a callback to be invoked when a browser becomes idle.
|
|
222
|
+
*
|
|
223
|
+
* @param callback A function to be invoked when a browser becomes idle.
|
|
224
|
+
* @param lView An optional LView that hosts an instance of a defer block. LView is
|
|
225
|
+
* used to register a cleanup callback in case that LView got destroyed before
|
|
226
|
+
* callback was invoked. In this case, an `idle` callback is never invoked. This is
|
|
227
|
+
* helpful for cases when a defer block has scheduled rendering, but an underlying
|
|
228
|
+
* LView got destroyed prior to th block rendering.
|
|
229
|
+
*/
|
|
230
|
+
function onIdle(callback, lView) {
|
|
231
|
+
let id;
|
|
232
|
+
const removeIdleCallback = () => _cancelIdleCallback(id);
|
|
233
|
+
id = _requestIdleCallback(() => {
|
|
234
|
+
removeIdleCallback();
|
|
235
|
+
if (lView !== null) {
|
|
236
|
+
// The idle callback is invoked, we no longer need
|
|
237
|
+
// to retain a cleanup callback in an LView.
|
|
238
|
+
removeLViewOnDestroy(lView, removeIdleCallback);
|
|
239
|
+
}
|
|
240
|
+
callback();
|
|
241
|
+
});
|
|
242
|
+
if (lView !== null) {
|
|
243
|
+
// Store a cleanup function on LView, so that we cancel idle
|
|
244
|
+
// callback in case this LView is destroyed before a callback
|
|
245
|
+
// is invoked.
|
|
246
|
+
storeLViewOnDestroy(lView, removeIdleCallback);
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
/**
|
|
250
|
+
* Calculates a data slot index for defer block info (either static or
|
|
251
|
+
* instance-specific), given an index of a defer instruction.
|
|
252
|
+
*/
|
|
253
|
+
function getDeferBlockDataIndex(deferBlockIndex) {
|
|
254
|
+
// Instance state is located at the *next* position
|
|
255
|
+
// after the defer block slot in an LView or TView.data.
|
|
256
|
+
return deferBlockIndex + 1;
|
|
257
|
+
}
|
|
258
|
+
/** Retrieves a defer block state from an LView, given a TNode that represents a block. */
|
|
259
|
+
function getLDeferBlockDetails(lView, tNode) {
|
|
260
|
+
const tView = lView[TVIEW];
|
|
261
|
+
const slotIndex = getDeferBlockDataIndex(tNode.index);
|
|
262
|
+
ngDevMode && assertIndexInDeclRange(tView, slotIndex);
|
|
263
|
+
return lView[slotIndex];
|
|
264
|
+
}
|
|
265
|
+
/** Stores a defer block instance state in LView. */
|
|
266
|
+
function setLDeferBlockDetails(lView, deferBlockIndex, lDetails) {
|
|
267
|
+
const tView = lView[TVIEW];
|
|
268
|
+
const slotIndex = getDeferBlockDataIndex(deferBlockIndex);
|
|
269
|
+
ngDevMode && assertIndexInDeclRange(tView, slotIndex);
|
|
270
|
+
lView[slotIndex] = lDetails;
|
|
271
|
+
}
|
|
272
|
+
/** Retrieves static info about a defer block, given a TView and a TNode that represents a block. */
|
|
273
|
+
function getTDeferBlockDetails(tView, tNode) {
|
|
274
|
+
const slotIndex = getDeferBlockDataIndex(tNode.index);
|
|
275
|
+
ngDevMode && assertIndexInDeclRange(tView, slotIndex);
|
|
276
|
+
return tView.data[slotIndex];
|
|
277
|
+
}
|
|
278
|
+
/** Stores a defer block static info in `TView.data`. */
|
|
279
|
+
function setTDeferBlockDetails(tView, deferBlockIndex, deferBlockConfig) {
|
|
280
|
+
const slotIndex = getDeferBlockDataIndex(deferBlockIndex);
|
|
281
|
+
ngDevMode && assertIndexInDeclRange(tView, slotIndex);
|
|
282
|
+
tView.data[slotIndex] = deferBlockConfig;
|
|
283
|
+
}
|
|
284
|
+
/**
|
|
285
|
+
* Transitions a defer block to the new state. Updates the necessary
|
|
286
|
+
* data structures and renders corresponding block.
|
|
287
|
+
*
|
|
288
|
+
* @param newState New state that should be applied to the defer block.
|
|
289
|
+
* @param tNode TNode that represents a defer block.
|
|
290
|
+
* @param lContainer Represents an instance of a defer block.
|
|
291
|
+
* @param stateTmplIndex Index of a template that should be rendered.
|
|
292
|
+
*/
|
|
293
|
+
function renderDeferBlockState(newState, tNode, lContainer, stateTmplIndex) {
|
|
294
|
+
const hostLView = lContainer[PARENT];
|
|
295
|
+
// Check if this view is not destroyed. Since the loading process was async,
|
|
296
|
+
// the view might end up being destroyed by the time rendering happens.
|
|
297
|
+
if (isDestroyed(hostLView))
|
|
298
|
+
return;
|
|
299
|
+
// Make sure this TNode belongs to TView that represents host LView.
|
|
300
|
+
ngDevMode && assertTNodeForLView(tNode, hostLView);
|
|
301
|
+
const lDetails = getLDeferBlockDetails(hostLView, tNode);
|
|
302
|
+
ngDevMode && assertDefined(lDetails, 'Expected a defer block state defined');
|
|
303
|
+
// Note: we transition to the next state if the previous state was represented
|
|
304
|
+
// with a number that is less than the next state. For example, if the current
|
|
305
|
+
// state is "loading" (represented as `2`), we should not show a placeholder
|
|
306
|
+
// (represented as `1`).
|
|
307
|
+
if (lDetails[DEFER_BLOCK_STATE] < newState && stateTmplIndex !== null) {
|
|
308
|
+
lDetails[DEFER_BLOCK_STATE] = newState;
|
|
309
|
+
const hostTView = hostLView[TVIEW];
|
|
310
|
+
const adjustedIndex = stateTmplIndex + HEADER_OFFSET;
|
|
311
|
+
const tNode = getTNode(hostTView, adjustedIndex);
|
|
312
|
+
// There is only 1 view that can be present in an LContainer that
|
|
313
|
+
// represents a `{#defer}` block, so always refer to the first one.
|
|
314
|
+
const viewIndex = 0;
|
|
315
|
+
removeLViewFromLContainer(lContainer, viewIndex);
|
|
316
|
+
const dehydratedView = findMatchingDehydratedView(lContainer, tNode.tView.ssrId);
|
|
317
|
+
const embeddedLView = createAndRenderEmbeddedLView(hostLView, tNode, null, { dehydratedView });
|
|
318
|
+
addLViewToLContainer(lContainer, embeddedLView, viewIndex, shouldAddViewToDom(tNode, dehydratedView));
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
/**
|
|
322
|
+
* Trigger loading of defer block dependencies if the process hasn't started yet.
|
|
323
|
+
*
|
|
324
|
+
* @param tDetails Static information about this defer block.
|
|
325
|
+
* @param tView TView of a host view.
|
|
326
|
+
* @param lView LView of a host view.
|
|
327
|
+
*/
|
|
328
|
+
function triggerResourceLoading(tDetails, tView, lView) {
|
|
329
|
+
const injector = lView[INJECTOR];
|
|
330
|
+
if (!shouldTriggerDeferBlock(injector) ||
|
|
331
|
+
(tDetails.loadingState !== 0 /* DeferDependenciesLoadingState.NOT_STARTED */ &&
|
|
332
|
+
tDetails.loadingState !== 1 /* DeferDependenciesLoadingState.SCHEDULED */)) {
|
|
333
|
+
// If the loading status is different from initial one, it means that
|
|
334
|
+
// the loading of dependencies is in progress and there is nothing to do
|
|
335
|
+
// in this function. All details can be obtained from the `tDetails` object.
|
|
336
|
+
return;
|
|
337
|
+
}
|
|
338
|
+
const primaryBlockTNode = getPrimaryBlockTNode(tView, tDetails);
|
|
339
|
+
// Switch from NOT_STARTED -> IN_PROGRESS state.
|
|
340
|
+
tDetails.loadingState = 2 /* DeferDependenciesLoadingState.IN_PROGRESS */;
|
|
341
|
+
// Check if dependency function interceptor is configured.
|
|
342
|
+
const deferDependencyInterceptor = injector.get(DEFER_BLOCK_DEPENDENCY_INTERCEPTOR, null, { optional: true });
|
|
343
|
+
const dependenciesFn = deferDependencyInterceptor ?
|
|
344
|
+
deferDependencyInterceptor.intercept(tDetails.dependencyResolverFn) :
|
|
345
|
+
tDetails.dependencyResolverFn;
|
|
346
|
+
// The `dependenciesFn` might be `null` when all dependencies within
|
|
347
|
+
// a given `{#defer}` block were eagerly references elsewhere in a file,
|
|
348
|
+
// thus no dynamic `import()`s were produced.
|
|
349
|
+
if (!dependenciesFn) {
|
|
350
|
+
tDetails.loadingPromise = Promise.resolve().then(() => {
|
|
351
|
+
tDetails.loadingState = 3 /* DeferDependenciesLoadingState.COMPLETE */;
|
|
352
|
+
});
|
|
353
|
+
return;
|
|
354
|
+
}
|
|
355
|
+
// Start downloading of defer block dependencies.
|
|
356
|
+
tDetails.loadingPromise = Promise.allSettled(dependenciesFn()).then(results => {
|
|
357
|
+
let failed = false;
|
|
358
|
+
const directiveDefs = [];
|
|
359
|
+
const pipeDefs = [];
|
|
360
|
+
for (const result of results) {
|
|
361
|
+
if (result.status === 'fulfilled') {
|
|
362
|
+
const dependency = result.value;
|
|
363
|
+
const directiveDef = getComponentDef(dependency) || getDirectiveDef(dependency);
|
|
364
|
+
if (directiveDef) {
|
|
365
|
+
directiveDefs.push(directiveDef);
|
|
366
|
+
}
|
|
367
|
+
else {
|
|
368
|
+
const pipeDef = getPipeDef(dependency);
|
|
369
|
+
if (pipeDef) {
|
|
370
|
+
pipeDefs.push(pipeDef);
|
|
371
|
+
}
|
|
372
|
+
}
|
|
373
|
+
}
|
|
374
|
+
else {
|
|
375
|
+
failed = true;
|
|
376
|
+
break;
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
// Loading is completed, we no longer need this Promise.
|
|
380
|
+
tDetails.loadingPromise = null;
|
|
381
|
+
if (failed) {
|
|
382
|
+
tDetails.loadingState = 4 /* DeferDependenciesLoadingState.FAILED */;
|
|
383
|
+
}
|
|
384
|
+
else {
|
|
385
|
+
tDetails.loadingState = 3 /* DeferDependenciesLoadingState.COMPLETE */;
|
|
386
|
+
// Update directive and pipe registries to add newly downloaded dependencies.
|
|
387
|
+
const primaryBlockTView = primaryBlockTNode.tView;
|
|
388
|
+
if (directiveDefs.length > 0) {
|
|
389
|
+
primaryBlockTView.directiveRegistry = primaryBlockTView.directiveRegistry ?
|
|
390
|
+
[...primaryBlockTView.directiveRegistry, ...directiveDefs] :
|
|
391
|
+
directiveDefs;
|
|
392
|
+
}
|
|
393
|
+
if (pipeDefs.length > 0) {
|
|
394
|
+
primaryBlockTView.pipeRegistry = primaryBlockTView.pipeRegistry ?
|
|
395
|
+
[...primaryBlockTView.pipeRegistry, ...pipeDefs] :
|
|
396
|
+
pipeDefs;
|
|
397
|
+
}
|
|
398
|
+
}
|
|
399
|
+
});
|
|
400
|
+
}
|
|
401
|
+
/** Utility function to render `{:placeholder}` content (if present) */
|
|
402
|
+
function renderPlaceholder(lView, tNode) {
|
|
403
|
+
const tView = lView[TVIEW];
|
|
404
|
+
const lContainer = lView[tNode.index];
|
|
405
|
+
ngDevMode && assertLContainer(lContainer);
|
|
406
|
+
const tDetails = getTDeferBlockDetails(tView, tNode);
|
|
407
|
+
renderDeferBlockState(1 /* DeferBlockInstanceState.PLACEHOLDER */, tNode, lContainer, tDetails.placeholderTmplIndex);
|
|
408
|
+
}
|
|
409
|
+
/**
|
|
410
|
+
* Subscribes to the "loading" Promise and renders corresponding defer sub-block,
|
|
411
|
+
* based on the loading results.
|
|
412
|
+
*
|
|
413
|
+
* @param lContainer Represents an instance of a defer block.
|
|
414
|
+
* @param tNode Represents defer block info shared across all instances.
|
|
415
|
+
*/
|
|
416
|
+
function renderDeferStateAfterResourceLoading(tDetails, tNode, lContainer) {
|
|
417
|
+
ngDevMode &&
|
|
418
|
+
assertDefined(tDetails.loadingPromise, 'Expected loading Promise to exist on this defer block');
|
|
419
|
+
tDetails.loadingPromise.then(() => {
|
|
420
|
+
if (tDetails.loadingState === 3 /* DeferDependenciesLoadingState.COMPLETE */) {
|
|
421
|
+
ngDevMode && assertDeferredDependenciesLoaded(tDetails);
|
|
422
|
+
// Everything is loaded, show the primary block content
|
|
423
|
+
renderDeferBlockState(3 /* DeferBlockInstanceState.COMPLETE */, tNode, lContainer, tDetails.primaryTmplIndex);
|
|
424
|
+
}
|
|
425
|
+
else if (tDetails.loadingState === 4 /* DeferDependenciesLoadingState.FAILED */) {
|
|
426
|
+
renderDeferBlockState(4 /* DeferBlockInstanceState.ERROR */, tNode, lContainer, tDetails.errorTmplIndex);
|
|
427
|
+
}
|
|
428
|
+
});
|
|
429
|
+
}
|
|
430
|
+
/** Retrieves a TNode that represents main content of a defer block. */
|
|
431
|
+
function getPrimaryBlockTNode(tView, tDetails) {
|
|
432
|
+
const adjustedIndex = tDetails.primaryTmplIndex + HEADER_OFFSET;
|
|
433
|
+
return getTNode(tView, adjustedIndex);
|
|
434
|
+
}
|
|
435
|
+
/**
|
|
436
|
+
* Attempts to trigger loading of defer block dependencies.
|
|
437
|
+
* If the block is already in a loading, completed or an error state -
|
|
438
|
+
* no additional actions are taken.
|
|
439
|
+
*/
|
|
440
|
+
function triggerDeferBlock(lView, tNode) {
|
|
441
|
+
const tView = lView[TVIEW];
|
|
442
|
+
const lContainer = lView[tNode.index];
|
|
443
|
+
const injector = lView[INJECTOR];
|
|
444
|
+
ngDevMode && assertLContainer(lContainer);
|
|
445
|
+
if (!shouldTriggerDeferBlock(injector))
|
|
446
|
+
return;
|
|
447
|
+
const tDetails = getTDeferBlockDetails(tView, tNode);
|
|
448
|
+
// Condition is triggered, try to render loading state and start downloading.
|
|
449
|
+
// Note: if a block is in a loading, completed or an error state, this call would be a noop.
|
|
450
|
+
renderDeferBlockState(2 /* DeferBlockInstanceState.LOADING */, tNode, lContainer, tDetails.loadingTmplIndex);
|
|
451
|
+
switch (tDetails.loadingState) {
|
|
452
|
+
case 0 /* DeferDependenciesLoadingState.NOT_STARTED */:
|
|
453
|
+
case 1 /* DeferDependenciesLoadingState.SCHEDULED */:
|
|
454
|
+
triggerResourceLoading(tDetails, lView[TVIEW], lView);
|
|
455
|
+
// The `loadingState` might have changed to "loading".
|
|
456
|
+
if (tDetails.loadingState ===
|
|
457
|
+
2 /* DeferDependenciesLoadingState.IN_PROGRESS */) {
|
|
458
|
+
renderDeferStateAfterResourceLoading(tDetails, tNode, lContainer);
|
|
459
|
+
}
|
|
460
|
+
break;
|
|
461
|
+
case 2 /* DeferDependenciesLoadingState.IN_PROGRESS */:
|
|
462
|
+
renderDeferStateAfterResourceLoading(tDetails, tNode, lContainer);
|
|
463
|
+
break;
|
|
464
|
+
case 3 /* DeferDependenciesLoadingState.COMPLETE */:
|
|
465
|
+
ngDevMode && assertDeferredDependenciesLoaded(tDetails);
|
|
466
|
+
renderDeferBlockState(3 /* DeferBlockInstanceState.COMPLETE */, tNode, lContainer, tDetails.primaryTmplIndex);
|
|
467
|
+
break;
|
|
468
|
+
case 4 /* DeferDependenciesLoadingState.FAILED */:
|
|
469
|
+
renderDeferBlockState(4 /* DeferBlockInstanceState.ERROR */, tNode, lContainer, tDetails.errorTmplIndex);
|
|
470
|
+
break;
|
|
471
|
+
default:
|
|
472
|
+
if (ngDevMode) {
|
|
473
|
+
throwError('Unknown defer block state');
|
|
474
|
+
}
|
|
475
|
+
}
|
|
476
|
+
}
|
|
477
|
+
/**
|
|
478
|
+
* Asserts whether all dependencies for a defer block are loaded.
|
|
479
|
+
* Always run this function (in dev mode) before rendering a defer
|
|
480
|
+
* block in completed state.
|
|
481
|
+
*/
|
|
482
|
+
function assertDeferredDependenciesLoaded(tDetails) {
|
|
483
|
+
assertEqual(tDetails.loadingState, 3 /* DeferDependenciesLoadingState.COMPLETE */, 'Expecting all deferred dependencies to be loaded.');
|
|
484
|
+
}
|
|
485
|
+
/**
|
|
486
|
+
* **INTERNAL**, avoid referencing it in application code.
|
|
487
|
+
*
|
|
488
|
+
* Injector token that allows to provide `DeferBlockDependencyInterceptor` class
|
|
489
|
+
* implementation.
|
|
490
|
+
*/
|
|
491
|
+
export const DEFER_BLOCK_DEPENDENCY_INTERCEPTOR = new InjectionToken(ngDevMode ? 'DEFER_BLOCK_DEPENDENCY_INTERCEPTOR' : '');
|
|
492
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"defer.js","sourceRoot":"","sources":["../../../../../../../../packages/core/src/render3/instructions/defer.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 {InjectionToken, Injector} from '../../di';\nimport {findMatchingDehydratedView} from '../../hydration/views';\nimport {populateDehydratedViewsInContainer} from '../../linker/view_container_ref';\nimport {assertDefined, assertEqual, throwError} from '../../util/assert';\nimport {assertIndexInDeclRange, assertLContainer, assertTNodeForLView} from '../assert';\nimport {bindingUpdated} from '../bindings';\nimport {getComponentDef, getDirectiveDef, getPipeDef} from '../definition';\nimport {LContainer} from '../interfaces/container';\nimport {DEFER_BLOCK_STATE, DeferBlockInstanceState, DeferDependenciesLoadingState, DeferredLoadingBlockConfig, DeferredPlaceholderBlockConfig, DependencyResolverFn, LDeferBlockDetails, TDeferBlockDetails} from '../interfaces/defer';\nimport {DirectiveDefList, PipeDefList} from '../interfaces/definition';\nimport {TContainerNode, TNode} from '../interfaces/node';\nimport {isDestroyed} from '../interfaces/type_checks';\nimport {HEADER_OFFSET, INJECTOR, LView, PARENT, TVIEW, TView} from '../interfaces/view';\nimport {getCurrentTNode, getLView, getSelectedTNode, getTView, nextBindingIndex} from '../state';\nimport {isPlatformBrowser} from '../util/misc_utils';\nimport {getConstant, getTNode, removeLViewOnDestroy, storeLViewOnDestroy} from '../util/view_utils';\nimport {addLViewToLContainer, createAndRenderEmbeddedLView, removeLViewFromLContainer, shouldAddViewToDom} from '../view_manipulation';\n\nimport {ɵɵtemplate} from './template';\n\n/**\n * Returns whether defer blocks should be triggered.\n *\n * Currently, defer blocks are not triggered on the server,\n * only placeholder content is rendered (if provided).\n */\nfunction shouldTriggerDeferBlock(injector: Injector): boolean {\n  return isPlatformBrowser(injector);\n}\n\n/**\n * Shims for the `requestIdleCallback` and `cancelIdleCallback` functions for environments\n * where those functions are not available (e.g. Node.js).\n */\nconst _requestIdleCallback =\n    typeof requestIdleCallback !== 'undefined' ? requestIdleCallback : setTimeout;\nconst _cancelIdleCallback =\n    typeof requestIdleCallback !== 'undefined' ? cancelIdleCallback : clearTimeout;\n\n/**\n * Creates runtime data structures for `{#defer}` blocks.\n *\n * @param index Index of the `defer` instruction.\n * @param primaryTmplIndex Index of the template with the primary block content.\n * @param dependencyResolverFn Function that contains dependencies for this defer block.\n * @param loadingTmplIndex Index of the template with the `{:loading}` block content.\n * @param placeholderTmplIndex Index of the template with the `{:placeholder}` block content.\n * @param errorTmplIndex Index of the template with the `{:error}` block content.\n * @param loadingConfigIndex Index in the constants array of the configuration of the `{:loading}`.\n *     block.\n * @param placeholderConfigIndexIndex in the constants array of the configuration of the\n *     `{:placeholder}` block.\n *\n * @codeGenApi\n */\nexport function ɵɵdefer(\n    index: number, primaryTmplIndex: number, dependencyResolverFn?: DependencyResolverFn|null,\n    loadingTmplIndex?: number|null, placeholderTmplIndex?: number|null,\n    errorTmplIndex?: number|null, loadingConfigIndex?: number|null,\n    placeholderConfigIndex?: number|null) {\n  const lView = getLView();\n  const tView = getTView();\n  const tViewConsts = tView.consts;\n  const adjustedIndex = index + HEADER_OFFSET;\n\n  ɵɵtemplate(index, null, 0, 0);\n\n  if (tView.firstCreatePass) {\n    const deferBlockConfig: TDeferBlockDetails = {\n      primaryTmplIndex,\n      loadingTmplIndex: loadingTmplIndex ?? null,\n      placeholderTmplIndex: placeholderTmplIndex ?? null,\n      errorTmplIndex: errorTmplIndex ?? null,\n      placeholderBlockConfig: placeholderConfigIndex != null ?\n          getConstant<DeferredPlaceholderBlockConfig>(tViewConsts, placeholderConfigIndex) :\n          null,\n      loadingBlockConfig: loadingConfigIndex != null ?\n          getConstant<DeferredLoadingBlockConfig>(tViewConsts, loadingConfigIndex) :\n          null,\n      dependencyResolverFn: dependencyResolverFn ?? null,\n      loadingState: DeferDependenciesLoadingState.NOT_STARTED,\n      loadingPromise: null,\n    };\n\n    setTDeferBlockDetails(tView, adjustedIndex, deferBlockConfig);\n  }\n\n  // Lookup dehydrated views that belong to this LContainer.\n  // In client-only mode, this operation is noop.\n  const lContainer = lView[adjustedIndex];\n  populateDehydratedViewsInContainer(lContainer);\n\n  // Init instance-specific defer details and store it.\n  const lDetails = [];\n  lDetails[DEFER_BLOCK_STATE] = DeferBlockInstanceState.INITIAL;\n  setLDeferBlockDetails(lView, adjustedIndex, lDetails as LDeferBlockDetails);\n}\n\n/**\n * Loads defer block dependencies when a trigger value becomes truthy.\n * @codeGenApi\n */\nexport function ɵɵdeferWhen(rawValue: unknown) {\n  const lView = getLView();\n  const bindingIndex = nextBindingIndex();\n\n  if (bindingUpdated(lView, bindingIndex, rawValue)) {\n    const value = Boolean(rawValue);  // handle truthy or falsy values\n    const tNode = getSelectedTNode();\n    const lDetails = getLDeferBlockDetails(lView, tNode);\n    const renderedState = lDetails[DEFER_BLOCK_STATE];\n    if (value === false && renderedState === DeferBlockInstanceState.INITIAL) {\n      // If nothing is rendered yet, render a placeholder (if defined).\n      renderPlaceholder(lView, tNode);\n    } else if (\n        value === true &&\n        (renderedState === DeferBlockInstanceState.INITIAL ||\n         renderedState === DeferBlockInstanceState.PLACEHOLDER)) {\n      // The `when` condition has changed to `true`, trigger defer block loading\n      // if the block is either in initial (nothing is rendered) or a placeholder\n      // state.\n      triggerDeferBlock(lView, tNode);\n    }\n  }\n}\n\n/**\n * Prefetches the deferred content when a value becomes truthy.\n * @codeGenApi\n */\nexport function ɵɵdeferPrefetchWhen(rawValue: unknown) {\n  const lView = getLView();\n  const bindingIndex = nextBindingIndex();\n\n  if (bindingUpdated(lView, bindingIndex, rawValue)) {\n    const value = Boolean(rawValue);  // handle truthy or falsy values\n    const tView = lView[TVIEW];\n    const tNode = getSelectedTNode();\n    const tDetails = getTDeferBlockDetails(tView, tNode);\n    if (value === true && tDetails.loadingState === DeferDependenciesLoadingState.NOT_STARTED) {\n      // If loading has not been started yet, trigger it now.\n      triggerResourceLoading(tDetails, tView, lView);\n    }\n  }\n}\n\n/**\n * Sets up handlers that represent `on idle` deferred trigger.\n * @codeGenApi\n */\nexport function ɵɵdeferOnIdle() {\n  const lView = getLView();\n  const tNode = getCurrentTNode()!;\n\n  renderPlaceholder(lView, tNode);\n\n  // Note: we pass an `lView` as a second argument to cancel an `idle`\n  // callback in case an LView got destroyed before an `idle` callback\n  // is invoked.\n  onIdle(() => triggerDeferBlock(lView, tNode), lView);\n}\n\n/**\n * Creates runtime data structures for the `prefetch on idle` deferred trigger.\n * @codeGenApi\n */\nexport function ɵɵdeferPrefetchOnIdle() {\n  const lView = getLView();\n  const tNode = getCurrentTNode()!;\n  const tView = lView[TVIEW];\n  const tDetails = getTDeferBlockDetails(tView, tNode);\n\n  if (tDetails.loadingState === DeferDependenciesLoadingState.NOT_STARTED) {\n    // Set loading to the scheduled state, so that we don't register it again.\n    tDetails.loadingState = DeferDependenciesLoadingState.SCHEDULED;\n\n    // In case of prefetching, we intentionally avoid cancelling prefetching if\n    // an underlying LView get destroyed (thus passing `null` as a second argument),\n    // because there might be other LViews (that represent embedded views) that\n    // depend on resource loading.\n    onIdle(() => triggerResourceLoading(tDetails, tView, lView), null /* LView */);\n  }\n}\n\n/**\n * Creates runtime data structures for the `on immediate` deferred trigger.\n * @codeGenApi\n */\nexport function ɵɵdeferOnImmediate() {}  // TODO: implement runtime logic.\n\n\n/**\n * Creates runtime data structures for the `prefetch on immediate` deferred trigger.\n * @codeGenApi\n */\nexport function ɵɵdeferPrefetchOnImmediate() {}  // TODO: implement runtime logic.\n\n/**\n * Creates runtime data structures for the `on timer` deferred trigger.\n * @param delay Amount of time to wait before loading the content.\n * @codeGenApi\n */\nexport function ɵɵdeferOnTimer(delay: number) {}  // TODO: implement runtime logic.\n\n/**\n * Creates runtime data structures for the `prefetch on timer` deferred trigger.\n * @param delay Amount of time to wait before prefetching the content.\n * @codeGenApi\n */\nexport function ɵɵdeferPrefetchOnTimer(delay: number) {}  // TODO: implement runtime logic.\n\n/**\n * Creates runtime data structures for the `on hover` deferred trigger.\n * @codeGenApi\n */\nexport function ɵɵdeferOnHover() {}  // TODO: implement runtime logic.\n\n/**\n * Creates runtime data structures for the `prefetch on hover` deferred trigger.\n * @codeGenApi\n */\nexport function ɵɵdeferPrefetchOnHover() {}  // TODO: implement runtime logic.\n\n/**\n * Creates runtime data structures for the `on interaction` deferred trigger.\n * @param target Optional element on which to listen for hover events.\n * @codeGenApi\n */\nexport function ɵɵdeferOnInteraction(target?: unknown) {}  // TODO: implement runtime logic.\n\n/**\n * Creates runtime data structures for the `prefetch on interaction` deferred trigger.\n * @param target Optional element on which to listen for hover events.\n * @codeGenApi\n */\nexport function ɵɵdeferPrefetchOnInteraction(target?: unknown) {}  // TODO: implement runtime logic.\n\n/**\n * Creates runtime data structures for the `on viewport` deferred trigger.\n * @param target Optional element on which to listen for hover events.\n * @codeGenApi\n */\nexport function ɵɵdeferOnViewport(target?: unknown) {}  // TODO: implement runtime logic.\n\n/**\n * Creates runtime data structures for the `prefetch on viewport` deferred trigger.\n * @param target Optional element on which to listen for hover events.\n * @codeGenApi\n */\nexport function ɵɵdeferPrefetchOnViewport(target?: unknown) {}  // TODO: implement runtime logic.\n\n/********** Helper functions **********/\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 An optional LView that hosts an instance of a defer block. LView is\n *    used to register a cleanup callback in case that LView got destroyed before\n *    callback was invoked. In this case, an `idle` callback is never invoked. This is\n *    helpful for cases when a defer block has scheduled rendering, but an underlying\n *    LView got destroyed prior to th block rendering.\n */\nfunction onIdle(callback: VoidFunction, lView: LView|null) {\n  let id: number;\n  const removeIdleCallback = () => _cancelIdleCallback(id);\n  id = _requestIdleCallback(() => {\n         removeIdleCallback();\n         if (lView !== null) {\n           // The idle callback is invoked, we no longer need\n           // to retain a cleanup callback in an LView.\n           removeLViewOnDestroy(lView, removeIdleCallback);\n         }\n         callback();\n       }) as number;\n\n  if (lView !== null) {\n    // Store a cleanup function on LView, so that we cancel idle\n    // callback in case this LView is destroyed before a callback\n    // is invoked.\n    storeLViewOnDestroy(lView, removeIdleCallback);\n  }\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 */\nfunction 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. */\nfunction 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. */\nfunction 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. */\nfunction 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`. */\nfunction 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\n/**\n * Transitions a defer block to the new state. Updates the  necessary\n * data structures and renders corresponding block.\n *\n * @param newState New state that should be applied to the defer block.\n * @param tNode TNode that represents a defer block.\n * @param lContainer Represents an instance of a defer block.\n * @param stateTmplIndex Index of a template that should be rendered.\n */\nfunction renderDeferBlockState(\n    newState: DeferBlockInstanceState, tNode: TNode, lContainer: LContainer,\n    stateTmplIndex: number|null): void {\n  const hostLView = lContainer[PARENT];\n\n  // Check if this view is not destroyed. Since the loading process was async,\n  // the view might end up being destroyed by the time rendering happens.\n  if (isDestroyed(hostLView)) return;\n\n  // Make sure this TNode belongs to TView that represents host LView.\n  ngDevMode && assertTNodeForLView(tNode, hostLView);\n\n  const lDetails = getLDeferBlockDetails(hostLView, tNode);\n\n  ngDevMode && assertDefined(lDetails, 'Expected a defer block state defined');\n\n  // Note: we transition to the next state if the previous state was represented\n  // with a number that is less than the next state. For example, if the current\n  // state is \"loading\" (represented as `2`), we should not show a placeholder\n  // (represented as `1`).\n  if (lDetails[DEFER_BLOCK_STATE] < newState && stateTmplIndex !== null) {\n    lDetails[DEFER_BLOCK_STATE] = newState;\n    const hostTView = hostLView[TVIEW];\n    const adjustedIndex = stateTmplIndex + HEADER_OFFSET;\n    const tNode = getTNode(hostTView, adjustedIndex) as TContainerNode;\n\n    // There is only 1 view that can be present in an LContainer that\n    // represents a `{#defer}` block, so always refer to the first one.\n    const viewIndex = 0;\n\n    removeLViewFromLContainer(lContainer, viewIndex);\n\n    const dehydratedView = findMatchingDehydratedView(lContainer, tNode.tView!.ssrId);\n    const embeddedLView = createAndRenderEmbeddedLView(hostLView, tNode, null, {dehydratedView});\n    addLViewToLContainer(\n        lContainer, embeddedLView, viewIndex, shouldAddViewToDom(tNode, dehydratedView));\n  }\n}\n\n/**\n * Trigger loading of defer block dependencies if the process hasn't started yet.\n *\n * @param tDetails Static information about this defer block.\n * @param tView TView of a host view.\n * @param lView LView of a host view.\n */\nfunction triggerResourceLoading(tDetails: TDeferBlockDetails, tView: TView, lView: LView) {\n  const injector = lView[INJECTOR]!;\n\n  if (!shouldTriggerDeferBlock(injector) ||\n      (tDetails.loadingState !== DeferDependenciesLoadingState.NOT_STARTED &&\n       tDetails.loadingState !== DeferDependenciesLoadingState.SCHEDULED)) {\n    // If the loading status is different from initial one, it means that\n    // the loading of dependencies is in progress and there is nothing to do\n    // in this function. All details can be obtained from the `tDetails` object.\n    return;\n  }\n\n  const primaryBlockTNode = getPrimaryBlockTNode(tView, tDetails);\n\n  // Switch from NOT_STARTED -> IN_PROGRESS state.\n  tDetails.loadingState = DeferDependenciesLoadingState.IN_PROGRESS;\n\n  // Check if dependency function interceptor is configured.\n  const deferDependencyInterceptor =\n      injector.get(DEFER_BLOCK_DEPENDENCY_INTERCEPTOR, null, {optional: true});\n\n  const dependenciesFn = deferDependencyInterceptor ?\n      deferDependencyInterceptor.intercept(tDetails.dependencyResolverFn) :\n      tDetails.dependencyResolverFn;\n\n  // The `dependenciesFn` might be `null` when all dependencies within\n  // a given `{#defer}` block were eagerly references elsewhere in a file,\n  // thus no dynamic `import()`s were produced.\n  if (!dependenciesFn) {\n    tDetails.loadingPromise = Promise.resolve().then(() => {\n      tDetails.loadingState = DeferDependenciesLoadingState.COMPLETE;\n    });\n    return;\n  }\n\n  // Start downloading of defer block dependencies.\n  tDetails.loadingPromise = Promise.allSettled(dependenciesFn()).then(results => {\n    let failed = false;\n    const directiveDefs: DirectiveDefList = [];\n    const pipeDefs: PipeDefList = [];\n\n    for (const result of results) {\n      if (result.status === 'fulfilled') {\n        const dependency = result.value;\n        const directiveDef = getComponentDef(dependency) || getDirectiveDef(dependency);\n        if (directiveDef) {\n          directiveDefs.push(directiveDef);\n        } else {\n          const pipeDef = getPipeDef(dependency);\n          if (pipeDef) {\n            pipeDefs.push(pipeDef);\n          }\n        }\n      } else {\n        failed = true;\n        break;\n      }\n    }\n\n    // Loading is completed, we no longer need this Promise.\n    tDetails.loadingPromise = null;\n\n    if (failed) {\n      tDetails.loadingState = DeferDependenciesLoadingState.FAILED;\n    } else {\n      tDetails.loadingState = DeferDependenciesLoadingState.COMPLETE;\n\n      // Update directive and pipe registries to add newly downloaded dependencies.\n      const primaryBlockTView = primaryBlockTNode.tView!;\n      if (directiveDefs.length > 0) {\n        primaryBlockTView.directiveRegistry = primaryBlockTView.directiveRegistry ?\n            [...primaryBlockTView.directiveRegistry, ...directiveDefs] :\n            directiveDefs;\n      }\n      if (pipeDefs.length > 0) {\n        primaryBlockTView.pipeRegistry = primaryBlockTView.pipeRegistry ?\n            [...primaryBlockTView.pipeRegistry, ...pipeDefs] :\n            pipeDefs;\n      }\n    }\n  });\n}\n\n/** Utility function to render `{:placeholder}` content (if present) */\nfunction renderPlaceholder(lView: LView, tNode: TNode) {\n  const tView = lView[TVIEW];\n  const lContainer = lView[tNode.index];\n  ngDevMode && assertLContainer(lContainer);\n\n  const tDetails = getTDeferBlockDetails(tView, tNode);\n  renderDeferBlockState(\n      DeferBlockInstanceState.PLACEHOLDER, tNode, lContainer, tDetails.placeholderTmplIndex);\n}\n\n/**\n * Subscribes to the \"loading\" Promise and renders corresponding defer sub-block,\n * based on the loading results.\n *\n * @param lContainer Represents an instance of a defer block.\n * @param tNode Represents defer block info shared across all instances.\n */\nfunction renderDeferStateAfterResourceLoading(\n    tDetails: TDeferBlockDetails, tNode: TNode, lContainer: LContainer) {\n  ngDevMode &&\n      assertDefined(\n          tDetails.loadingPromise, 'Expected loading Promise to exist on this defer block');\n\n  tDetails.loadingPromise!.then(() => {\n    if (tDetails.loadingState === DeferDependenciesLoadingState.COMPLETE) {\n      ngDevMode && assertDeferredDependenciesLoaded(tDetails);\n\n      // Everything is loaded, show the primary block content\n      renderDeferBlockState(\n          DeferBlockInstanceState.COMPLETE, tNode, lContainer, tDetails.primaryTmplIndex);\n\n    } else if (tDetails.loadingState === DeferDependenciesLoadingState.FAILED) {\n      renderDeferBlockState(\n          DeferBlockInstanceState.ERROR, tNode, lContainer, tDetails.errorTmplIndex);\n    }\n  });\n}\n\n/** Retrieves a TNode that represents main content of a defer block. */\nfunction getPrimaryBlockTNode(tView: TView, tDetails: TDeferBlockDetails): TContainerNode {\n  const adjustedIndex = tDetails.primaryTmplIndex + HEADER_OFFSET;\n  return getTNode(tView, adjustedIndex) as TContainerNode;\n}\n\n/**\n * Attempts to trigger loading of defer block dependencies.\n * If the block is already in a loading, completed or an error state -\n * no additional actions are taken.\n */\nfunction triggerDeferBlock(lView: LView, tNode: TNode) {\n  const tView = lView[TVIEW];\n  const lContainer = lView[tNode.index];\n  const injector = lView[INJECTOR]!;\n  ngDevMode && assertLContainer(lContainer);\n\n  if (!shouldTriggerDeferBlock(injector)) return;\n\n  const tDetails = getTDeferBlockDetails(tView, tNode);\n\n  // Condition is triggered, try to render loading state and start downloading.\n  // Note: if a block is in a loading, completed or an error state, this call would be a noop.\n  renderDeferBlockState(\n      DeferBlockInstanceState.LOADING, tNode, lContainer, tDetails.loadingTmplIndex);\n\n  switch (tDetails.loadingState) {\n    case DeferDependenciesLoadingState.NOT_STARTED:\n    case DeferDependenciesLoadingState.SCHEDULED:\n      triggerResourceLoading(tDetails, lView[TVIEW], lView);\n\n      // The `loadingState` might have changed to \"loading\".\n      if ((tDetails.loadingState as DeferDependenciesLoadingState) ===\n          DeferDependenciesLoadingState.IN_PROGRESS) {\n        renderDeferStateAfterResourceLoading(tDetails, tNode, lContainer);\n      }\n      break;\n    case DeferDependenciesLoadingState.IN_PROGRESS:\n      renderDeferStateAfterResourceLoading(tDetails, tNode, lContainer);\n      break;\n    case DeferDependenciesLoadingState.COMPLETE:\n      ngDevMode && assertDeferredDependenciesLoaded(tDetails);\n      renderDeferBlockState(\n          DeferBlockInstanceState.COMPLETE, tNode, lContainer, tDetails.primaryTmplIndex);\n      break;\n    case DeferDependenciesLoadingState.FAILED:\n      renderDeferBlockState(\n          DeferBlockInstanceState.ERROR, tNode, lContainer, tDetails.errorTmplIndex);\n      break;\n    default:\n      if (ngDevMode) {\n        throwError('Unknown defer block state');\n      }\n  }\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 */\nfunction assertDeferredDependenciesLoaded(tDetails: TDeferBlockDetails) {\n  assertEqual(\n      tDetails.loadingState, DeferDependenciesLoadingState.COMPLETE,\n      'Expecting all deferred dependencies to be loaded.');\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\n/**\n * **INTERNAL**, avoid referencing it in application code.\n *\n * Injector token that allows to provide `DeferBlockDependencyInterceptor` class\n * implementation.\n */\nexport const DEFER_BLOCK_DEPENDENCY_INTERCEPTOR =\n    new InjectionToken<DeferBlockDependencyInterceptor>(\n        ngDevMode ? 'DEFER_BLOCK_DEPENDENCY_INTERCEPTOR' : '');\n"]}
|