2017-12-01 14:23:03 -08:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* 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
|
|
|
|
*/
|
|
|
|
|
2018-01-25 20:41:57 -08:00
|
|
|
import {assertNotNull} from './assert';
|
|
|
|
import {callHooks} from './hooks';
|
2018-01-11 13:09:21 -08:00
|
|
|
import {LContainer, unusedValueExportToPlacateAjd as unused1} from './interfaces/container';
|
2018-05-17 12:54:57 -07:00
|
|
|
import {LContainerNode, LElementNode, LNode, LProjectionNode, LTextNode, LViewNode, TNodeType, unusedValueExportToPlacateAjd as unused2} from './interfaces/node';
|
2018-01-25 15:32:21 +01:00
|
|
|
import {unusedValueExportToPlacateAjd as unused3} from './interfaces/projection';
|
2018-04-10 20:57:09 -07:00
|
|
|
import {ProceduralRenderer3, RElement, RNode, RText, Renderer3, isProceduralRenderer, unusedValueExportToPlacateAjd as unused4} from './interfaces/renderer';
|
2018-01-23 10:57:48 -08:00
|
|
|
import {HookData, LView, LViewOrLContainer, TView, unusedValueExportToPlacateAjd as unused5} from './interfaces/view';
|
2017-12-01 14:23:03 -08:00
|
|
|
import {assertNodeType} from './node_assert';
|
2018-04-10 20:57:09 -07:00
|
|
|
import {stringify} from './util';
|
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
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2018-02-02 14:59:31 -08:00
|
|
|
* Returns the first RNode following the given LNode in the same parent DOM element.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-01-25 15:32:21 +01:00
|
|
|
* This is needed in order to insert the given node with insertBefore.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-01-25 15:32:21 +01:00
|
|
|
* @param node The node whose following DOM node must be found.
|
|
|
|
* @param stopNode A parent node at which the lookup in the tree should be stopped, or null if the
|
|
|
|
* lookup should not be stopped until the result is found.
|
2018-02-02 14:59:31 -08:00
|
|
|
* @returns RNode before which the provided node should be inserted or null if the lookup was
|
|
|
|
* stopped
|
2018-01-25 15:32:21 +01:00
|
|
|
* or if there is no native node after the given logical node in the same native parent.
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-02-02 14:59:31 -08:00
|
|
|
function findNextRNodeSibling(node: LNode | null, stopNode: LNode | null): RElement|RText|null {
|
2018-01-25 15:32:21 +01:00
|
|
|
let currentNode = node;
|
|
|
|
while (currentNode && currentNode !== stopNode) {
|
|
|
|
let pNextOrParent = currentNode.pNextOrParent;
|
|
|
|
if (pNextOrParent) {
|
2018-05-17 12:54:57 -07:00
|
|
|
while (pNextOrParent.tNode.type !== TNodeType.Projection) {
|
2018-02-02 14:59:31 -08:00
|
|
|
const nativeNode = findFirstRNode(pNextOrParent);
|
2018-01-25 15:32:21 +01:00
|
|
|
if (nativeNode) {
|
|
|
|
return nativeNode;
|
|
|
|
}
|
|
|
|
pNextOrParent = pNextOrParent.pNextOrParent !;
|
|
|
|
}
|
|
|
|
currentNode = pNextOrParent;
|
|
|
|
} else {
|
2018-05-11 20:57:37 -07:00
|
|
|
let currentSibling = getNextLNode(currentNode);
|
2018-01-25 15:32:21 +01:00
|
|
|
while (currentSibling) {
|
2018-02-02 14:59:31 -08:00
|
|
|
const nativeNode = findFirstRNode(currentSibling);
|
2018-01-25 15:32:21 +01:00
|
|
|
if (nativeNode) {
|
|
|
|
return nativeNode;
|
|
|
|
}
|
2018-05-11 20:57:37 -07:00
|
|
|
currentSibling = getNextLNode(currentSibling);
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
2018-05-29 15:08:30 -07:00
|
|
|
const parentNode = getParentLNode(currentNode);
|
2018-01-25 15:32:21 +01:00
|
|
|
currentNode = null;
|
|
|
|
if (parentNode) {
|
2018-05-17 12:54:57 -07:00
|
|
|
const parentType = parentNode.tNode.type;
|
|
|
|
if (parentType === TNodeType.Container || parentType === TNodeType.View) {
|
2018-01-25 15:32:21 +01:00
|
|
|
currentNode = parentNode;
|
|
|
|
}
|
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-05-11 20:57:37 -07:00
|
|
|
/** Retrieves the sibling node for the given node. */
|
|
|
|
export function getNextLNode(node: LNode): LNode|null {
|
|
|
|
// View nodes don't have TNodes, so their next must be retrieved through their LView.
|
2018-05-17 12:54:57 -07:00
|
|
|
if (node.tNode.type === TNodeType.View) {
|
2018-05-11 20:57:37 -07:00
|
|
|
const lView = node.data as LView;
|
|
|
|
return lView.next ? (lView.next as LView).node : null;
|
|
|
|
}
|
2018-05-16 05:56:01 -07:00
|
|
|
return node.tNode.next ? node.view.data[node.tNode.next !.index as number] : null;
|
2018-05-11 20:57:37 -07:00
|
|
|
}
|
|
|
|
|
2018-05-24 13:13:51 -07:00
|
|
|
/** Retrieves the first child of a given node */
|
|
|
|
export function getChildLNode(node: LNode): LNode|null {
|
|
|
|
if (node.tNode.child) {
|
|
|
|
const view = node.tNode.type === TNodeType.View ? node.data as LView : node.view;
|
|
|
|
return view.data[node.tNode.child.index as number];
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2018-05-29 15:08:30 -07:00
|
|
|
/** Retrieves the parent LNode of a given node. */
|
|
|
|
export function getParentLNode(node: LElementNode | LTextNode | LProjectionNode): LElementNode|
|
|
|
|
LViewNode;
|
|
|
|
export function getParentLNode(node: LViewNode): LContainerNode|null;
|
|
|
|
export function getParentLNode(node: LNode): LElementNode|LContainerNode|LViewNode|null;
|
|
|
|
export function getParentLNode(node: LNode): LElementNode|LContainerNode|LViewNode|null {
|
|
|
|
if (node.tNode.index === null) return null;
|
|
|
|
const parent = node.tNode.parent;
|
|
|
|
return parent ? node.view.data[parent.index as number] : node.view.node;
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2018-01-25 15:32:21 +01:00
|
|
|
* Get the next node in the LNode tree, taking into account the place where a node is
|
|
|
|
* projected (in the shadow DOM) rather than where it comes from (in the light DOM).
|
2018-02-02 14:59:31 -08:00
|
|
|
*
|
2018-01-25 15:32:21 +01:00
|
|
|
* @param node The node whose next node in the LNode tree must be found.
|
2018-02-02 14:59:31 -08:00
|
|
|
* @return LNode|null The next sibling in the LNode tree.
|
2018-01-25 15:32:21 +01:00
|
|
|
*/
|
2018-02-02 14:59:31 -08:00
|
|
|
function getNextLNodeWithProjection(node: LNode): LNode|null {
|
2018-01-25 15:32:21 +01:00
|
|
|
const pNextOrParent = node.pNextOrParent;
|
2018-02-02 14:59:31 -08:00
|
|
|
|
2018-01-25 15:32:21 +01:00
|
|
|
if (pNextOrParent) {
|
2018-02-02 14:59:31 -08:00
|
|
|
// The node is projected
|
2018-05-17 12:54:57 -07:00
|
|
|
const isLastProjectedNode = pNextOrParent.tNode.type === TNodeType.Projection;
|
2018-02-02 14:59:31 -08:00
|
|
|
// returns pNextOrParent if we are not at the end of the list, null otherwise
|
|
|
|
return isLastProjectedNode ? null : pNextOrParent;
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
2018-02-02 14:59:31 -08:00
|
|
|
|
|
|
|
// returns node.next because the the node is not projected
|
2018-05-11 20:57:37 -07:00
|
|
|
return getNextLNode(node);
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Find the next node in the LNode tree, taking into account the place where a node is
|
|
|
|
* projected (in the shadow DOM) rather than where it comes from (in the light DOM).
|
2018-02-02 14:59:31 -08:00
|
|
|
*
|
2018-01-25 15:32:21 +01:00
|
|
|
* If there is no sibling node, this function goes to the next sibling of the parent node...
|
|
|
|
* until it reaches rootNode (at which point null is returned).
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-01-25 15:32:21 +01:00
|
|
|
* @param initialNode The node whose following node in the LNode tree must be found.
|
|
|
|
* @param rootNode The root node at which the lookup should stop.
|
2018-02-02 14:59:31 -08:00
|
|
|
* @return LNode|null The following node in the LNode tree.
|
2018-01-25 15:32:21 +01:00
|
|
|
*/
|
2018-01-26 11:36:31 +01:00
|
|
|
function getNextOrParentSiblingNode(initialNode: LNode, rootNode: LNode): LNode|null {
|
2018-01-25 15:32:21 +01:00
|
|
|
let node: LNode|null = initialNode;
|
2018-02-02 14:59:31 -08:00
|
|
|
let nextNode = getNextLNodeWithProjection(node);
|
2018-01-25 15:32:21 +01:00
|
|
|
while (node && !nextNode) {
|
|
|
|
// if node.pNextOrParent is not null here, it is not the next node
|
|
|
|
// (because, at this point, nextNode is null, so it is the parent)
|
2018-05-29 15:08:30 -07:00
|
|
|
node = node.pNextOrParent || getParentLNode(node);
|
2018-02-02 14:59:31 -08:00
|
|
|
if (node === rootNode) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
nextNode = node && getNextLNodeWithProjection(node);
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
|
|
|
return nextNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-02-02 14:59:31 -08:00
|
|
|
* Returns the first RNode inside the given LNode.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-01-25 15:32:21 +01:00
|
|
|
* @param node The node whose first DOM node must be found
|
2018-02-02 14:59:31 -08:00
|
|
|
* @returns RNode The first RNode of the given LNode or null if there is none.
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-02-02 14:59:31 -08:00
|
|
|
function findFirstRNode(rootNode: LNode): RElement|RText|null {
|
2018-01-25 15:32:21 +01:00
|
|
|
let node: LNode|null = rootNode;
|
|
|
|
while (node) {
|
|
|
|
let nextNode: LNode|null = null;
|
2018-05-17 12:54:57 -07:00
|
|
|
if (node.tNode.type === TNodeType.Element) {
|
2018-02-02 14:59:31 -08:00
|
|
|
// A LElementNode has a matching RNode in LElementNode.native
|
2018-01-26 11:36:31 +01:00
|
|
|
return (node as LElementNode).native;
|
2018-05-17 12:54:57 -07:00
|
|
|
} else if (node.tNode.type === TNodeType.Container) {
|
2018-04-03 10:18:25 +02:00
|
|
|
const lContainerNode: LContainerNode = (node as LContainerNode);
|
|
|
|
const childContainerData: LContainer = lContainerNode.dynamicLContainerNode ?
|
|
|
|
lContainerNode.dynamicLContainerNode.data :
|
|
|
|
lContainerNode.data;
|
2018-05-24 13:13:51 -07:00
|
|
|
nextNode =
|
|
|
|
childContainerData.views.length ? getChildLNode(childContainerData.views[0]) : null;
|
2018-05-17 12:54:57 -07:00
|
|
|
} else if (node.tNode.type === TNodeType.Projection) {
|
2018-02-02 14:59:31 -08:00
|
|
|
// For Projection look at the first projected node
|
2018-01-26 11:36:31 +01:00
|
|
|
nextNode = (node as LProjectionNode).data.head;
|
2018-01-25 15:32:21 +01:00
|
|
|
} else {
|
2018-02-02 14:59:31 -08:00
|
|
|
// Otherwise look at the first child
|
2018-05-24 13:13:51 -07:00
|
|
|
nextNode = getChildLNode(node as LViewNode);
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
2018-02-02 14:59:31 -08:00
|
|
|
|
|
|
|
node = nextNode === null ? getNextOrParentSiblingNode(node, rootNode) : nextNode;
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
|
|
|
return null;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-04-10 20:57:09 -07:00
|
|
|
export function createTextNode(value: any, renderer: Renderer3): RText {
|
|
|
|
return isProceduralRenderer(renderer) ? renderer.createText(stringify(value)) :
|
|
|
|
renderer.createTextNode(stringify(value));
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Adds or removes 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 remove all elements in the
|
|
|
|
* views beneath it.
|
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param container The container to which the root view belongs
|
|
|
|
* @param rootNode The view from which elements should be added or removed
|
|
|
|
* @param insertMode Whether or not elements should be added (if false, removing)
|
|
|
|
* @param beforeNode The node before which elements should be added, if insert mode
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2017-12-01 14:23:03 -08:00
|
|
|
export function addRemoveViewFromContainer(
|
2018-01-08 20:17:13 -08:00
|
|
|
container: LContainerNode, rootNode: LViewNode, insertMode: true,
|
|
|
|
beforeNode: RNode | null): void;
|
2017-12-01 14:23:03 -08:00
|
|
|
export function addRemoveViewFromContainer(
|
2018-01-08 20:17:13 -08:00
|
|
|
container: LContainerNode, rootNode: LViewNode, insertMode: false): void;
|
2017-12-01 14:23:03 -08:00
|
|
|
export function addRemoveViewFromContainer(
|
2018-01-08 20:17:13 -08:00
|
|
|
container: LContainerNode, rootNode: LViewNode, insertMode: boolean,
|
|
|
|
beforeNode?: RNode | null): void {
|
2018-05-17 12:54:57 -07:00
|
|
|
ngDevMode && assertNodeType(container, TNodeType.Container);
|
|
|
|
ngDevMode && assertNodeType(rootNode, TNodeType.View);
|
2018-01-25 15:32:21 +01:00
|
|
|
const parentNode = container.data.renderParent;
|
|
|
|
const parent = parentNode ? parentNode.native : null;
|
2018-05-24 13:13:51 -07:00
|
|
|
let node: LNode|null = getChildLNode(rootNode);
|
2017-12-01 14:23:03 -08:00
|
|
|
if (parent) {
|
|
|
|
while (node) {
|
|
|
|
let nextNode: LNode|null = null;
|
|
|
|
const renderer = container.view.renderer;
|
2018-05-17 12:54:57 -07:00
|
|
|
if (node.tNode.type === TNodeType.Element) {
|
2018-02-06 20:08:46 -08:00
|
|
|
if (insertMode) {
|
|
|
|
isProceduralRenderer(renderer) ?
|
|
|
|
renderer.insertBefore(parent, node.native !, beforeNode as RNode | null) :
|
|
|
|
parent.insertBefore(node.native !, beforeNode as RNode | null, true);
|
|
|
|
} else {
|
|
|
|
isProceduralRenderer(renderer) ? renderer.removeChild(parent as RElement, node.native !) :
|
|
|
|
parent.removeChild(node.native !);
|
|
|
|
}
|
2018-05-11 20:57:37 -07:00
|
|
|
nextNode = getNextLNode(node);
|
2018-05-17 12:54:57 -07:00
|
|
|
} else if (node.tNode.type === TNodeType.Container) {
|
2017-12-01 14:23:03 -08:00
|
|
|
// if we get to a container, it must be a root node of a view because we are only
|
|
|
|
// propagating down into child views / containers and not child elements
|
2018-01-08 20:17:13 -08:00
|
|
|
const childContainerData: LContainer = (node as LContainerNode).data;
|
2018-01-25 15:32:21 +01:00
|
|
|
childContainerData.renderParent = parentNode;
|
2018-05-24 13:13:51 -07:00
|
|
|
nextNode =
|
|
|
|
childContainerData.views.length ? getChildLNode(childContainerData.views[0]) : null;
|
2018-05-17 12:54:57 -07:00
|
|
|
} else if (node.tNode.type === TNodeType.Projection) {
|
2018-01-26 11:36:31 +01:00
|
|
|
nextNode = (node as LProjectionNode).data.head;
|
2017-12-01 14:23:03 -08:00
|
|
|
} else {
|
2018-05-24 13:13:51 -07:00
|
|
|
nextNode = getChildLNode(node as LViewNode);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
if (nextNode === null) {
|
2018-01-26 11:36:31 +01:00
|
|
|
node = getNextOrParentSiblingNode(node, rootNode);
|
2017-12-01 14:23:03 -08:00
|
|
|
} else {
|
|
|
|
node = nextNode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
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-01-08 20:17:13 -08:00
|
|
|
export function destroyViewTree(rootView: LView): void {
|
2018-04-19 22:59:58 +02:00
|
|
|
// A view to cleanup doesn't have children so we should not try to descend down the view tree.
|
|
|
|
if (!rootView.child) {
|
|
|
|
return cleanUpView(rootView);
|
|
|
|
}
|
|
|
|
let viewOrContainer: LViewOrLContainer|null = rootView.child;
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
while (viewOrContainer) {
|
|
|
|
let next: LViewOrLContainer|null = null;
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
if (viewOrContainer.views && viewOrContainer.views.length) {
|
|
|
|
next = viewOrContainer.views[0].data;
|
|
|
|
} else if (viewOrContainer.child) {
|
|
|
|
next = viewOrContainer.child;
|
|
|
|
} else if (viewOrContainer.next) {
|
2018-04-19 22:59:58 +02:00
|
|
|
// Only move to the side and clean if operating below rootView -
|
|
|
|
// otherwise we would start cleaning up sibling views of the rootView.
|
2018-01-08 20:17:13 -08:00
|
|
|
cleanUpView(viewOrContainer as LView);
|
|
|
|
next = viewOrContainer.next;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (next == null) {
|
2018-04-19 22:59:58 +02:00
|
|
|
// If the viewOrContainer is the rootView and next is null it means that we are dealing
|
|
|
|
// with a root view that doesn't have children. We didn't descend into child views
|
|
|
|
// so no need to go back up the views tree.
|
2018-02-21 15:08:18 +01:00
|
|
|
while (viewOrContainer && !viewOrContainer !.next && viewOrContainer !== rootView) {
|
2018-01-08 20:17:13 -08:00
|
|
|
cleanUpView(viewOrContainer as LView);
|
|
|
|
viewOrContainer = getParentState(viewOrContainer, rootView);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2018-01-08 20:17:13 -08:00
|
|
|
cleanUpView(viewOrContainer as LView || rootView);
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
next = viewOrContainer && viewOrContainer.next;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2018-01-08 20:17:13 -08:00
|
|
|
viewOrContainer = 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).
|
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param container The container into which the view should be inserted
|
2018-05-11 20:57:37 -07:00
|
|
|
* @param viewNode The view to insert
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param index The index at which to insert the view
|
|
|
|
* @returns The inserted view
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function insertView(
|
2018-05-11 20:57:37 -07:00
|
|
|
container: LContainerNode, viewNode: LViewNode, index: number): LViewNode {
|
2017-12-01 14:23:03 -08:00
|
|
|
const state = container.data;
|
2017-12-13 19:34:46 -08:00
|
|
|
const views = state.views;
|
2017-12-01 14:23:03 -08:00
|
|
|
|
|
|
|
if (index > 0) {
|
|
|
|
// This is a new view, we need to add it to the children.
|
2018-05-11 20:57:37 -07:00
|
|
|
views[index - 1].data.next = viewNode.data as LView;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-03-08 12:10:20 +01:00
|
|
|
if (index < views.length) {
|
2018-05-11 20:57:37 -07:00
|
|
|
viewNode.data.next = views[index].data;
|
|
|
|
views.splice(index, 0, viewNode);
|
2018-03-08 12:10:20 +01:00
|
|
|
} else {
|
2018-05-11 20:57:37 -07:00
|
|
|
views.push(viewNode);
|
|
|
|
viewNode.data.next = null;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// If the container's renderParent is null, we know that it is a root node of its own parent view
|
|
|
|
// and we should wait until that parent processes its nodes (otherwise, we will insert this view's
|
|
|
|
// nodes twice - once now and once when its parent inserts its views).
|
|
|
|
if (container.data.renderParent !== null) {
|
2018-05-11 20:57:37 -07:00
|
|
|
let beforeNode = findNextRNodeSibling(viewNode, container);
|
|
|
|
|
2018-01-25 15:32:21 +01:00
|
|
|
if (!beforeNode) {
|
2018-01-26 11:36:31 +01:00
|
|
|
let containerNextNativeNode = container.native;
|
2018-01-25 15:32:21 +01:00
|
|
|
if (containerNextNativeNode === undefined) {
|
2018-04-03 10:18:25 +02:00
|
|
|
containerNextNativeNode = container.native = findNextRNodeSibling(container, null);
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
|
|
|
beforeNode = containerNextNativeNode;
|
|
|
|
}
|
2018-05-11 20:57:37 -07:00
|
|
|
addRemoveViewFromContainer(container, viewNode, true, beforeNode);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-05-11 20:57:37 -07:00
|
|
|
return viewNode;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Removes a view from a container.
|
|
|
|
*
|
2017-12-13 19:34:46 -08:00
|
|
|
* This method splices the view from the container's array of active views. It also
|
2017-12-13 11:04:46 -08:00
|
|
|
* removes the view's elements from the DOM and conducts cleanup (e.g. removing
|
|
|
|
* listeners, calling onDestroys).
|
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param container The container from which to remove a view
|
|
|
|
* @param removeIndex The index of the view to remove
|
|
|
|
* @returns The removed view
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function removeView(container: LContainerNode, removeIndex: number): LViewNode {
|
2017-12-13 19:34:46 -08:00
|
|
|
const views = container.data.views;
|
|
|
|
const viewNode = views[removeIndex];
|
2017-12-01 14:23:03 -08:00
|
|
|
if (removeIndex > 0) {
|
2018-05-11 20:57:37 -07:00
|
|
|
views[removeIndex - 1].data.next = viewNode.data.next as LView;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2017-12-13 19:34:46 -08:00
|
|
|
views.splice(removeIndex, 1);
|
2017-12-01 14:23:03 -08:00
|
|
|
destroyViewTree(viewNode.data);
|
|
|
|
addRemoveViewFromContainer(container, viewNode, false);
|
|
|
|
// Notify query that view has been removed
|
2018-01-29 14:51:37 +01:00
|
|
|
container.data.queries && container.data.queries.removeView(removeIndex);
|
2017-12-01 14:23:03 -08:00
|
|
|
return viewNode;
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2018-01-08 20:17:13 -08:00
|
|
|
* Determines which LViewOrLContainer to jump to when traversing back up the
|
2017-12-13 11:04:46 -08:00
|
|
|
* tree in destroyViewTree.
|
|
|
|
*
|
2018-01-08 20:17:13 -08:00
|
|
|
* Normally, the view's parent LView should be checked, but in the case of
|
2017-12-13 11:04:46 -08:00
|
|
|
* embedded views, the container (which is the view node's parent, but not the
|
2018-01-08 20:17:13 -08:00
|
|
|
* LView's parent) needs to be checked for a possible next property.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-01-08 20:17:13 -08:00
|
|
|
* @param state The LViewOrLContainer for which we need a parent state
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param rootView The rootView, so we don't propagate too far up the view tree
|
2018-01-08 20:17:13 -08:00
|
|
|
* @returns The correct parent LViewOrLContainer
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function getParentState(state: LViewOrLContainer, rootView: LView): LViewOrLContainer|null {
|
2017-12-01 14:23:03 -08:00
|
|
|
let node;
|
2018-05-17 12:54:57 -07:00
|
|
|
if ((node = (state as LView) !.node) && node.tNode.type === TNodeType.View) {
|
2017-12-01 14:23:03 -08:00
|
|
|
// if it's an embedded view, the state needs to go up to the container, in case the
|
|
|
|
// container has a next
|
2018-05-29 15:08:30 -07:00
|
|
|
return getParentLNode(node) !.data as any;
|
2017-12-01 14:23:03 -08:00
|
|
|
} else {
|
|
|
|
// otherwise, use parent view for containers or component views
|
|
|
|
return state.parent === rootView ? null : state.parent;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Removes all listeners and call all onDestroys in a given view.
|
|
|
|
*
|
2018-01-08 20:17:13 -08:00
|
|
|
* @param view The LView to clean up
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
function cleanUpView(view: LView): void {
|
2018-01-23 10:57:48 -08:00
|
|
|
removeListeners(view);
|
|
|
|
executeOnDestroys(view);
|
2018-03-21 15:10:34 -07:00
|
|
|
executePipeOnDestroys(view);
|
2018-01-23 10:57:48 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Removes listeners and unsubscribes from output subscriptions */
|
|
|
|
function removeListeners(view: LView): void {
|
2018-01-08 20:17:13 -08:00
|
|
|
const cleanup = view.cleanup !;
|
2018-01-23 10:57:48 -08:00
|
|
|
if (cleanup != null) {
|
|
|
|
for (let i = 0; i < cleanup.length - 1; i += 2) {
|
|
|
|
if (typeof cleanup[i] === 'string') {
|
|
|
|
cleanup ![i + 1].removeEventListener(cleanup[i], cleanup[i + 2], cleanup[i + 3]);
|
|
|
|
i += 2;
|
|
|
|
} else {
|
|
|
|
cleanup[i].call(cleanup[i + 1]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
view.cleanup = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Calls onDestroy hooks for this view */
|
|
|
|
function executeOnDestroys(view: LView): void {
|
|
|
|
const tView = view.tView;
|
|
|
|
let destroyHooks: HookData|null;
|
|
|
|
if (tView != null && (destroyHooks = tView.destroyHooks) != null) {
|
2018-03-21 15:10:34 -07:00
|
|
|
callHooks(view.directives !, destroyHooks);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Calls pipe destroy hooks for this view */
|
|
|
|
function executePipeOnDestroys(view: LView): void {
|
|
|
|
const pipeDestroyHooks = view.tView && view.tView.pipeDestroyHooks;
|
|
|
|
if (pipeDestroyHooks) {
|
|
|
|
callHooks(view.data !, pipeDestroyHooks);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2018-02-06 14:32:52 -08:00
|
|
|
* Returns whether a native element should be inserted in the given parent.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-02-06 14:32:52 -08:00
|
|
|
* The native node can be inserted when its parent is:
|
|
|
|
* - A regular element => Yes
|
|
|
|
* - A component host element =>
|
|
|
|
* - if the `currentView` === the parent `view`: The element is in the content (vs the
|
|
|
|
* template)
|
|
|
|
* => don't add as the parent component will project if needed.
|
|
|
|
* - `currentView` !== the parent `view` => The element is in the template (vs the content),
|
|
|
|
* add it
|
|
|
|
* - View element => delay insertion, will be done on `viewEnd()`
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-01-26 11:36:31 +01:00
|
|
|
* @param parent The parent in which to insert the child
|
2018-02-06 14:32:52 -08:00
|
|
|
* @param currentView The LView being processed
|
|
|
|
* @return boolean Whether the child element should be inserted.
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-02-06 14:32:52 -08:00
|
|
|
export function canInsertNativeNode(parent: LNode, currentView: LView): boolean {
|
2018-05-17 12:54:57 -07:00
|
|
|
const parentIsElement = parent.tNode.type === TNodeType.Element;
|
2018-02-06 14:32:52 -08:00
|
|
|
|
|
|
|
return parentIsElement &&
|
|
|
|
(parent.view !== currentView || parent.data === null /* Regular Element. */);
|
2018-01-26 11:36:31 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2018-02-06 20:08:46 -08:00
|
|
|
* Appends the `child` element to the `parent`.
|
2018-01-26 11:36:31 +01:00
|
|
|
*
|
2018-02-06 14:32:52 -08:00
|
|
|
* The element insertion might be delayed {@link canInsertNativeNode}
|
2018-01-26 11:36:31 +01:00
|
|
|
*
|
|
|
|
* @param parent The parent to which to append the child
|
|
|
|
* @param child The child that should be appended
|
|
|
|
* @param currentView The current LView
|
|
|
|
* @returns Whether or not the child was appended
|
|
|
|
*/
|
|
|
|
export function appendChild(parent: LNode, child: RNode | null, currentView: LView): boolean {
|
|
|
|
if (child !== null && canInsertNativeNode(parent, currentView)) {
|
|
|
|
// We only add element if not in View or not projected.
|
2017-12-01 14:23:03 -08:00
|
|
|
const renderer = currentView.renderer;
|
2018-02-06 20:08:46 -08:00
|
|
|
isProceduralRenderer(renderer) ? renderer.appendChild(parent.native !as RElement, child) :
|
|
|
|
parent.native !.appendChild(child);
|
2017-12-01 14:23:03 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2018-02-06 14:32:52 -08:00
|
|
|
* Inserts the provided node before the correct element in the DOM.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-02-06 14:32:52 -08:00
|
|
|
* The element insertion might be delayed {@link canInsertNativeNode}
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param node Node to insert
|
2018-01-08 20:17:13 -08:00
|
|
|
* @param currentView Current LView
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function insertChild(node: LNode, currentView: LView): void {
|
2018-05-29 15:08:30 -07:00
|
|
|
const parent = getParentLNode(node) !;
|
2018-01-26 11:36:31 +01:00
|
|
|
if (canInsertNativeNode(parent, currentView)) {
|
2018-02-02 14:59:31 -08:00
|
|
|
let nativeSibling: RNode|null = findNextRNodeSibling(node, null);
|
2017-12-01 14:23:03 -08:00
|
|
|
const renderer = currentView.renderer;
|
2018-02-06 20:08:46 -08:00
|
|
|
isProceduralRenderer(renderer) ?
|
|
|
|
renderer.insertBefore(parent.native !, node.native !, nativeSibling) :
|
2017-12-01 14:23:03 -08:00
|
|
|
parent.native !.insertBefore(node.native !, nativeSibling, false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Appends a projected node to the DOM, or in the case of a projected container,
|
2018-01-26 11:36:31 +01:00
|
|
|
* appends the nodes from all of the container's active views to the DOM.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param node The node to process
|
|
|
|
* @param currentParent The last parent element to be processed
|
2018-01-08 20:17:13 -08:00
|
|
|
* @param currentView Current LView
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-26 11:36:31 +01:00
|
|
|
export function appendProjectedNode(
|
2018-04-10 17:37:11 +02:00
|
|
|
node: LElementNode | LTextNode | LContainerNode, currentParent: LElementNode,
|
2018-01-25 15:32:21 +01:00
|
|
|
currentView: LView): void {
|
2018-05-17 12:54:57 -07:00
|
|
|
if (node.tNode.type !== TNodeType.Container) {
|
2018-01-26 11:36:31 +01:00
|
|
|
appendChild(currentParent, (node as LElementNode | LTextNode).native, currentView);
|
2018-04-10 17:37:11 +02:00
|
|
|
} else {
|
2017-12-08 11:48:54 -08:00
|
|
|
// The node we are adding is a Container and we are adding it to Element which
|
|
|
|
// is not a component (no more re-projection).
|
|
|
|
// Alternatively a container is projected at the root of a component's template
|
|
|
|
// and can't be re-projected (as not content of any component).
|
|
|
|
// Assignee the final projection location in those cases.
|
2018-01-08 20:17:13 -08:00
|
|
|
const lContainer = (node as LContainerNode).data;
|
2018-04-10 17:37:11 +02:00
|
|
|
lContainer.renderParent = currentParent;
|
2018-01-08 20:17:13 -08:00
|
|
|
const views = lContainer.views;
|
2017-12-01 14:23:03 -08:00
|
|
|
for (let i = 0; i < views.length; i++) {
|
2018-01-08 20:17:13 -08:00
|
|
|
addRemoveViewFromContainer(node as LContainerNode, views[i], true, null);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
2018-04-09 17:35:50 +02:00
|
|
|
if (node.dynamicLContainerNode) {
|
2018-04-10 17:37:11 +02:00
|
|
|
node.dynamicLContainerNode.data.renderParent = currentParent;
|
2018-04-09 17:35:50 +02:00
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|