2017-12-01 14:23:03 -08:00
|
|
|
/**
|
|
|
|
* @license
|
2020-05-19 12:08:49 -07:00
|
|
|
* Copyright Google LLC All Rights Reserved.
|
2017-12-01 14:23:03 -08:00
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2019-01-23 15:23:28 -08:00
|
|
|
import {ViewEncapsulation} from '../metadata/view';
|
2020-09-14 11:21:15 -07:00
|
|
|
import {Renderer2} from '../render/api';
|
2019-07-19 15:42:08 +02:00
|
|
|
import {addToArray, removeFromArray} from '../util/array_utils';
|
2020-09-25 15:01:56 -07:00
|
|
|
import {assertDefined, assertDomNode, assertEqual, assertIndexInRange, assertString} from '../util/assert';
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
|
2019-06-07 20:46:11 -07:00
|
|
|
import {assertLContainer, assertLView, assertTNodeForLView} from './assert';
|
2018-10-12 15:02:54 -07:00
|
|
|
import {attachPatchData} from './context_discovery';
|
2020-05-12 10:35:52 -07:00
|
|
|
import {CONTAINER_HEADER_OFFSET, HAS_TRANSPLANTED_VIEWS, LContainer, MOVED_VIEWS, NATIVE, unusedValueExportToPlacateAjd as unused1} from './interfaces/container';
|
2019-01-23 20:00:05 +01:00
|
|
|
import {ComponentDef} from './interfaces/definition';
|
2020-09-25 15:01:56 -07:00
|
|
|
import {icuContainerIterate} from './interfaces/i18n';
|
2019-01-31 10:38:43 +01:00
|
|
|
import {NodeInjectorFactory} from './interfaces/injector';
|
2020-09-25 15:01:56 -07:00
|
|
|
import {TElementNode, TIcuContainerNode, TNode, TNodeFlags, TNodeType, TProjectionNode, unusedValueExportToPlacateAjd as unused2} from './interfaces/node';
|
2018-01-25 15:32:21 +01:00
|
|
|
import {unusedValueExportToPlacateAjd as unused3} from './interfaces/projection';
|
2020-09-25 15:01:56 -07:00
|
|
|
import {isProceduralRenderer, ProceduralRenderer3, RComment, RElement, Renderer3, RNode, RText, unusedValueExportToPlacateAjd as unused4} from './interfaces/renderer';
|
2019-11-22 16:33:05 +01:00
|
|
|
import {isLContainer, isLView} from './interfaces/type_checks';
|
2020-09-14 11:21:15 -07:00
|
|
|
import {CHILD_HEAD, CLEANUP, DECLARATION_COMPONENT_VIEW, DECLARATION_LCONTAINER, DestroyHookData, FLAGS, HookData, HookFn, HOST, LView, LViewFlags, NEXT, PARENT, QUERIES, RENDERER, T_HOST, TVIEW, TView, TViewType, unusedValueExportToPlacateAjd as unused5} from './interfaces/view';
|
2019-06-07 20:46:11 -07:00
|
|
|
import {assertNodeOfPossibleTypes, assertNodeType} from './node_assert';
|
2019-11-13 16:22:55 -08:00
|
|
|
import {getLViewParent} from './util/view_traversal_utils';
|
2020-09-14 13:06:05 -07:00
|
|
|
import {getNativeByTNode, unwrapRNode, updateTransplantedViewCount} from './util/view_utils';
|
2017-12-14 15:03:46 -08:00
|
|
|
|
2018-01-11 13:09:21 -08:00
|
|
|
const unusedValueToPlacateAjd = unused1 + unused2 + unused3 + unused4 + unused5;
|
2017-12-20 10:47:22 -08:00
|
|
|
|
2018-09-08 10:55:41 -07:00
|
|
|
const enum WalkTNodeTreeAction {
|
2019-06-07 20:46:11 -07:00
|
|
|
/** node create in the native environment. Run on initial creation. */
|
|
|
|
Create = 0,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* node insert in the native environment.
|
|
|
|
* Run when existing node has been detached and needs to be re-attached.
|
|
|
|
*/
|
|
|
|
Insert = 1,
|
2018-05-31 15:45:46 +02:00
|
|
|
|
|
|
|
/** node detach from the native environment */
|
2019-06-07 20:46:11 -07:00
|
|
|
Detach = 2,
|
2018-05-31 15:45:46 +02:00
|
|
|
|
|
|
|
/** node destruction using the renderer's API */
|
2019-06-07 20:46:11 -07:00
|
|
|
Destroy = 3,
|
2018-05-31 15:45:46 +02:00
|
|
|
}
|
|
|
|
|
2018-07-03 20:04:36 -07:00
|
|
|
|
2018-06-06 17:30:48 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* NOTE: for performance reasons, the possible actions are inlined within the function instead of
|
|
|
|
* being passed as an argument.
|
|
|
|
*/
|
2019-06-07 20:46:11 -07:00
|
|
|
function applyToElementOrContainer(
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
action: WalkTNodeTreeAction, renderer: Renderer3, parent: RElement|null,
|
|
|
|
lNodeToHandle: RNode|LContainer|LView, beforeNode?: RNode|null) {
|
2019-07-22 20:55:07 -07:00
|
|
|
// If this slot was allocated for a text node dynamically created by i18n, the text node itself
|
|
|
|
// won't be created until i18nApply() in the update block, so this node should be skipped.
|
|
|
|
// For more info, see "ICU expressions should work inside an ngTemplateOutlet inside an ngFor"
|
|
|
|
// in `i18n_spec.ts`.
|
|
|
|
if (lNodeToHandle != null) {
|
|
|
|
let lContainer: LContainer|undefined;
|
|
|
|
let isComponent = false;
|
|
|
|
// We are expecting an RNode, but in the case of a component or LContainer the `RNode` is
|
2019-06-07 20:46:11 -07:00
|
|
|
// wrapped in an array which needs to be unwrapped. We need to know if it is a component and if
|
2019-07-22 20:55:07 -07:00
|
|
|
// it has LContainer so that we can process all of those cases appropriately.
|
|
|
|
if (isLContainer(lNodeToHandle)) {
|
|
|
|
lContainer = lNodeToHandle;
|
|
|
|
} else if (isLView(lNodeToHandle)) {
|
|
|
|
isComponent = true;
|
|
|
|
ngDevMode && assertDefined(lNodeToHandle[HOST], 'HOST must be defined for a component LView');
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
lNodeToHandle = lNodeToHandle[HOST]!;
|
2019-07-22 20:55:07 -07:00
|
|
|
}
|
|
|
|
const rNode: RNode = unwrapRNode(lNodeToHandle);
|
2019-10-14 14:46:26 -07:00
|
|
|
ngDevMode && !isProceduralRenderer(renderer) && assertDomNode(rNode);
|
2019-06-07 20:46:11 -07:00
|
|
|
|
2019-06-07 20:46:11 -07:00
|
|
|
if (action === WalkTNodeTreeAction.Create && parent !== null) {
|
|
|
|
if (beforeNode == null) {
|
|
|
|
nativeAppendChild(renderer, parent, rNode);
|
|
|
|
} else {
|
2020-09-25 15:01:56 -07:00
|
|
|
nativeInsertBefore(renderer, parent, rNode, beforeNode || null, true);
|
2019-06-07 20:46:11 -07:00
|
|
|
}
|
|
|
|
} else if (action === WalkTNodeTreeAction.Insert && parent !== null) {
|
2020-09-25 15:01:56 -07:00
|
|
|
nativeInsertBefore(renderer, parent, rNode, beforeNode || null, true);
|
2019-07-22 20:55:07 -07:00
|
|
|
} else if (action === WalkTNodeTreeAction.Detach) {
|
|
|
|
nativeRemoveNode(renderer, rNode, isComponent);
|
|
|
|
} else if (action === WalkTNodeTreeAction.Destroy) {
|
|
|
|
ngDevMode && ngDevMode.rendererDestroyNode++;
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
(renderer as ProceduralRenderer3).destroyNode!(rNode);
|
2019-07-22 20:55:07 -07:00
|
|
|
}
|
|
|
|
if (lContainer != null) {
|
2019-06-07 20:46:11 -07:00
|
|
|
applyContainer(renderer, action, lContainer, parent, beforeNode);
|
2019-07-22 20:55:07 -07:00
|
|
|
}
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2020-09-25 15:01:56 -07:00
|
|
|
export function createTextNode(renderer: Renderer3, value: string): RText {
|
2019-08-27 15:16:50 +02:00
|
|
|
ngDevMode && ngDevMode.rendererCreateTextNode++;
|
|
|
|
ngDevMode && ngDevMode.rendererSetText++;
|
|
|
|
return isProceduralRenderer(renderer) ? renderer.createText(value) :
|
|
|
|
renderer.createTextNode(value);
|
2018-04-10 20:57:09 -07:00
|
|
|
}
|
|
|
|
|
2020-09-25 15:01:56 -07:00
|
|
|
export function updateTextNode(renderer: Renderer3, rNode: RText, value: string): void {
|
|
|
|
ngDevMode && ngDevMode.rendererSetText++;
|
|
|
|
isProceduralRenderer(renderer) ? renderer.setValue(rNode, value) : rNode.textContent = value;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function createCommentNode(renderer: Renderer3, value: string): RComment {
|
|
|
|
ngDevMode && ngDevMode.rendererCreateComment++;
|
|
|
|
// isProceduralRenderer check is not needed because both `Renderer2` and `Renderer3` have the same
|
|
|
|
// method name.
|
|
|
|
return renderer.createComment(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a native element from a tag name, using a renderer.
|
|
|
|
* @param renderer A renderer to use
|
|
|
|
* @param name the tag name
|
|
|
|
* @param namespace Optional namespace for element.
|
|
|
|
* @returns the element created
|
|
|
|
*/
|
|
|
|
export function createElementNode(
|
|
|
|
renderer: Renderer3, name: string, namespace: string|null): RElement {
|
|
|
|
ngDevMode && ngDevMode.rendererCreateElement++;
|
|
|
|
if (isProceduralRenderer(renderer)) {
|
|
|
|
return renderer.createElement(name, namespace);
|
|
|
|
} else {
|
|
|
|
return namespace === null ? renderer.createElement(name) :
|
|
|
|
renderer.createElementNS(namespace, name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2020-09-14 11:21:15 -07:00
|
|
|
* Removes all DOM elements associated with a view.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
|
|
|
* Because some root nodes of the view may be containers, we sometimes need
|
|
|
|
* to propagate deeply into the nested containers to remove all elements in the
|
|
|
|
* views beneath it.
|
|
|
|
*
|
2020-01-30 14:57:44 -08:00
|
|
|
* @param tView The `TView' of the `LView` from which elements should be added or removed
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param lView The view from which elements should be added or removed
|
2020-09-14 11:21:15 -07:00
|
|
|
*/
|
|
|
|
export function removeViewFromContainer(tView: TView, lView: LView): void {
|
|
|
|
const renderer = lView[RENDERER];
|
|
|
|
applyView(tView, lView, renderer, WalkTNodeTreeAction.Detach, null, null);
|
|
|
|
lView[HOST] = null;
|
|
|
|
lView[T_HOST] = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Adds all DOM elements associated with a view.
|
|
|
|
*
|
|
|
|
* Because some root nodes of the view may be containers, we sometimes need
|
|
|
|
* to propagate deeply into the nested containers to add all elements in the
|
|
|
|
* views beneath it.
|
|
|
|
*
|
|
|
|
* @param tView The `TView' of the `LView` from which elements should be added or removed
|
|
|
|
* @param parentTNode The `TNode` where the `LView` should be attached to.
|
|
|
|
* @param renderer Current renderer to use for DOM manipulations.
|
|
|
|
* @param lView The view from which elements should be added or removed
|
|
|
|
* @param parentNativeNode The parent `RElement` where it should be inserted into.
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param beforeNode The node before which elements should be added, if insert mode
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2020-09-14 11:21:15 -07:00
|
|
|
export function addViewToContainer(
|
|
|
|
tView: TView, parentTNode: TNode, renderer: Renderer3, lView: LView, parentNativeNode: RElement,
|
|
|
|
beforeNode: RNode|null): void {
|
|
|
|
lView[HOST] = parentNativeNode;
|
|
|
|
lView[T_HOST] = parentTNode;
|
|
|
|
applyView(tView, lView, renderer, WalkTNodeTreeAction.Insert, parentNativeNode, beforeNode);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2020-09-14 11:21:15 -07:00
|
|
|
|
2019-03-07 16:09:12 +01:00
|
|
|
/**
|
|
|
|
* Detach a `LView` from the DOM by detaching its nodes.
|
|
|
|
*
|
2020-01-30 14:57:44 -08:00
|
|
|
* @param tView The `TView' of the `LView` to be detached
|
2019-03-07 16:09:12 +01:00
|
|
|
* @param lView the `LView` to be detached.
|
|
|
|
*/
|
2020-01-30 14:57:44 -08:00
|
|
|
export function renderDetachView(tView: TView, lView: LView) {
|
|
|
|
applyView(tView, lView, lView[RENDERER], WalkTNodeTreeAction.Detach, null, null);
|
2019-03-07 16:09:12 +01:00
|
|
|
}
|
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
/**
|
2018-04-19 22:59:58 +02:00
|
|
|
* Traverses down and up the tree of views and containers to remove listeners and
|
2017-12-13 11:04:46 -08:00
|
|
|
* call onDestroy callbacks.
|
2017-12-01 14:23:03 -08:00
|
|
|
*
|
|
|
|
* Notes:
|
2017-12-13 11:04:46 -08:00
|
|
|
* - Because it's used for onDestroy calls, it needs to be bottom-up.
|
2017-12-01 14:23:03 -08:00
|
|
|
* - Must process containers instead of their views to avoid splicing
|
|
|
|
* when views are destroyed and re-added.
|
2017-12-13 11:04:46 -08:00
|
|
|
* - Using a while loop because it's faster than recursion
|
2017-12-01 14:23:03 -08:00
|
|
|
* - Destroy only called on movement to sibling or movement to parent (laterally or up)
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param rootView The view to destroy
|
2017-12-01 14:23:03 -08:00
|
|
|
*/
|
2018-11-21 21:14:06 -08:00
|
|
|
export function destroyViewTree(rootView: LView): void {
|
2018-05-30 13:43:14 -07:00
|
|
|
// If the view has no children, we can clean it up and return early.
|
2019-01-28 14:45:31 -08:00
|
|
|
let lViewOrLContainer = rootView[CHILD_HEAD];
|
|
|
|
if (!lViewOrLContainer) {
|
2020-01-30 14:57:44 -08:00
|
|
|
return cleanUpView(rootView[TVIEW], rootView);
|
2018-04-19 22:59:58 +02:00
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2019-01-28 14:45:31 -08:00
|
|
|
while (lViewOrLContainer) {
|
2018-11-21 21:14:06 -08:00
|
|
|
let next: LView|LContainer|null = null;
|
2018-06-07 22:42:32 -07:00
|
|
|
|
2019-01-28 14:45:31 -08:00
|
|
|
if (isLView(lViewOrLContainer)) {
|
2018-11-21 21:14:06 -08:00
|
|
|
// If LView, traverse down to child.
|
2019-01-28 14:45:31 -08:00
|
|
|
next = lViewOrLContainer[CHILD_HEAD];
|
2019-02-05 21:07:03 -08:00
|
|
|
} else {
|
2019-01-28 14:45:31 -08:00
|
|
|
ngDevMode && assertLContainer(lViewOrLContainer);
|
2019-02-05 21:07:03 -08:00
|
|
|
// If container, traverse down to its first LView.
|
2019-04-29 21:17:13 +02:00
|
|
|
const firstView: LView|undefined = lViewOrLContainer[CONTAINER_HEADER_OFFSET];
|
|
|
|
if (firstView) next = firstView;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2019-01-28 14:45:31 -08:00
|
|
|
if (!next) {
|
2018-06-07 22:42:32 -07:00
|
|
|
// Only clean up view when moving to the side or up, as destroy hooks
|
|
|
|
// should be called in order from the bottom up.
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
while (lViewOrLContainer && !lViewOrLContainer![NEXT] && lViewOrLContainer !== rootView) {
|
2020-09-14 13:06:05 -07:00
|
|
|
if (isLView(lViewOrLContainer)) {
|
|
|
|
cleanUpView(lViewOrLContainer[TVIEW], lViewOrLContainer);
|
|
|
|
}
|
|
|
|
lViewOrLContainer = lViewOrLContainer[PARENT];
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2020-01-30 14:57:44 -08:00
|
|
|
if (lViewOrLContainer === null) lViewOrLContainer = rootView;
|
2020-09-14 13:06:05 -07:00
|
|
|
if (isLView(lViewOrLContainer)) {
|
|
|
|
cleanUpView(lViewOrLContainer[TVIEW], lViewOrLContainer);
|
|
|
|
}
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
next = lViewOrLContainer && lViewOrLContainer![NEXT];
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2019-01-28 14:45:31 -08:00
|
|
|
lViewOrLContainer = next;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Inserts a view into a container.
|
|
|
|
*
|
2017-12-13 19:34:46 -08:00
|
|
|
* This adds the view to the container's array of active views in the correct
|
2017-12-13 11:04:46 -08:00
|
|
|
* position. It also adds the view's elements to the DOM if the container isn't a
|
|
|
|
* root node of another view (in that case, the view's elements will be added when
|
|
|
|
* the container's parent view is added later).
|
|
|
|
*
|
2020-01-30 14:57:44 -08:00
|
|
|
* @param tView The `TView' of the `LView` to insert
|
2018-09-17 14:32:45 -07:00
|
|
|
* @param lView The view to insert
|
|
|
|
* @param lContainer The container into which the view should be inserted
|
2019-01-28 14:45:31 -08:00
|
|
|
* @param index Which index in the container to insert the child view into
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2020-01-30 14:57:44 -08:00
|
|
|
export function insertView(tView: TView, lView: LView, lContainer: LContainer, index: number) {
|
2019-01-28 14:45:31 -08:00
|
|
|
ngDevMode && assertLView(lView);
|
|
|
|
ngDevMode && assertLContainer(lContainer);
|
2019-04-29 21:17:13 +02:00
|
|
|
const indexInContainer = CONTAINER_HEADER_OFFSET + index;
|
|
|
|
const containerLength = lContainer.length;
|
|
|
|
|
2017-12-01 14:23:03 -08:00
|
|
|
if (index > 0) {
|
|
|
|
// This is a new view, we need to add it to the children.
|
2019-04-29 21:17:13 +02:00
|
|
|
lContainer[indexInContainer - 1][NEXT] = lView;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2019-04-29 21:17:13 +02:00
|
|
|
if (index < containerLength - CONTAINER_HEADER_OFFSET) {
|
|
|
|
lView[NEXT] = lContainer[indexInContainer];
|
2019-07-19 15:42:08 +02:00
|
|
|
addToArray(lContainer, CONTAINER_HEADER_OFFSET + index, lView);
|
2018-03-08 12:10:20 +01:00
|
|
|
} else {
|
2019-04-29 21:17:13 +02:00
|
|
|
lContainer.push(lView);
|
2018-06-26 11:39:56 -07:00
|
|
|
lView[NEXT] = null;
|
|
|
|
}
|
|
|
|
|
2019-01-28 14:45:31 -08:00
|
|
|
lView[PARENT] = lContainer;
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2019-06-07 10:55:48 +02:00
|
|
|
// track views where declaration and insertion points are different
|
|
|
|
const declarationLContainer = lView[DECLARATION_LCONTAINER];
|
|
|
|
if (declarationLContainer !== null && lContainer !== declarationLContainer) {
|
|
|
|
trackMovedView(declarationLContainer, lView);
|
|
|
|
}
|
|
|
|
|
|
|
|
// notify query that a new view has been added
|
|
|
|
const lQueries = lView[QUERIES];
|
|
|
|
if (lQueries !== null) {
|
2020-01-30 14:57:44 -08:00
|
|
|
lQueries.insertView(tView);
|
2018-05-28 11:57:36 +02:00
|
|
|
}
|
|
|
|
|
2018-05-31 15:45:46 +02:00
|
|
|
// Sets the attached flag
|
2018-06-26 11:39:56 -07:00
|
|
|
lView[FLAGS] |= LViewFlags.Attached;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2019-06-07 10:55:48 +02:00
|
|
|
/**
|
|
|
|
* Track views created from the declaration container (TemplateRef) and inserted into a
|
|
|
|
* different LContainer.
|
|
|
|
*/
|
|
|
|
function trackMovedView(declarationContainer: LContainer, lView: LView) {
|
2019-11-07 06:32:59 +00:00
|
|
|
ngDevMode && assertDefined(lView, 'LView required');
|
2019-06-07 10:55:48 +02:00
|
|
|
ngDevMode && assertLContainer(declarationContainer);
|
2019-11-07 06:32:59 +00:00
|
|
|
const movedViews = declarationContainer[MOVED_VIEWS];
|
2019-11-08 15:13:22 -08:00
|
|
|
const insertedLContainer = lView[PARENT] as LContainer;
|
|
|
|
ngDevMode && assertLContainer(insertedLContainer);
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
const insertedComponentLView = insertedLContainer[PARENT]![DECLARATION_COMPONENT_VIEW];
|
2019-11-08 15:13:22 -08:00
|
|
|
ngDevMode && assertDefined(insertedComponentLView, 'Missing insertedComponentLView');
|
2020-01-13 15:12:16 -08:00
|
|
|
const declaredComponentLView = lView[DECLARATION_COMPONENT_VIEW];
|
|
|
|
ngDevMode && assertDefined(declaredComponentLView, 'Missing declaredComponentLView');
|
|
|
|
if (declaredComponentLView !== insertedComponentLView) {
|
|
|
|
// At this point the declaration-component is not same as insertion-component; this means that
|
|
|
|
// this is a transplanted view. Mark the declared lView as having transplanted views so that
|
|
|
|
// those views can participate in CD.
|
2020-05-12 10:35:52 -07:00
|
|
|
declarationContainer[HAS_TRANSPLANTED_VIEWS] = true;
|
2019-11-08 15:13:22 -08:00
|
|
|
}
|
2019-11-07 06:32:59 +00:00
|
|
|
if (movedViews === null) {
|
2019-06-07 10:55:48 +02:00
|
|
|
declarationContainer[MOVED_VIEWS] = [lView];
|
|
|
|
} else {
|
2019-11-07 06:32:59 +00:00
|
|
|
movedViews.push(lView);
|
2019-06-07 10:55:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function detachMovedView(declarationContainer: LContainer, lView: LView) {
|
|
|
|
ngDevMode && assertLContainer(declarationContainer);
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
ngDevMode &&
|
|
|
|
assertDefined(
|
|
|
|
declarationContainer[MOVED_VIEWS],
|
|
|
|
'A projected view should belong to a non-empty projected views collection');
|
|
|
|
const movedViews = declarationContainer[MOVED_VIEWS]!;
|
2020-01-13 15:12:16 -08:00
|
|
|
const declarationViewIndex = movedViews.indexOf(lView);
|
|
|
|
const insertionLContainer = lView[PARENT] as LContainer;
|
|
|
|
ngDevMode && assertLContainer(insertionLContainer);
|
|
|
|
|
|
|
|
// If the view was marked for refresh but then detached before it was checked (where the flag
|
|
|
|
// would be cleared and the counter decremented), we need to decrement the view counter here
|
|
|
|
// instead.
|
|
|
|
if (lView[FLAGS] & LViewFlags.RefreshTransplantedView) {
|
2020-09-09 10:49:35 -07:00
|
|
|
lView[FLAGS] &= ~LViewFlags.RefreshTransplantedView;
|
2020-01-13 15:12:16 -08:00
|
|
|
updateTransplantedViewCount(insertionLContainer, -1);
|
|
|
|
}
|
|
|
|
|
|
|
|
movedViews.splice(declarationViewIndex, 1);
|
2019-06-07 10:55:48 +02:00
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2018-05-31 15:45:46 +02:00
|
|
|
* Detaches a view from a container.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2019-07-19 15:42:08 +02:00
|
|
|
* This method removes the view from the container's array of active views. It also
|
2018-05-31 15:45:46 +02:00
|
|
|
* removes the view's elements from the DOM.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-09-17 14:32:45 -07:00
|
|
|
* @param lContainer The container from which to detach a view
|
2018-05-31 15:45:46 +02:00
|
|
|
* @param removeIndex The index of the view to detach
|
2018-12-04 11:56:24 +01:00
|
|
|
* @returns Detached LView instance.
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2019-04-19 19:46:52 +02:00
|
|
|
export function detachView(lContainer: LContainer, removeIndex: number): LView|undefined {
|
2019-04-29 21:17:13 +02:00
|
|
|
if (lContainer.length <= CONTAINER_HEADER_OFFSET) return;
|
|
|
|
|
|
|
|
const indexInContainer = CONTAINER_HEADER_OFFSET + removeIndex;
|
|
|
|
const viewToDetach = lContainer[indexInContainer];
|
2019-06-07 10:55:48 +02:00
|
|
|
|
2019-04-19 19:46:52 +02:00
|
|
|
if (viewToDetach) {
|
2019-06-07 10:55:48 +02:00
|
|
|
const declarationLContainer = viewToDetach[DECLARATION_LCONTAINER];
|
|
|
|
if (declarationLContainer !== null && declarationLContainer !== lContainer) {
|
|
|
|
detachMovedView(declarationLContainer, viewToDetach);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-04-19 19:46:52 +02:00
|
|
|
if (removeIndex > 0) {
|
2019-04-29 21:17:13 +02:00
|
|
|
lContainer[indexInContainer - 1][NEXT] = viewToDetach[NEXT] as LView;
|
2019-04-19 19:46:52 +02:00
|
|
|
}
|
2019-07-19 15:42:08 +02:00
|
|
|
const removedLView = removeFromArray(lContainer, CONTAINER_HEADER_OFFSET + removeIndex);
|
2020-09-14 11:21:15 -07:00
|
|
|
removeViewFromContainer(viewToDetach[TVIEW], viewToDetach);
|
2018-09-12 08:47:03 -07:00
|
|
|
|
2019-06-07 10:55:48 +02:00
|
|
|
// notify query that a view has been removed
|
|
|
|
const lQueries = removedLView[QUERIES];
|
|
|
|
if (lQueries !== null) {
|
|
|
|
lQueries.detachView(removedLView[TVIEW]);
|
2019-04-19 19:46:52 +02:00
|
|
|
}
|
2019-06-07 10:55:48 +02:00
|
|
|
|
2019-04-19 19:46:52 +02:00
|
|
|
viewToDetach[PARENT] = null;
|
|
|
|
viewToDetach[NEXT] = null;
|
|
|
|
// Unsets the attached flag
|
|
|
|
viewToDetach[FLAGS] &= ~LViewFlags.Attached;
|
2018-05-28 11:57:36 +02:00
|
|
|
}
|
2018-12-04 11:56:24 +01:00
|
|
|
return viewToDetach;
|
2018-05-31 15:45:46 +02:00
|
|
|
}
|
2018-05-28 11:57:36 +02:00
|
|
|
|
2018-05-31 15:45:46 +02:00
|
|
|
/**
|
|
|
|
* A standalone function which destroys an LView,
|
2020-01-30 14:57:44 -08:00
|
|
|
* conducting clean up (e.g. removing listeners, calling onDestroys).
|
2018-05-31 15:45:46 +02:00
|
|
|
*
|
2020-01-30 14:57:44 -08:00
|
|
|
* @param tView The `TView' of the `LView` to be destroyed
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param lView The view to be destroyed.
|
2018-05-31 15:45:46 +02:00
|
|
|
*/
|
2020-01-30 14:57:44 -08:00
|
|
|
export function destroyLView(tView: TView, lView: LView) {
|
2019-06-07 20:46:11 -07:00
|
|
|
if (!(lView[FLAGS] & LViewFlags.Destroyed)) {
|
|
|
|
const renderer = lView[RENDERER];
|
2019-01-28 15:23:53 -08:00
|
|
|
if (isProceduralRenderer(renderer) && renderer.destroyNode) {
|
2020-01-30 14:57:44 -08:00
|
|
|
applyView(tView, lView, renderer, WalkTNodeTreeAction.Destroy, null, null);
|
2019-01-28 15:23:53 -08:00
|
|
|
}
|
|
|
|
|
2019-06-07 20:46:11 -07:00
|
|
|
destroyViewTree(lView);
|
2018-05-31 15:45:46 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2018-12-14 15:11:14 +01:00
|
|
|
* Calls onDestroys hooks for all directives and pipes in a given view and then removes all
|
|
|
|
* listeners. Listeners are removed as the last step so events delivered in the onDestroys hooks
|
|
|
|
* can be propagated to @Output listeners.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2020-01-30 14:57:44 -08:00
|
|
|
* @param tView `TView` for the `LView` to clean up.
|
|
|
|
* @param lView The LView to clean up
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2020-01-30 14:57:44 -08:00
|
|
|
function cleanUpView(tView: TView, lView: LView): void {
|
|
|
|
if (!(lView[FLAGS] & LViewFlags.Destroyed)) {
|
2019-03-01 22:05:49 +01:00
|
|
|
// Usually the Attached flag is removed when the view is detached from its parent, however
|
|
|
|
// if it's a root view, the flag won't be unset hence why we're also removing on destroy.
|
2020-01-30 14:57:44 -08:00
|
|
|
lView[FLAGS] &= ~LViewFlags.Attached;
|
2019-03-01 22:05:49 +01:00
|
|
|
|
2019-01-28 15:23:53 -08:00
|
|
|
// Mark the LView as destroyed *before* executing the onDestroy hooks. An onDestroy hook
|
|
|
|
// runs arbitrary user code, which could include its own `viewRef.destroy()` (or similar). If
|
|
|
|
// We don't flag the view as destroyed before the hooks, this could lead to an infinite loop.
|
|
|
|
// This also aligns with the ViewEngine behavior. It also means that the onDestroy hook is
|
|
|
|
// really more of an "afterDestroy" hook if you think about it.
|
2020-01-30 14:57:44 -08:00
|
|
|
lView[FLAGS] |= LViewFlags.Destroyed;
|
2019-01-28 15:23:53 -08:00
|
|
|
|
2020-01-30 14:57:44 -08:00
|
|
|
executeOnDestroys(tView, lView);
|
|
|
|
removeListeners(tView, lView);
|
2020-09-14 11:21:15 -07:00
|
|
|
// For component views only, the local renderer is destroyed at clean up time.
|
|
|
|
if (lView[TVIEW].type === TViewType.Component && isProceduralRenderer(lView[RENDERER])) {
|
2018-06-05 15:28:15 -07:00
|
|
|
ngDevMode && ngDevMode.rendererDestroy++;
|
2020-01-30 14:57:44 -08:00
|
|
|
(lView[RENDERER] as ProceduralRenderer3).destroy();
|
2018-06-05 15:28:15 -07:00
|
|
|
}
|
2019-06-07 10:55:48 +02:00
|
|
|
|
2020-01-30 14:57:44 -08:00
|
|
|
const declarationContainer = lView[DECLARATION_LCONTAINER];
|
2019-06-07 10:55:48 +02:00
|
|
|
// we are dealing with an embedded view that is still inserted into a container
|
2020-01-30 14:57:44 -08:00
|
|
|
if (declarationContainer !== null && isLContainer(lView[PARENT])) {
|
2019-06-07 10:55:48 +02:00
|
|
|
// and this is a projected view
|
2020-01-30 14:57:44 -08:00
|
|
|
if (declarationContainer !== lView[PARENT]) {
|
|
|
|
detachMovedView(declarationContainer, lView);
|
2019-06-07 10:55:48 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
// For embedded views still attached to a container: remove query result from this view.
|
2020-01-30 14:57:44 -08:00
|
|
|
const lQueries = lView[QUERIES];
|
2019-06-07 10:55:48 +02:00
|
|
|
if (lQueries !== null) {
|
2020-01-30 14:57:44 -08:00
|
|
|
lQueries.detachView(tView);
|
2019-06-07 10:55:48 +02:00
|
|
|
}
|
2019-03-01 14:45:04 +01:00
|
|
|
}
|
2018-05-22 17:40:59 +02:00
|
|
|
}
|
2018-01-23 10:57:48 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Removes listeners and unsubscribes from output subscriptions */
|
2020-01-30 14:57:44 -08:00
|
|
|
function removeListeners(tView: TView, lView: LView): void {
|
|
|
|
const tCleanup = tView.cleanup;
|
2019-06-07 20:46:11 -07:00
|
|
|
if (tCleanup !== null) {
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
const lCleanup = lView[CLEANUP]!;
|
2018-11-28 15:54:38 -08:00
|
|
|
for (let i = 0; i < tCleanup.length - 1; i += 2) {
|
|
|
|
if (typeof tCleanup[i] === 'string') {
|
2019-04-05 15:30:04 +02:00
|
|
|
// This is a native DOM listener
|
2018-12-19 15:03:47 -08:00
|
|
|
const idxOrTargetGetter = tCleanup[i + 1];
|
|
|
|
const target = typeof idxOrTargetGetter === 'function' ?
|
|
|
|
idxOrTargetGetter(lView) :
|
2019-02-23 11:14:35 -08:00
|
|
|
unwrapRNode(lView[idxOrTargetGetter]);
|
2018-11-28 15:54:38 -08:00
|
|
|
const listener = lCleanup[tCleanup[i + 2]];
|
|
|
|
const useCaptureOrSubIdx = tCleanup[i + 3];
|
|
|
|
if (typeof useCaptureOrSubIdx === 'boolean') {
|
2019-04-05 15:30:04 +02:00
|
|
|
// native DOM listener registered with Renderer3
|
2018-12-19 15:03:47 -08:00
|
|
|
target.removeEventListener(tCleanup[i], listener, useCaptureOrSubIdx);
|
2018-11-28 15:54:38 -08:00
|
|
|
} else {
|
|
|
|
if (useCaptureOrSubIdx >= 0) {
|
|
|
|
// unregister
|
|
|
|
lCleanup[useCaptureOrSubIdx]();
|
|
|
|
} else {
|
|
|
|
// Subscription
|
|
|
|
lCleanup[-useCaptureOrSubIdx].unsubscribe();
|
|
|
|
}
|
|
|
|
}
|
2018-01-23 10:57:48 -08:00
|
|
|
i += 2;
|
|
|
|
} else {
|
2018-06-05 15:28:15 -07:00
|
|
|
// This is a cleanup function that is grouped with the index of its context
|
2018-11-28 15:54:38 -08:00
|
|
|
const context = lCleanup[tCleanup[i + 1]];
|
|
|
|
tCleanup[i].call(context);
|
2018-01-23 10:57:48 -08:00
|
|
|
}
|
|
|
|
}
|
2018-11-21 21:14:06 -08:00
|
|
|
lView[CLEANUP] = null;
|
2018-01-23 10:57:48 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Calls onDestroy hooks for this view */
|
2020-01-30 14:57:44 -08:00
|
|
|
function executeOnDestroys(tView: TView, lView: LView): void {
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
let destroyHooks: DestroyHookData|null;
|
2019-01-31 10:38:43 +01:00
|
|
|
|
2018-01-23 10:57:48 -08:00
|
|
|
if (tView != null && (destroyHooks = tView.destroyHooks) != null) {
|
2019-01-31 10:38:43 +01:00
|
|
|
for (let i = 0; i < destroyHooks.length; i += 2) {
|
2020-01-30 14:57:44 -08:00
|
|
|
const context = lView[destroyHooks[i] as number];
|
2019-01-31 10:38:43 +01:00
|
|
|
|
|
|
|
// Only call the destroy hook if the context has been requested.
|
|
|
|
if (!(context instanceof NodeInjectorFactory)) {
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
const toCall = destroyHooks[i + 1] as HookFn | HookData;
|
|
|
|
|
|
|
|
if (Array.isArray(toCall)) {
|
|
|
|
for (let j = 0; j < toCall.length; j += 2) {
|
|
|
|
(toCall[j + 1] as HookFn).call(context[toCall[j] as number]);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
toCall.call(context);
|
|
|
|
}
|
2019-01-31 10:38:43 +01:00
|
|
|
}
|
|
|
|
}
|
2018-03-21 15:10:34 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-06 17:46:49 +02:00
|
|
|
/**
|
2019-01-09 16:33:25 +01:00
|
|
|
* Returns a native element if a node can be inserted into the given parent.
|
2018-06-26 11:39:56 -07:00
|
|
|
*
|
2018-06-22 15:37:38 +02:00
|
|
|
* There are two reasons why we may not be able to insert a element immediately.
|
2018-06-26 11:39:56 -07:00
|
|
|
* - Projection: When creating a child content element of a component, we have to skip the
|
2018-06-22 15:37:38 +02:00
|
|
|
* insertion because the content of a component will be projected.
|
|
|
|
* `<component><content>delayed due to projection</content></component>`
|
2018-06-26 11:39:56 -07:00
|
|
|
* - Parent container is disconnected: This can happen when we are inserting a view into
|
|
|
|
* parent container, which itself is disconnected. For example the parent container is part
|
2019-01-23 20:00:05 +01:00
|
|
|
* of a View which has not be inserted or is made for projection but has not been inserted
|
2018-06-22 15:37:38 +02:00
|
|
|
* into destination.
|
2020-09-25 15:01:56 -07:00
|
|
|
*
|
|
|
|
* @param tView: Current `TView`.
|
|
|
|
* @param tNode: `TNode` for which we wish to retrieve render parent.
|
|
|
|
* @param lView: Current `LView`.
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2020-09-25 15:01:56 -07:00
|
|
|
export function getParentRElement(tView: TView, tNode: TNode, lView: LView): RElement|null {
|
|
|
|
return getClosestRElement(tView, tNode.parent, lView);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get closest `RElement` or `null` if it can't be found.
|
|
|
|
*
|
|
|
|
* If `TNode` is `TNodeType.Element` => return `RElement` at `LView[tNode.index]` location.
|
|
|
|
* If `TNode` is `TNodeType.ElementContainer|IcuContain` => return the parent (recursively).
|
|
|
|
* If `TNode` is `null` then return host `RElement`:
|
|
|
|
* - return `null` if projection
|
|
|
|
* - return `null` if parent container is disconnected (we have no parent.)
|
|
|
|
*
|
|
|
|
* @param tView: Current `TView`.
|
|
|
|
* @param tNode: `TNode` for which we wish to retrieve `RElement` (or `null` if host element is
|
|
|
|
* needed).
|
|
|
|
* @param lView: Current `LView`.
|
|
|
|
* @returns `null` if the `RElement` can't be determined at this time (no parent / projection)
|
|
|
|
*/
|
|
|
|
export function getClosestRElement(tView: TView, tNode: TNode|null, lView: LView): RElement|null {
|
|
|
|
let parentTNode: TNode|null = tNode;
|
2019-01-09 16:33:25 +01:00
|
|
|
// Skip over element and ICU containers as those are represented by a comment node and
|
|
|
|
// can't be used as a render parent.
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
while (parentTNode != null &&
|
2020-09-14 13:06:05 -07:00
|
|
|
(parentTNode.type === TNodeType.ElementContainer ||
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
parentTNode.type === TNodeType.IcuContainer)) {
|
2019-06-07 20:46:11 -07:00
|
|
|
tNode = parentTNode;
|
|
|
|
parentTNode = tNode.parent;
|
|
|
|
}
|
2019-01-09 11:59:49 +01:00
|
|
|
|
2019-06-07 20:46:11 -07:00
|
|
|
// If the parent tNode is null, then we are inserting across views: either into an embedded view
|
|
|
|
// or a component view.
|
2020-09-14 13:06:05 -07:00
|
|
|
if (parentTNode === null) {
|
|
|
|
// We are inserting a root element of the component view into the component host element and
|
|
|
|
// it should always be eager.
|
2020-09-25 15:01:56 -07:00
|
|
|
return lView[HOST];
|
2018-06-22 15:37:38 +02:00
|
|
|
} else {
|
2020-09-25 15:01:56 -07:00
|
|
|
// ngDevMode && assertTNodeType(parentTNode, TNodeType.AnyRNode | TNodeType.Container);
|
2019-09-02 15:17:44 +02:00
|
|
|
if (parentTNode.flags & TNodeFlags.isComponentHost) {
|
2020-09-25 15:01:56 -07:00
|
|
|
ngDevMode && assertTNodeForLView(parentTNode, lView);
|
2020-01-30 14:57:44 -08:00
|
|
|
const tData = tView.data;
|
2019-06-07 20:46:11 -07:00
|
|
|
const tNode = tData[parentTNode.index] as TNode;
|
2019-01-23 20:00:05 +01:00
|
|
|
const encapsulation = (tData[tNode.directiveStart] as ComponentDef<any>).encapsulation;
|
|
|
|
// We've got a parent which is an element in the current view. We just need to verify if the
|
|
|
|
// parent element is not a component. Component's content nodes are not inserted immediately
|
|
|
|
// because they will be projected, and so doing insert at this point would be wasteful.
|
|
|
|
// Since the projection would then move it to its final destination. Note that we can't
|
|
|
|
// make this assumption when using the Shadow DOM, because the native projection placeholders
|
|
|
|
// (<content> or <slot>) have to be in place as elements are being inserted.
|
2020-10-08 16:59:29 +02:00
|
|
|
if (encapsulation === ViewEncapsulation.None ||
|
|
|
|
encapsulation === ViewEncapsulation.Emulated) {
|
2019-01-23 20:00:05 +01:00
|
|
|
return null;
|
|
|
|
}
|
2019-01-09 16:33:25 +01:00
|
|
|
}
|
2019-01-23 20:00:05 +01:00
|
|
|
|
2020-09-25 15:01:56 -07:00
|
|
|
return getNativeByTNode(parentTNode, lView) as RElement;
|
2018-06-22 15:37:38 +02:00
|
|
|
}
|
2018-01-26 11:36:31 +01:00
|
|
|
}
|
|
|
|
|
2018-08-01 15:19:27 +02:00
|
|
|
/**
|
|
|
|
* Inserts a native node before another native node for a given parent using {@link Renderer3}.
|
|
|
|
* This is a utility function that can be used when native nodes were determined - it abstracts an
|
|
|
|
* actual renderer being used.
|
|
|
|
*/
|
2018-10-17 11:01:35 +02:00
|
|
|
export function nativeInsertBefore(
|
2020-09-25 15:01:56 -07:00
|
|
|
renderer: Renderer3, parent: RElement, child: RNode, beforeNode: RNode|null,
|
|
|
|
isMove: boolean): void {
|
2019-05-07 23:56:37 +02:00
|
|
|
ngDevMode && ngDevMode.rendererInsertBefore++;
|
2018-08-01 15:19:27 +02:00
|
|
|
if (isProceduralRenderer(renderer)) {
|
2020-09-25 15:01:56 -07:00
|
|
|
renderer.insertBefore(parent, child, beforeNode, isMove);
|
2018-08-01 15:19:27 +02:00
|
|
|
} else {
|
2020-09-25 15:01:56 -07:00
|
|
|
parent.insertBefore(child, beforeNode, isMove);
|
2018-08-01 15:19:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-01-21 14:55:37 +01:00
|
|
|
function nativeAppendChild(renderer: Renderer3, parent: RElement, child: RNode): void {
|
2019-05-07 23:56:37 +02:00
|
|
|
ngDevMode && ngDevMode.rendererAppendChild++;
|
2019-06-07 20:46:11 -07:00
|
|
|
ngDevMode && assertDefined(parent, 'parent node must be defined');
|
2019-01-21 14:55:37 +01:00
|
|
|
if (isProceduralRenderer(renderer)) {
|
|
|
|
renderer.appendChild(parent, child);
|
|
|
|
} else {
|
|
|
|
parent.appendChild(child);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function nativeAppendOrInsertBefore(
|
2020-09-25 15:01:56 -07:00
|
|
|
renderer: Renderer3, parent: RElement, child: RNode, beforeNode: RNode|null, isMove: boolean) {
|
2019-06-07 20:46:11 -07:00
|
|
|
if (beforeNode !== null) {
|
2020-09-25 15:01:56 -07:00
|
|
|
nativeInsertBefore(renderer, parent, child, beforeNode, isMove);
|
2019-01-21 14:55:37 +01:00
|
|
|
} else {
|
|
|
|
nativeAppendChild(renderer, parent, child);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-04 10:51:45 +01:00
|
|
|
/** Removes a node from the DOM given its native parent. */
|
|
|
|
function nativeRemoveChild(
|
|
|
|
renderer: Renderer3, parent: RElement, child: RNode, isHostElement?: boolean): void {
|
2019-02-01 15:38:42 -08:00
|
|
|
if (isProceduralRenderer(renderer)) {
|
2019-02-04 10:51:45 +01:00
|
|
|
renderer.removeChild(parent, child, isHostElement);
|
2019-02-01 15:38:42 -08:00
|
|
|
} else {
|
2019-02-04 10:51:45 +01:00
|
|
|
parent.removeChild(child);
|
2019-02-01 15:38:42 -08:00
|
|
|
}
|
2019-01-04 12:08:04 +01:00
|
|
|
}
|
|
|
|
|
2018-10-17 11:01:35 +02:00
|
|
|
/**
|
|
|
|
* Returns a native parent of a given native node.
|
|
|
|
*/
|
|
|
|
export function nativeParentNode(renderer: Renderer3, node: RNode): RElement|null {
|
|
|
|
return (isProceduralRenderer(renderer) ? renderer.parentNode(node) : node.parentNode) as RElement;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns a native sibling of a given native node.
|
|
|
|
*/
|
|
|
|
export function nativeNextSibling(renderer: Renderer3, node: RNode): RNode|null {
|
|
|
|
return isProceduralRenderer(renderer) ? renderer.nextSibling(node) : node.nextSibling;
|
|
|
|
}
|
|
|
|
|
2018-01-26 11:36:31 +01:00
|
|
|
/**
|
2020-09-25 15:01:56 -07:00
|
|
|
* Find a node in front of which `currentTNode` should be inserted.
|
|
|
|
*
|
|
|
|
* This method determines the `RNode` in front of which we should insert the `currentRNode`. This
|
|
|
|
* takes `TNode.insertBeforeIndex` into account.
|
|
|
|
*
|
|
|
|
* @param parentTNode parent `TNode`
|
|
|
|
* @param currentTNode current `TNode` (The node which we would like to insert into the DOM)
|
|
|
|
* @param lView current `LView`
|
2019-01-21 14:55:37 +01:00
|
|
|
*/
|
2020-09-25 15:01:56 -07:00
|
|
|
function getInsertInFrontOfRNode(parentTNode: TNode, currentTNode: TNode, lView: LView): RNode|
|
|
|
|
null {
|
|
|
|
const tNodeInsertBeforeIndex = currentTNode.insertBeforeIndex;
|
|
|
|
const insertBeforeIndex =
|
|
|
|
Array.isArray(tNodeInsertBeforeIndex) ? tNodeInsertBeforeIndex[0] : tNodeInsertBeforeIndex;
|
|
|
|
if (insertBeforeIndex === null) {
|
|
|
|
if (parentTNode.type === TNodeType.ElementContainer ||
|
|
|
|
parentTNode.type === TNodeType.IcuContainer) {
|
|
|
|
return getNativeByTNode(parentTNode, lView);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ngDevMode && assertIndexInRange(lView, insertBeforeIndex);
|
|
|
|
return unwrapRNode(lView[insertBeforeIndex]);
|
2019-01-21 14:55:37 +01:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Appends the `child` native node (or a collection of nodes) to the `parent`.
|
2018-01-26 11:36:31 +01:00
|
|
|
*
|
2020-01-30 14:57:44 -08:00
|
|
|
* @param tView The `TView' to be appended
|
|
|
|
* @param lView The current LView
|
2020-09-25 15:01:56 -07:00
|
|
|
* @param childRNode The native child (or children) that should be appended
|
2018-09-13 16:07:23 -07:00
|
|
|
* @param childTNode The TNode of the child element
|
2018-01-26 11:36:31 +01:00
|
|
|
*/
|
2020-01-30 14:57:44 -08:00
|
|
|
export function appendChild(
|
2020-09-25 15:01:56 -07:00
|
|
|
tView: TView, lView: LView, childRNode: RNode|RNode[], childTNode: TNode): void {
|
|
|
|
const parentRNode = getParentRElement(tView, childTNode, lView);
|
|
|
|
const renderer = lView[RENDERER];
|
|
|
|
const parentTNode: TNode = childTNode.parent || lView[T_HOST]!;
|
|
|
|
const anchorNode = getInsertInFrontOfRNode(parentTNode, childTNode, lView);
|
|
|
|
if (parentRNode != null) {
|
|
|
|
if (Array.isArray(childRNode)) {
|
|
|
|
for (let i = 0; i < childRNode.length; i++) {
|
|
|
|
nativeAppendOrInsertBefore(renderer, parentRNode, childRNode[i], anchorNode, false);
|
2019-01-21 14:55:37 +01:00
|
|
|
}
|
2018-06-22 15:37:38 +02:00
|
|
|
} else {
|
2020-09-25 15:01:56 -07:00
|
|
|
nativeAppendOrInsertBefore(renderer, parentRNode, childRNode, anchorNode, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const tNodeInsertBeforeIndex = childTNode.insertBeforeIndex;
|
|
|
|
if (Array.isArray(tNodeInsertBeforeIndex) &&
|
|
|
|
(childTNode.flags & TNodeFlags.isComponentHost) === 0) {
|
|
|
|
// An array indicates that there are i18n nodes that need to be added as children of this
|
|
|
|
// `rChildNode`. These i18n nodes were created before this `rChildNode` was available and so
|
|
|
|
// only now can be added. The first element of the array is the normal index where we should
|
|
|
|
// insert the `rChildNode`. Additional elements are the extra nodes to be added as children of
|
|
|
|
// `rChildNode`.
|
|
|
|
processI18nText(renderer, childTNode, lView, childRNode, parentRNode, tNodeInsertBeforeIndex);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Process `TNode.insertBeforeIndex` by adding i18n text nodes.
|
|
|
|
*
|
|
|
|
* See `TNode.insertBeforeIndex`
|
|
|
|
*
|
|
|
|
* @param renderer
|
|
|
|
* @param childTNode
|
|
|
|
* @param lView
|
|
|
|
* @param childRNode
|
|
|
|
* @param parentRElement
|
|
|
|
* @param i18nChildren
|
|
|
|
*/
|
|
|
|
function processI18nText(
|
|
|
|
renderer: Renderer3, childTNode: TNode, lView: LView, childRNode: RNode|RNode[],
|
|
|
|
parentRElement: RElement|null, i18nChildren: number[]): void {
|
|
|
|
ngDevMode && assertDomNode(childRNode);
|
|
|
|
const isProcedural = isProceduralRenderer(renderer);
|
|
|
|
let i18nParent: RElement|null = childRNode as RElement;
|
|
|
|
let anchorRNode: RNode|null = null;
|
|
|
|
if (childTNode.type !== TNodeType.Element) {
|
|
|
|
anchorRNode = i18nParent;
|
|
|
|
i18nParent = parentRElement;
|
|
|
|
}
|
|
|
|
const isViewRoot = childTNode.parent === null;
|
|
|
|
if (i18nParent !== null) {
|
|
|
|
for (let i = 1; i < i18nChildren.length; i++) {
|
|
|
|
// No need to `unwrapRNode` because all of the indexes point to i18n text nodes.
|
|
|
|
// see `assertDomNode` below.
|
|
|
|
const i18nChild = lView[i18nChildren[i]];
|
|
|
|
nativeInsertBefore(renderer, i18nParent, i18nChild, anchorRNode, false);
|
2018-06-22 15:37:38 +02:00
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-06 15:29:27 +01:00
|
|
|
/**
|
|
|
|
* Returns the first native node for a given LView, starting from the provided TNode.
|
|
|
|
*
|
|
|
|
* Native nodes are returned in the order in which those appear in the native tree (DOM).
|
|
|
|
*/
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
function getFirstNativeNode(lView: LView, tNode: TNode|null): RNode|null {
|
2019-11-06 15:29:27 +01:00
|
|
|
if (tNode !== null) {
|
2020-03-06 11:57:29 -08:00
|
|
|
ngDevMode && assertNodeOfPossibleTypes(tNode, [
|
|
|
|
TNodeType.Element, TNodeType.Container, TNodeType.ElementContainer, TNodeType.IcuContainer,
|
|
|
|
TNodeType.Projection
|
|
|
|
]);
|
2019-11-06 15:29:27 +01:00
|
|
|
|
|
|
|
const tNodeType = tNode.type;
|
|
|
|
if (tNodeType === TNodeType.Element) {
|
|
|
|
return getNativeByTNode(tNode, lView);
|
|
|
|
} else if (tNodeType === TNodeType.Container) {
|
2019-11-07 11:42:58 +01:00
|
|
|
return getBeforeNodeForView(-1, lView[tNode.index]);
|
2020-09-25 15:01:56 -07:00
|
|
|
} else if (tNodeType === TNodeType.ElementContainer) {
|
2019-11-07 10:55:59 +01:00
|
|
|
const elIcuContainerChild = tNode.child;
|
|
|
|
if (elIcuContainerChild !== null) {
|
|
|
|
return getFirstNativeNode(lView, elIcuContainerChild);
|
|
|
|
} else {
|
2019-11-15 12:18:00 +01:00
|
|
|
const rNodeOrLContainer = lView[tNode.index];
|
|
|
|
if (isLContainer(rNodeOrLContainer)) {
|
|
|
|
return getBeforeNodeForView(-1, rNodeOrLContainer);
|
|
|
|
} else {
|
|
|
|
return unwrapRNode(rNodeOrLContainer);
|
|
|
|
}
|
2019-11-07 10:55:59 +01:00
|
|
|
}
|
2020-09-25 15:01:56 -07:00
|
|
|
} else if (tNodeType === TNodeType.IcuContainer) {
|
|
|
|
let nextRNode = icuContainerIterate(tNode as TIcuContainerNode, lView);
|
|
|
|
let rNode: RNode|null = nextRNode();
|
|
|
|
// If the ICU container has no nodes, than we use the ICU anchor as the node.
|
|
|
|
return rNode || unwrapRNode(lView[tNode.index]);
|
2019-11-06 15:29:27 +01:00
|
|
|
} else {
|
2019-11-13 16:22:55 -08:00
|
|
|
const componentView = lView[DECLARATION_COMPONENT_VIEW];
|
2019-11-06 15:29:27 +01:00
|
|
|
const componentHost = componentView[T_HOST] as TElementNode;
|
|
|
|
const parentView = getLViewParent(componentView);
|
|
|
|
const firstProjectedTNode: TNode|null =
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
(componentHost.projection as (TNode | null)[])[tNode.projection as number];
|
2019-11-06 15:29:27 +01:00
|
|
|
|
|
|
|
if (firstProjectedTNode != null) {
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
return getFirstNativeNode(parentView!, firstProjectedTNode);
|
2019-11-06 15:29:27 +01:00
|
|
|
} else {
|
|
|
|
return getFirstNativeNode(lView, tNode.next);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2019-06-14 07:55:17 +02:00
|
|
|
export function getBeforeNodeForView(viewIndexInContainer: number, lContainer: LContainer): RNode|
|
|
|
|
null {
|
2019-05-22 16:41:35 +02:00
|
|
|
const nextViewIndex = CONTAINER_HEADER_OFFSET + viewIndexInContainer + 1;
|
|
|
|
if (nextViewIndex < lContainer.length) {
|
|
|
|
const lView = lContainer[nextViewIndex] as LView;
|
2020-09-14 13:06:05 -07:00
|
|
|
const firstTNodeOfView = lView[TVIEW].firstChild;
|
2019-11-07 11:42:58 +01:00
|
|
|
if (firstTNodeOfView !== null) {
|
|
|
|
return getFirstNativeNode(lView, firstTNodeOfView);
|
|
|
|
}
|
2018-09-13 16:07:23 -07:00
|
|
|
}
|
2019-09-04 20:44:38 +02:00
|
|
|
|
|
|
|
return lContainer[NATIVE];
|
2018-09-13 16:07:23 -07:00
|
|
|
}
|
|
|
|
|
2018-06-18 16:55:43 +02:00
|
|
|
/**
|
2019-02-04 10:51:45 +01:00
|
|
|
* Removes a native node itself using a given renderer. To remove the node we are looking up its
|
|
|
|
* parent from the native tree as not all platforms / browsers support the equivalent of
|
|
|
|
* node.remove().
|
2018-06-18 16:55:43 +02:00
|
|
|
*
|
2019-02-04 10:51:45 +01:00
|
|
|
* @param renderer A renderer to be used
|
|
|
|
* @param rNode The native node that should be removed
|
|
|
|
* @param isHostElement A flag indicating if a node to be removed is a host of a component.
|
2018-06-18 16:55:43 +02:00
|
|
|
*/
|
2019-02-04 10:51:45 +01:00
|
|
|
export function nativeRemoveNode(renderer: Renderer3, rNode: RNode, isHostElement?: boolean): void {
|
2020-09-25 15:01:56 -07:00
|
|
|
ngDevMode && ngDevMode.rendererRemoveNode++;
|
2019-02-04 10:51:45 +01:00
|
|
|
const nativeParent = nativeParentNode(renderer, rNode);
|
|
|
|
if (nativeParent) {
|
|
|
|
nativeRemoveChild(renderer, nativeParent, rNode, isHostElement);
|
|
|
|
}
|
2018-06-18 16:55:43 +02:00
|
|
|
}
|
|
|
|
|
2019-06-11 19:01:49 +02:00
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2019-06-07 20:46:11 -07:00
|
|
|
* Performs the operation of `action` on the node. Typically this involves inserting or removing
|
|
|
|
* nodes on the LView or projection boundary.
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2019-06-07 20:46:11 -07:00
|
|
|
function applyNodes(
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
renderer: Renderer3, action: WalkTNodeTreeAction, tNode: TNode|null, lView: LView,
|
2020-09-25 15:01:56 -07:00
|
|
|
parentRElement: RElement|null, beforeNode: RNode|null, isProjection: boolean) {
|
2019-06-07 20:46:11 -07:00
|
|
|
while (tNode != null) {
|
|
|
|
ngDevMode && assertTNodeForLView(tNode, lView);
|
2020-03-06 11:57:29 -08:00
|
|
|
ngDevMode && assertNodeOfPossibleTypes(tNode, [
|
|
|
|
TNodeType.Container, TNodeType.Element, TNodeType.ElementContainer, TNodeType.Projection,
|
|
|
|
TNodeType.IcuContainer
|
|
|
|
]);
|
2019-06-07 20:46:11 -07:00
|
|
|
const rawSlotValue = lView[tNode.index];
|
|
|
|
const tNodeType = tNode.type;
|
|
|
|
if (isProjection) {
|
|
|
|
if (action === WalkTNodeTreeAction.Create) {
|
|
|
|
rawSlotValue && attachPatchData(unwrapRNode(rawSlotValue), lView);
|
|
|
|
tNode.flags |= TNodeFlags.isProjected;
|
|
|
|
}
|
2018-10-11 13:13:57 -07:00
|
|
|
}
|
2019-06-07 20:46:11 -07:00
|
|
|
if ((tNode.flags & TNodeFlags.isDetached) !== TNodeFlags.isDetached) {
|
2020-09-25 15:01:56 -07:00
|
|
|
if (tNodeType === TNodeType.ElementContainer) {
|
|
|
|
applyNodes(renderer, action, tNode.child, lView, parentRElement, beforeNode, false);
|
|
|
|
applyToElementOrContainer(action, renderer, parentRElement, rawSlotValue, beforeNode);
|
|
|
|
} else if (tNodeType === TNodeType.IcuContainer) {
|
|
|
|
const nextRNode = icuContainerIterate(tNode as TIcuContainerNode, lView);
|
|
|
|
let rNode: RNode|null;
|
|
|
|
while (rNode = nextRNode()) {
|
|
|
|
applyToElementOrContainer(action, renderer, parentRElement, rNode, beforeNode);
|
|
|
|
}
|
|
|
|
applyToElementOrContainer(action, renderer, parentRElement, rawSlotValue, beforeNode);
|
2019-06-07 20:46:11 -07:00
|
|
|
} else if (tNodeType === TNodeType.Projection) {
|
|
|
|
applyProjectionRecursive(
|
2020-09-25 15:01:56 -07:00
|
|
|
renderer, action, lView, tNode as TProjectionNode, parentRElement, beforeNode);
|
2019-06-07 20:46:11 -07:00
|
|
|
} else {
|
2020-03-06 11:57:29 -08:00
|
|
|
ngDevMode && assertNodeOfPossibleTypes(tNode, [TNodeType.Element, TNodeType.Container]);
|
2020-09-25 15:01:56 -07:00
|
|
|
applyToElementOrContainer(action, renderer, parentRElement, rawSlotValue, beforeNode);
|
2019-06-07 20:46:11 -07:00
|
|
|
}
|
2018-08-07 14:42:40 +02:00
|
|
|
}
|
2019-06-07 20:46:11 -07:00
|
|
|
tNode = isProjection ? tNode.projectionNext : tNode.next;
|
2018-04-09 17:35:50 +02:00
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2019-06-07 20:46:11 -07:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2019-06-07 20:46:11 -07:00
|
|
|
* `applyView` performs operation on the view as specified in `action` (insert, detach, destroy)
|
2019-06-07 20:46:11 -07:00
|
|
|
*
|
|
|
|
* Inserting a view without projection or containers at top level is simple. Just iterate over the
|
|
|
|
* root nodes of the View, and for each node perform the `action`.
|
|
|
|
*
|
|
|
|
* Things get more complicated with containers and projections. That is because coming across:
|
|
|
|
* - Container: implies that we have to insert/remove/destroy the views of that container as well
|
|
|
|
* which in turn can have their own Containers at the View roots.
|
|
|
|
* - Projection: implies that we have to insert/remove/destroy the nodes of the projection. The
|
|
|
|
* complication is that the nodes we are projecting can themselves have Containers
|
|
|
|
* or other Projections.
|
|
|
|
*
|
2019-06-07 20:46:11 -07:00
|
|
|
* As you can see this is a very recursive problem. Yes recursion is not most efficient but the
|
|
|
|
* code is complicated enough that trying to implemented with recursion becomes unmaintainable.
|
2019-06-07 20:46:11 -07:00
|
|
|
*
|
2020-01-30 14:57:44 -08:00
|
|
|
* @param tView The `TView' which needs to be inserted, detached, destroyed
|
|
|
|
* @param lView The LView which needs to be inserted, detached, destroyed.
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param renderer Renderer to use
|
|
|
|
* @param action action to perform (insert, detach, destroy)
|
2020-09-25 15:01:56 -07:00
|
|
|
* @param parentRElement parent DOM element for insertion (Removal does not need it).
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param beforeNode Before which node the insertions should happen.
|
|
|
|
*/
|
2020-09-14 11:21:15 -07:00
|
|
|
function applyView(
|
|
|
|
tView: TView, lView: LView, renderer: Renderer3, action: WalkTNodeTreeAction.Destroy,
|
2020-09-25 15:01:56 -07:00
|
|
|
parentRElement: null, beforeNode: null): void;
|
2020-09-14 11:21:15 -07:00
|
|
|
function applyView(
|
|
|
|
tView: TView, lView: LView, renderer: Renderer3, action: WalkTNodeTreeAction,
|
2020-09-25 15:01:56 -07:00
|
|
|
parentRElement: RElement|null, beforeNode: RNode|null): void;
|
2019-06-07 20:46:11 -07:00
|
|
|
function applyView(
|
2020-01-30 14:57:44 -08:00
|
|
|
tView: TView, lView: LView, renderer: Renderer3, action: WalkTNodeTreeAction,
|
2020-09-25 15:01:56 -07:00
|
|
|
parentRElement: RElement|null, beforeNode: RNode|null): void {
|
|
|
|
applyNodes(renderer, action, tView.firstChild, lView, parentRElement, beforeNode, false);
|
2019-06-07 20:46:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-06-07 20:46:11 -07:00
|
|
|
* `applyProjection` performs operation on the projection.
|
2019-06-07 20:46:11 -07:00
|
|
|
*
|
|
|
|
* Inserting a projection requires us to locate the projected nodes from the parent component. The
|
2019-06-24 17:27:12 +02:00
|
|
|
* complication is that those nodes themselves could be re-projected from their parent component.
|
2019-06-07 20:46:11 -07:00
|
|
|
*
|
2020-01-30 14:57:44 -08:00
|
|
|
* @param tView The `TView` of `LView` which needs to be inserted, detached, destroyed
|
|
|
|
* @param lView The `LView` which needs to be inserted, detached, destroyed.
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param tProjectionNode node to project
|
|
|
|
*/
|
2020-01-30 14:57:44 -08:00
|
|
|
export function applyProjection(tView: TView, lView: LView, tProjectionNode: TProjectionNode) {
|
2019-06-07 20:46:11 -07:00
|
|
|
const renderer = lView[RENDERER];
|
2020-09-25 15:01:56 -07:00
|
|
|
const parentRNode = getParentRElement(tView, tProjectionNode, lView);
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
const parentTNode = tProjectionNode.parent || lView[T_HOST]!;
|
2020-09-25 15:01:56 -07:00
|
|
|
let beforeNode = getInsertInFrontOfRNode(parentTNode, tProjectionNode, lView);
|
2019-06-07 20:46:11 -07:00
|
|
|
applyProjectionRecursive(
|
2020-09-25 15:01:56 -07:00
|
|
|
renderer, WalkTNodeTreeAction.Create, lView, tProjectionNode, parentRNode, beforeNode);
|
2019-06-07 20:46:11 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* `applyProjectionRecursive` performs operation on the projection specified by `action` (insert,
|
|
|
|
* detach, destroy)
|
|
|
|
*
|
|
|
|
* Inserting a projection requires us to locate the projected nodes from the parent component. The
|
|
|
|
* complication is that those nodes themselves could be re-projected from their parent component.
|
|
|
|
*
|
|
|
|
* @param renderer Render to use
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param action action to perform (insert, detach, destroy)
|
|
|
|
* @param lView The LView which needs to be inserted, detached, destroyed.
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param tProjectionNode node to project
|
2020-09-25 15:01:56 -07:00
|
|
|
* @param parentRElement parent DOM element for insertion/removal.
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param beforeNode Before which node the insertions should happen.
|
|
|
|
*/
|
2019-06-07 20:46:11 -07:00
|
|
|
function applyProjectionRecursive(
|
2019-06-07 20:46:11 -07:00
|
|
|
renderer: Renderer3, action: WalkTNodeTreeAction, lView: LView,
|
2020-09-25 15:01:56 -07:00
|
|
|
tProjectionNode: TProjectionNode, parentRElement: RElement|null, beforeNode: RNode|null) {
|
2019-11-13 16:22:55 -08:00
|
|
|
const componentLView = lView[DECLARATION_COMPONENT_VIEW];
|
2019-06-07 20:46:11 -07:00
|
|
|
const componentNode = componentLView[T_HOST] as TElementNode;
|
2019-06-07 20:46:11 -07:00
|
|
|
ngDevMode &&
|
|
|
|
assertEqual(typeof tProjectionNode.projection, 'number', 'expecting projection index');
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
const nodeToProjectOrRNodes = componentNode.projection![tProjectionNode.projection]!;
|
2019-06-07 20:46:11 -07:00
|
|
|
if (Array.isArray(nodeToProjectOrRNodes)) {
|
|
|
|
// This should not exist, it is a bit of a hack. When we bootstrap a top level node and we
|
|
|
|
// need to support passing projectable nodes, so we cheat and put them in the TNode
|
|
|
|
// of the Host TView. (Yes we put instance info at the T Level). We can get away with it
|
|
|
|
// because we know that that TView is not shared and therefore it will not be a problem.
|
|
|
|
// This should be refactored and cleaned up.
|
|
|
|
for (let i = 0; i < nodeToProjectOrRNodes.length; i++) {
|
|
|
|
const rNode = nodeToProjectOrRNodes[i];
|
2020-09-25 15:01:56 -07:00
|
|
|
applyToElementOrContainer(action, renderer, parentRElement, rNode, beforeNode);
|
2019-06-07 20:46:11 -07:00
|
|
|
}
|
2019-06-07 20:46:11 -07:00
|
|
|
} else {
|
|
|
|
let nodeToProject: TNode|null = nodeToProjectOrRNodes;
|
|
|
|
const projectedComponentLView = componentLView[PARENT] as LView;
|
|
|
|
applyNodes(
|
2020-09-25 15:01:56 -07:00
|
|
|
renderer, action, nodeToProject, projectedComponentLView, parentRElement, beforeNode, true);
|
2019-06-07 20:46:11 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
2019-06-07 20:46:11 -07:00
|
|
|
* `applyContainer` performs an operation on the container and its views as specified by
|
2019-06-24 17:27:12 +02:00
|
|
|
* `action` (insert, detach, destroy)
|
2019-06-07 20:46:11 -07:00
|
|
|
*
|
|
|
|
* Inserting a Container is complicated by the fact that the container may have Views which
|
|
|
|
* themselves have containers or projections.
|
|
|
|
*
|
|
|
|
* @param renderer Renderer to use
|
|
|
|
* @param action action to perform (insert, detach, destroy)
|
|
|
|
* @param lContainer The LContainer which needs to be inserted, detached, destroyed.
|
2020-09-25 15:01:56 -07:00
|
|
|
* @param parentRElement parent DOM element for insertion/removal.
|
2019-06-07 20:46:11 -07:00
|
|
|
* @param beforeNode Before which node the insertions should happen.
|
|
|
|
*/
|
2019-06-07 20:46:11 -07:00
|
|
|
function applyContainer(
|
2019-06-07 20:46:11 -07:00
|
|
|
renderer: Renderer3, action: WalkTNodeTreeAction, lContainer: LContainer,
|
2020-09-25 15:01:56 -07:00
|
|
|
parentRElement: RElement|null, beforeNode: RNode|null|undefined) {
|
2019-06-07 20:46:11 -07:00
|
|
|
ngDevMode && assertLContainer(lContainer);
|
|
|
|
const anchor = lContainer[NATIVE]; // LContainer has its own before node.
|
|
|
|
const native = unwrapRNode(lContainer);
|
2019-06-24 17:27:12 +02:00
|
|
|
// An LContainer can be created dynamically on any node by injecting ViewContainerRef.
|
2020-09-25 15:01:56 -07:00
|
|
|
// Asking for a ViewContainerRef on an element will result in a creation of a separate anchor
|
|
|
|
// node (comment in the DOM) that will be different from the LContainer's host node. In this
|
|
|
|
// particular case we need to execute action on 2 nodes:
|
2019-06-07 20:46:11 -07:00
|
|
|
// - container's host node (this is done in the executeActionOnElementOrContainer)
|
2019-06-07 20:46:11 -07:00
|
|
|
// - container's host node (this is done here)
|
|
|
|
if (anchor !== native) {
|
2020-09-25 15:01:56 -07:00
|
|
|
// This is very strange to me (Misko). I would expect that the native is same as anchor. I
|
|
|
|
// don't see a reason why they should be different, but they are.
|
2019-06-07 20:46:11 -07:00
|
|
|
//
|
|
|
|
// If they are we need to process the second anchor as well.
|
2020-09-25 15:01:56 -07:00
|
|
|
applyToElementOrContainer(action, renderer, parentRElement, anchor, beforeNode);
|
2019-06-07 20:46:11 -07:00
|
|
|
}
|
|
|
|
for (let i = CONTAINER_HEADER_OFFSET; i < lContainer.length; i++) {
|
|
|
|
const lView = lContainer[i] as LView;
|
2020-09-25 15:01:56 -07:00
|
|
|
applyView(lView[TVIEW], lView, renderer, action, parentRElement, anchor);
|
2019-06-07 20:46:11 -07:00
|
|
|
}
|
2019-06-24 17:27:12 +02:00
|
|
|
}
|
2020-01-15 16:52:54 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Writes class/style to element.
|
|
|
|
*
|
|
|
|
* @param renderer Renderer to use.
|
|
|
|
* @param isClassBased `true` if it should be written to `class` (`false` to write to `style`)
|
|
|
|
* @param rNode The Node to write to.
|
|
|
|
* @param prop Property to write to. This would be the class/style name.
|
2020-01-28 16:26:56 -08:00
|
|
|
* @param value Value to write. If `null`/`undefined`/`false` this is considered a remove (set/add
|
|
|
|
* otherwise).
|
2020-01-15 16:52:54 -08:00
|
|
|
*/
|
|
|
|
export function applyStyling(
|
|
|
|
renderer: Renderer3, isClassBased: boolean, rNode: RElement, prop: string, value: any) {
|
|
|
|
const isProcedural = isProceduralRenderer(renderer);
|
|
|
|
if (isClassBased) {
|
2020-01-28 16:26:56 -08:00
|
|
|
// We actually want JS true/false here because any truthy value should add the class
|
|
|
|
if (!value) {
|
2020-01-15 16:52:54 -08:00
|
|
|
ngDevMode && ngDevMode.rendererRemoveClass++;
|
|
|
|
if (isProcedural) {
|
|
|
|
(renderer as Renderer2).removeClass(rNode, prop);
|
|
|
|
} else {
|
|
|
|
(rNode as HTMLElement).classList.remove(prop);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ngDevMode && ngDevMode.rendererAddClass++;
|
|
|
|
if (isProcedural) {
|
|
|
|
(renderer as Renderer2).addClass(rNode, prop);
|
|
|
|
} else {
|
|
|
|
ngDevMode && assertDefined((rNode as HTMLElement).classList, 'HTMLElement expected');
|
|
|
|
(rNode as HTMLElement).classList.add(prop);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2020-09-25 15:01:56 -07:00
|
|
|
// TODO(misko): Can't import RendererStyleFlags2.DashCase as it causes imports to be resolved
|
|
|
|
// in different order which causes failures. Using direct constant as workaround for now.
|
2020-01-15 16:52:54 -08:00
|
|
|
const flags = prop.indexOf('-') == -1 ? undefined : 2 /* RendererStyleFlags2.DashCase */;
|
2020-01-28 16:26:56 -08:00
|
|
|
if (value == null /** || value === undefined */) {
|
2020-01-15 16:52:54 -08:00
|
|
|
ngDevMode && ngDevMode.rendererRemoveStyle++;
|
|
|
|
if (isProcedural) {
|
|
|
|
(renderer as Renderer2).removeStyle(rNode, prop, flags);
|
|
|
|
} else {
|
|
|
|
(rNode as HTMLElement).style.removeProperty(prop);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
ngDevMode && ngDevMode.rendererSetStyle++;
|
|
|
|
if (isProcedural) {
|
|
|
|
(renderer as Renderer2).setStyle(rNode, prop, value, flags);
|
|
|
|
} else {
|
|
|
|
ngDevMode && assertDefined((rNode as HTMLElement).style, 'HTMLElement expected');
|
|
|
|
(rNode as HTMLElement).style.setProperty(prop, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Write `cssText` to `RElement`.
|
|
|
|
*
|
|
|
|
* This function does direct write without any reconciliation. Used for writing initial values, so
|
|
|
|
* that static styling values do not pull in the style parser.
|
|
|
|
*
|
|
|
|
* @param renderer Renderer to use
|
|
|
|
* @param element The element which needs to be updated.
|
|
|
|
* @param newValue The new class list to write.
|
|
|
|
*/
|
|
|
|
export function writeDirectStyle(renderer: Renderer3, element: RElement, newValue: string) {
|
|
|
|
ngDevMode && assertString(newValue, '\'newValue\' should be a string');
|
|
|
|
if (isProceduralRenderer(renderer)) {
|
|
|
|
renderer.setAttribute(element, 'style', newValue);
|
|
|
|
} else {
|
|
|
|
(element as HTMLElement).style.cssText = newValue;
|
|
|
|
}
|
|
|
|
ngDevMode && ngDevMode.rendererSetStyle++;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Write `className` to `RElement`.
|
|
|
|
*
|
|
|
|
* This function does direct write without any reconciliation. Used for writing initial values, so
|
|
|
|
* that static styling values do not pull in the style parser.
|
|
|
|
*
|
|
|
|
* @param renderer Renderer to use
|
|
|
|
* @param element The element which needs to be updated.
|
|
|
|
* @param newValue The new class list to write.
|
|
|
|
*/
|
|
|
|
export function writeDirectClass(renderer: Renderer3, element: RElement, newValue: string) {
|
|
|
|
ngDevMode && assertString(newValue, '\'newValue\' should be a string');
|
|
|
|
if (isProceduralRenderer(renderer)) {
|
|
|
|
if (newValue === '') {
|
|
|
|
// There are tests in `google3` which expect `element.getAttribute('class')` to be `null`.
|
|
|
|
renderer.removeAttribute(element, 'class');
|
|
|
|
} else {
|
|
|
|
renderer.setAttribute(element, 'class', newValue);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
element.className = newValue;
|
|
|
|
}
|
|
|
|
ngDevMode && ngDevMode.rendererSetClassName++;
|
fix(core): ngOnDestroy on multi providers called with incorrect context (#35840)
Currently destroy hooks are stored in memory as `[1, hook, 5, hook]` where
the numbers represent the index at which to find the context and `hook` is
the function to be invoked. This breaks down for `multi` providers,
because the value at the index will be an array of providers, resulting in
the hook being invoked with an array of all the multi provider values,
rather than the provider that was destroyed. In ViewEngine `ngOnDestroy`
wasn't being called for `multi` providers at all.
These changes fix the issue by changing the structure of the destroy hooks to `[1, hook, 5, [0, hook, 3, hook]]` where the indexes inside the inner array point to the provider inside of the multi provider array. Note that this is slightly different from the original design which called for the structure to be `[1, hook, 5, [hook, hook]`, because in the process of implementing it, I realized that we wouldn't get passing the correct context if only some of the `multi` providers have `ngOnDestroy` and others don't.
I've run the newly-added `view_destroy_hooks` benchmark against these changes and compared it to master. The difference seems to be insignificant (between 1% and 2% slower).
Fixes #35231.
PR Close #35840
2020-03-03 21:05:26 +01:00
|
|
|
}
|