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-07-03 20:04:36 -07:00
|
|
|
import {assertDefined} from './assert';
|
2018-08-28 16:49:52 -07:00
|
|
|
import {attachPatchData, readElementValue} from './context_discovery';
|
2018-01-25 20:41:57 -08:00
|
|
|
import {callHooks} from './hooks';
|
2018-10-11 13:13:57 -07:00
|
|
|
import {HOST_NATIVE, LContainer, NATIVE, RENDER_PARENT, VIEWS, unusedValueExportToPlacateAjd as unused1} from './interfaces/container';
|
2018-10-09 14:47:18 -07:00
|
|
|
import {LContainerNode, LElementContainerNode, LElementNode, LTextNode, TContainerNode, TElementContainerNode, TElementNode, TNode, TNodeFlags, TNodeType, TViewNode, unusedValueExportToPlacateAjd as unused2} from './interfaces/node';
|
2018-01-25 15:32:21 +01:00
|
|
|
import {unusedValueExportToPlacateAjd as unused3} from './interfaces/projection';
|
2018-06-06 17:30:48 +02:00
|
|
|
import {ProceduralRenderer3, RComment, RElement, RNode, RText, Renderer3, isProceduralRenderer, unusedValueExportToPlacateAjd as unused4} from './interfaces/renderer';
|
2018-10-11 13:13:57 -07:00
|
|
|
import {StylingIndex} from './interfaces/styling';
|
2018-10-08 16:04:46 -07:00
|
|
|
import {CLEANUP, CONTAINER_INDEX, FLAGS, HEADER_OFFSET, HOST_NODE, HookData, LViewData, LViewFlags, NEXT, PARENT, QUERIES, RENDERER, TVIEW, unusedValueExportToPlacateAjd as unused5} from './interfaces/view';
|
2018-09-13 16:07:23 -07:00
|
|
|
import {assertNodeType} from './node_assert';
|
2018-10-11 13:13:57 -07:00
|
|
|
import {getLNode, isLContainer, 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
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
/** Retrieves the parent LNode of a given node. */
|
|
|
|
export function getParentLNode(tNode: TNode, currentView: LViewData): LElementNode|
|
|
|
|
LElementContainerNode|LContainerNode|null {
|
|
|
|
return tNode.parent == null ? getHostElementNode(currentView) :
|
2018-09-17 14:32:45 -07:00
|
|
|
getLNode(tNode.parent, currentView);
|
2018-05-24 13:13:51 -07:00
|
|
|
}
|
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
/**
|
|
|
|
* Gets the host LElementNode given a view. Will return null if the host element is an
|
|
|
|
* LViewNode, since they are being phased out.
|
|
|
|
*/
|
|
|
|
export function getHostElementNode(currentView: LViewData): LElementNode|null {
|
|
|
|
const hostTNode = currentView[HOST_NODE] as TElementNode;
|
|
|
|
return hostTNode && hostTNode.type !== TNodeType.View ?
|
2018-09-17 14:32:45 -07:00
|
|
|
(getLNode(hostTNode, currentView[PARENT] !) as LElementNode) :
|
2018-09-13 16:07:23 -07:00
|
|
|
null;
|
|
|
|
}
|
|
|
|
|
2018-10-11 13:13:57 -07:00
|
|
|
export function getLContainer(tNode: TViewNode, embeddedView: LViewData): LContainer|null {
|
2018-09-13 16:07:23 -07:00
|
|
|
if (tNode.index === -1) {
|
2018-06-26 11:39:56 -07:00
|
|
|
// This is a dynamically created view inside a dynamic container.
|
|
|
|
// If the host index is -1, the view has not yet been inserted, so it has no parent.
|
2018-09-13 16:07:23 -07:00
|
|
|
const containerHostIndex = embeddedView[CONTAINER_INDEX];
|
2018-10-11 13:13:57 -07:00
|
|
|
return containerHostIndex > -1 ? embeddedView[PARENT] ![containerHostIndex] : null;
|
2018-09-13 16:07:23 -07:00
|
|
|
} else {
|
|
|
|
// This is a inline view node (e.g. embeddedViewStart)
|
2018-10-11 13:13:57 -07:00
|
|
|
return embeddedView[PARENT] ![tNode.parent !.index] as LContainer;
|
2018-06-18 18:07:05 +02:00
|
|
|
}
|
2018-05-29 15:08:30 -07:00
|
|
|
}
|
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
|
2018-08-01 15:19:27 +02:00
|
|
|
/**
|
|
|
|
* Retrieves render parent LElementNode for a given view.
|
2018-09-13 16:07:23 -07:00
|
|
|
* Might be null if a view is not yet attached to any container.
|
2018-08-01 15:19:27 +02:00
|
|
|
*/
|
2018-09-13 16:07:23 -07:00
|
|
|
export function getContainerRenderParent(tViewNode: TViewNode, view: LViewData): LElementNode|null {
|
2018-10-11 13:13:57 -07:00
|
|
|
const container = getLContainer(tViewNode, view);
|
|
|
|
return container ? container[RENDER_PARENT] : null;
|
2018-08-01 15:19:27 +02:00
|
|
|
}
|
|
|
|
|
2018-09-08 10:55:41 -07:00
|
|
|
const enum WalkTNodeTreeAction {
|
2018-05-31 15:45:46 +02:00
|
|
|
/** node insert in the native environment */
|
2018-06-06 17:30:48 +02:00
|
|
|
Insert = 0,
|
2018-05-31 15:45:46 +02:00
|
|
|
|
|
|
|
/** node detach from the native environment */
|
2018-06-06 17:30:48 +02:00
|
|
|
Detach = 1,
|
2018-05-31 15:45:46 +02:00
|
|
|
|
|
|
|
/** node destruction using the renderer's API */
|
2018-06-06 17:30:48 +02:00
|
|
|
Destroy = 2,
|
2018-05-31 15:45:46 +02:00
|
|
|
}
|
|
|
|
|
2018-07-03 20:04:36 -07:00
|
|
|
|
|
|
|
/**
|
2018-09-08 10:55:41 -07:00
|
|
|
* Stack used to keep track of projection nodes in walkTNodeTree.
|
2018-07-03 20:04:36 -07:00
|
|
|
*
|
2018-09-08 10:55:41 -07:00
|
|
|
* This is deliberately created outside of walkTNodeTree to avoid allocating
|
2018-07-03 20:04:36 -07:00
|
|
|
* a new array each time the function is called. Instead the array will be
|
|
|
|
* re-used by each invocation. This works because the function is not reentrant.
|
|
|
|
*/
|
2018-09-08 10:55:41 -07:00
|
|
|
const projectionNodeStack: (LViewData | TNode)[] = [];
|
2018-07-03 20:04:36 -07:00
|
|
|
|
2018-05-31 15:45:46 +02:00
|
|
|
/**
|
2018-09-08 10:55:41 -07:00
|
|
|
* Walks a tree of TNodes, applying a transformation on the element nodes, either only on the first
|
2018-05-31 15:45:46 +02:00
|
|
|
* one found, or on all of them.
|
|
|
|
*
|
2018-09-08 10:55:41 -07:00
|
|
|
* @param viewToWalk the view to walk
|
2018-06-06 17:30:48 +02:00
|
|
|
* @param action identifies the action to be performed on the LElement nodes.
|
|
|
|
* @param renderer the current renderer.
|
|
|
|
* @param renderParentNode Optional the render parent node to be set in all LContainerNodes found,
|
|
|
|
* required for action modes Insert and Destroy.
|
|
|
|
* @param beforeNode Optional the node before which elements should be added, required for action
|
|
|
|
* Insert.
|
2018-05-31 15:45:46 +02:00
|
|
|
*/
|
2018-09-08 10:55:41 -07:00
|
|
|
function walkTNodeTree(
|
|
|
|
viewToWalk: LViewData, action: WalkTNodeTreeAction, renderer: Renderer3,
|
2018-05-31 15:45:46 +02:00
|
|
|
renderParentNode?: LElementNode | null, beforeNode?: RNode | null) {
|
2018-09-08 10:55:41 -07:00
|
|
|
const rootTNode = viewToWalk[TVIEW].node as TViewNode;
|
2018-07-03 20:04:36 -07:00
|
|
|
let projectionNodeIndex = -1;
|
2018-09-08 10:55:41 -07:00
|
|
|
let currentView = viewToWalk;
|
|
|
|
let tNode: TNode|null = rootTNode.child as TNode;
|
|
|
|
while (tNode) {
|
|
|
|
let nextTNode: TNode|null = null;
|
2018-06-06 17:30:48 +02:00
|
|
|
const parent = renderParentNode ? renderParentNode.native : null;
|
2018-09-08 10:55:41 -07:00
|
|
|
if (tNode.type === TNodeType.Element) {
|
2018-09-17 14:32:45 -07:00
|
|
|
const elementNode = getLNode(tNode, currentView);
|
2018-09-08 10:55:41 -07:00
|
|
|
executeNodeAction(action, renderer, parent, elementNode.native !, beforeNode);
|
2018-10-11 13:13:57 -07:00
|
|
|
const nodeOrContainer = currentView[tNode.index];
|
|
|
|
if (isLContainer(nodeOrContainer)) {
|
|
|
|
// This element has an LContainer, and its comment needs to be handled
|
|
|
|
executeNodeAction(action, renderer, parent, nodeOrContainer[NATIVE], beforeNode);
|
2018-05-31 15:45:46 +02:00
|
|
|
}
|
2018-09-08 10:55:41 -07:00
|
|
|
} else if (tNode.type === TNodeType.Container) {
|
2018-10-11 13:13:57 -07:00
|
|
|
const lContainer = currentView ![tNode.index] as LContainer;
|
|
|
|
executeNodeAction(action, renderer, parent, lContainer[NATIVE], beforeNode);
|
|
|
|
|
|
|
|
if (renderParentNode) lContainer[RENDER_PARENT] = renderParentNode;
|
2018-09-08 10:55:41 -07:00
|
|
|
|
2018-10-11 13:13:57 -07:00
|
|
|
if (lContainer[VIEWS].length) {
|
|
|
|
currentView = lContainer[VIEWS][0];
|
2018-09-08 10:55:41 -07:00
|
|
|
nextTNode = currentView[TVIEW].node;
|
|
|
|
|
2018-06-06 17:30:48 +02:00
|
|
|
// When the walker enters a container, then the beforeNode has to become the local native
|
|
|
|
// comment node.
|
2018-10-11 13:13:57 -07:00
|
|
|
beforeNode = lContainer[NATIVE];
|
2018-06-06 17:30:48 +02:00
|
|
|
}
|
2018-09-08 10:55:41 -07:00
|
|
|
} else if (tNode.type === TNodeType.Projection) {
|
2018-09-13 16:07:23 -07:00
|
|
|
const componentView = findComponentView(currentView !);
|
|
|
|
const componentHost = componentView[HOST_NODE] as TElementNode;
|
2018-09-08 10:55:41 -07:00
|
|
|
const head: TNode|null =
|
2018-09-13 16:07:23 -07:00
|
|
|
(componentHost.projection as(TNode | null)[])[tNode.projection as number];
|
2018-09-08 10:55:41 -07:00
|
|
|
|
|
|
|
// Must store both the TNode and the view because this projection node could be nested
|
|
|
|
// deeply inside embedded views, and we need to get back down to this particular nested view.
|
|
|
|
projectionNodeStack[++projectionNodeIndex] = tNode;
|
|
|
|
projectionNodeStack[++projectionNodeIndex] = currentView !;
|
|
|
|
if (head) {
|
2018-09-13 16:07:23 -07:00
|
|
|
currentView = componentView[PARENT] !;
|
2018-09-08 10:55:41 -07:00
|
|
|
nextTNode = currentView[TVIEW].data[head.index] as TNode;
|
|
|
|
}
|
2018-01-25 15:32:21 +01:00
|
|
|
} else {
|
2018-09-08 10:55:41 -07:00
|
|
|
// Otherwise, this is a View or an ElementContainer
|
|
|
|
nextTNode = tNode.child;
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
2018-02-02 14:59:31 -08:00
|
|
|
|
2018-09-08 10:55:41 -07:00
|
|
|
if (nextTNode === null) {
|
2018-07-03 20:04:36 -07:00
|
|
|
// this last node was projected, we need to get back down to its projection node
|
2018-09-08 10:55:41 -07:00
|
|
|
if (tNode.next === null && (tNode.flags & TNodeFlags.isProjected)) {
|
|
|
|
currentView = projectionNodeStack[projectionNodeIndex--] as LViewData;
|
|
|
|
tNode = projectionNodeStack[projectionNodeIndex--] as TNode;
|
2018-07-03 20:04:36 -07:00
|
|
|
}
|
2018-09-08 10:55:41 -07:00
|
|
|
nextTNode = tNode.next;
|
|
|
|
|
2018-06-06 17:30:48 +02: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).
|
|
|
|
*
|
|
|
|
* If there is no sibling node, then it goes to the next sibling of the parent node...
|
|
|
|
* until it reaches rootNode (at which point null is returned).
|
|
|
|
*/
|
2018-09-08 10:55:41 -07:00
|
|
|
while (!nextTNode) {
|
|
|
|
// If parent is null, we're crossing the view boundary, so we should get the host TNode.
|
|
|
|
tNode = tNode.parent || currentView[TVIEW].node;
|
|
|
|
|
|
|
|
if (tNode === null || tNode === rootTNode) return null;
|
2018-07-03 20:04:36 -07:00
|
|
|
|
|
|
|
// When exiting a container, the beforeNode must be restored to the previous value
|
2018-09-08 10:55:41 -07:00
|
|
|
if (tNode.type === TNodeType.Container) {
|
|
|
|
currentView = currentView[PARENT] !;
|
2018-10-11 13:13:57 -07:00
|
|
|
beforeNode = currentView[tNode.index][NATIVE];
|
2018-09-08 10:55:41 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
if (tNode.type === TNodeType.View && currentView[NEXT]) {
|
|
|
|
currentView = currentView[NEXT] as LViewData;
|
|
|
|
nextTNode = currentView[TVIEW].node;
|
|
|
|
} else {
|
|
|
|
nextTNode = tNode.next;
|
2018-06-06 17:30:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-09-08 10:55:41 -07:00
|
|
|
tNode = nextTNode;
|
2018-06-06 17:30:48 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-03 20:04:36 -07:00
|
|
|
/**
|
|
|
|
* Given a current view, finds the nearest component's host (LElement).
|
|
|
|
*
|
|
|
|
* @param lViewData LViewData for which we want a host element node
|
|
|
|
* @returns The host node
|
|
|
|
*/
|
2018-09-13 16:07:23 -07:00
|
|
|
export function findComponentView(lViewData: LViewData): LViewData {
|
|
|
|
let rootTNode = lViewData[HOST_NODE];
|
2018-07-03 20:04:36 -07:00
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
while (rootTNode && rootTNode.type === TNodeType.View) {
|
|
|
|
ngDevMode && assertDefined(lViewData[PARENT], 'viewData.parent');
|
2018-07-03 20:04:36 -07:00
|
|
|
lViewData = lViewData[PARENT] !;
|
2018-09-13 16:07:23 -07:00
|
|
|
rootTNode = lViewData[HOST_NODE];
|
2018-07-03 20:04:36 -07:00
|
|
|
}
|
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
return lViewData;
|
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.
|
|
|
|
*/
|
|
|
|
function executeNodeAction(
|
2018-09-08 10:55:41 -07:00
|
|
|
action: WalkTNodeTreeAction, renderer: Renderer3, parent: RElement | null,
|
2018-06-06 17:30:48 +02:00
|
|
|
node: RComment | RElement | RText, beforeNode?: RNode | null) {
|
2018-09-08 10:55:41 -07:00
|
|
|
if (action === WalkTNodeTreeAction.Insert) {
|
2018-06-06 17:30:48 +02:00
|
|
|
isProceduralRenderer(renderer !) ?
|
|
|
|
(renderer as ProceduralRenderer3).insertBefore(parent !, node, beforeNode as RNode | null) :
|
|
|
|
parent !.insertBefore(node, beforeNode as RNode | null, true);
|
2018-09-08 10:55:41 -07:00
|
|
|
} else if (action === WalkTNodeTreeAction.Detach) {
|
2018-06-06 17:30:48 +02:00
|
|
|
isProceduralRenderer(renderer !) ?
|
|
|
|
(renderer as ProceduralRenderer3).removeChild(parent !, node) :
|
|
|
|
parent !.removeChild(node);
|
2018-09-08 10:55:41 -07:00
|
|
|
} else if (action === WalkTNodeTreeAction.Destroy) {
|
2018-06-06 17:30:48 +02:00
|
|
|
ngDevMode && ngDevMode.rendererDestroyNode++;
|
|
|
|
(renderer as ProceduralRenderer3).destroyNode !(node);
|
2018-01-25 15:32:21 +01:00
|
|
|
}
|
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.
|
|
|
|
*
|
2018-09-08 10:55:41 -07:00
|
|
|
* @param viewToWalk The view from which elements should be added or removed
|
2017-12-13 16:32:21 -08:00
|
|
|
* @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-09-13 16:07:23 -07:00
|
|
|
viewToWalk: LViewData, insertMode: true, beforeNode: RNode | null): void;
|
|
|
|
export function addRemoveViewFromContainer(viewToWalk: LViewData, insertMode: false): void;
|
2017-12-01 14:23:03 -08:00
|
|
|
export function addRemoveViewFromContainer(
|
2018-09-13 16:07:23 -07:00
|
|
|
viewToWalk: LViewData, insertMode: boolean, beforeNode?: RNode | null): void {
|
|
|
|
const parentNode = getContainerRenderParent(viewToWalk[TVIEW].node as TViewNode, viewToWalk);
|
2018-01-25 15:32:21 +01:00
|
|
|
const parent = parentNode ? parentNode.native : null;
|
2018-09-08 10:55:41 -07:00
|
|
|
ngDevMode && assertNodeType(viewToWalk[TVIEW].node as TNode, TNodeType.View);
|
2017-12-01 14:23:03 -08:00
|
|
|
if (parent) {
|
2018-09-13 16:07:23 -07:00
|
|
|
const renderer = viewToWalk[RENDERER];
|
2018-09-08 10:55:41 -07:00
|
|
|
walkTNodeTree(
|
|
|
|
viewToWalk, insertMode ? WalkTNodeTreeAction.Insert : WalkTNodeTreeAction.Detach, renderer,
|
|
|
|
parentNode, beforeNode);
|
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-06-07 22:42:32 -07:00
|
|
|
export function destroyViewTree(rootView: LViewData): void {
|
2018-05-30 13:43:14 -07:00
|
|
|
// If the view has no children, we can clean it up and return early.
|
2018-06-07 22:42:32 -07:00
|
|
|
if (rootView[TVIEW].childIndex === -1) {
|
2018-04-19 22:59:58 +02:00
|
|
|
return cleanUpView(rootView);
|
|
|
|
}
|
2018-06-07 22:42:32 -07:00
|
|
|
let viewOrContainer: LViewData|LContainer|null = getLViewChild(rootView);
|
2017-12-01 14:23:03 -08:00
|
|
|
|
2018-01-08 20:17:13 -08:00
|
|
|
while (viewOrContainer) {
|
2018-06-07 22:42:32 -07:00
|
|
|
let next: LViewData|LContainer|null = null;
|
|
|
|
|
|
|
|
if (viewOrContainer.length >= HEADER_OFFSET) {
|
|
|
|
// If LViewData, traverse down to child.
|
|
|
|
const view = viewOrContainer as LViewData;
|
|
|
|
if (view[TVIEW].childIndex > -1) next = getLViewChild(view);
|
|
|
|
} else {
|
|
|
|
// If container, traverse down to its first LViewData.
|
|
|
|
const container = viewOrContainer as LContainer;
|
2018-09-12 08:47:03 -07:00
|
|
|
if (container[VIEWS].length) next = container[VIEWS][0];
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (next == null) {
|
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.
|
|
|
|
while (viewOrContainer && !viewOrContainer ![NEXT] && viewOrContainer !== rootView) {
|
2018-06-05 15:28:15 -07:00
|
|
|
cleanUpView(viewOrContainer);
|
2018-01-08 20:17:13 -08:00
|
|
|
viewOrContainer = getParentState(viewOrContainer, rootView);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2018-06-05 15:28:15 -07:00
|
|
|
cleanUpView(viewOrContainer || rootView);
|
2018-06-07 22:42:32 -07: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).
|
|
|
|
*
|
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
|
|
|
|
* @param parentView The new parent of the inserted view
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param index The index at which to insert the view
|
2018-09-17 14:32:45 -07:00
|
|
|
* @param containerIndex The index of the container node, if dynamic
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-09-13 16:07:23 -07:00
|
|
|
export function insertView(
|
2018-09-17 14:32:45 -07:00
|
|
|
lView: LViewData, lContainer: LContainer, parentView: LViewData, index: number,
|
|
|
|
containerIndex: number) {
|
|
|
|
const views = lContainer[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-09-12 08:47:03 -07:00
|
|
|
views[index - 1][NEXT] = lView;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-03-08 12:10:20 +01:00
|
|
|
if (index < views.length) {
|
2018-09-12 08:47:03 -07:00
|
|
|
lView[NEXT] = views[index];
|
|
|
|
views.splice(index, 0, lView);
|
2018-03-08 12:10:20 +01:00
|
|
|
} else {
|
2018-09-12 08:47:03 -07:00
|
|
|
views.push(lView);
|
2018-06-26 11:39:56 -07:00
|
|
|
lView[NEXT] = null;
|
|
|
|
}
|
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
// Dynamically inserted views need a reference to their parent container's host so it's
|
2018-06-26 11:39:56 -07:00
|
|
|
// possible to jump from a view to its container's next when walking the node tree.
|
2018-09-13 16:07:23 -07:00
|
|
|
if (containerIndex > -1) {
|
|
|
|
lView[CONTAINER_INDEX] = containerIndex;
|
2018-09-17 14:32:45 -07:00
|
|
|
lView[PARENT] = parentView;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-05-28 11:57:36 +02:00
|
|
|
// Notify query that a new view has been added
|
2018-06-07 22:42:32 -07:00
|
|
|
if (lView[QUERIES]) {
|
|
|
|
lView[QUERIES] !.insertView(index);
|
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
|
|
|
}
|
|
|
|
|
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
|
|
|
*
|
2017-12-13 19:34:46 -08:00
|
|
|
* This method splices 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-09-17 14:32:45 -07:00
|
|
|
* @param detached Whether or not this view is already detached.
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-09-17 14:32:45 -07:00
|
|
|
export function detachView(lContainer: LContainer, removeIndex: number, detached: boolean) {
|
|
|
|
const views = lContainer[VIEWS];
|
2018-09-12 08:47:03 -07:00
|
|
|
const viewToDetach = views[removeIndex];
|
2017-12-01 14:23:03 -08:00
|
|
|
if (removeIndex > 0) {
|
2018-09-12 08:47:03 -07:00
|
|
|
views[removeIndex - 1][NEXT] = viewToDetach[NEXT] as LViewData;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2017-12-13 19:34:46 -08:00
|
|
|
views.splice(removeIndex, 1);
|
2018-09-13 16:07:23 -07:00
|
|
|
if (!detached) {
|
|
|
|
addRemoveViewFromContainer(viewToDetach, false);
|
2018-06-18 16:55:43 +02:00
|
|
|
}
|
2018-09-12 08:47:03 -07:00
|
|
|
|
|
|
|
if (viewToDetach[QUERIES]) {
|
|
|
|
viewToDetach[QUERIES] !.removeView();
|
2018-05-28 11:57:36 +02:00
|
|
|
}
|
2018-09-12 08:47:03 -07:00
|
|
|
viewToDetach[CONTAINER_INDEX] = -1;
|
2018-09-13 16:07:23 -07:00
|
|
|
viewToDetach[PARENT] = null;
|
2018-05-31 15:45:46 +02:00
|
|
|
// Unsets the attached flag
|
2018-09-12 08:47:03 -07:00
|
|
|
viewToDetach[FLAGS] &= ~LViewFlags.Attached;
|
2018-05-31 15:45:46 +02:00
|
|
|
}
|
2018-05-28 11:57:36 +02:00
|
|
|
|
2018-05-31 15:45:46 +02:00
|
|
|
/**
|
|
|
|
* Removes a view from a container, i.e. detaches it and then destroys the underlying LView.
|
|
|
|
*
|
2018-09-17 14:32:45 -07:00
|
|
|
* @param lContainer The container from which to remove a view
|
|
|
|
* @param tContainer The TContainer node associated with the LContainer
|
2018-05-31 15:45:46 +02:00
|
|
|
* @param removeIndex The index of the view to remove
|
|
|
|
*/
|
2018-09-13 16:07:23 -07:00
|
|
|
export function removeView(
|
2018-10-09 14:47:18 -07:00
|
|
|
lContainer: LContainer, containerHost: TElementNode | TContainerNode | TElementContainerNode,
|
|
|
|
removeIndex: number) {
|
2018-09-17 14:32:45 -07:00
|
|
|
const view = lContainer[VIEWS][removeIndex];
|
2018-10-09 14:47:18 -07:00
|
|
|
detachView(lContainer, removeIndex, !!containerHost.detached);
|
2018-09-24 10:30:29 +02:00
|
|
|
destroyLView(view);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2018-06-07 22:42:32 -07:00
|
|
|
/** Gets the child of the given LViewData */
|
|
|
|
export function getLViewChild(viewData: LViewData): LViewData|LContainer|null {
|
2018-10-11 13:13:57 -07:00
|
|
|
const childIndex = viewData[TVIEW].childIndex;
|
|
|
|
if (childIndex === -1) return null;
|
2018-05-30 13:43:14 -07:00
|
|
|
|
2018-10-11 13:13:57 -07:00
|
|
|
const value: LElementNode|LContainerNode|LContainer = viewData[childIndex];
|
2018-05-30 13:43:14 -07:00
|
|
|
|
2018-10-11 13:13:57 -07:00
|
|
|
// If it's an array, it's an LContainer. Otherwise, it's a component node, so LViewData
|
|
|
|
// is stored in data.
|
|
|
|
return Array.isArray(value) ? value : value.data;
|
2018-05-30 13:43:14 -07:00
|
|
|
}
|
|
|
|
|
2018-05-31 15:45:46 +02:00
|
|
|
/**
|
|
|
|
* A standalone function which destroys an LView,
|
|
|
|
* conducting cleanup (e.g. removing listeners, calling onDestroys).
|
|
|
|
*
|
|
|
|
* @param view The view to be destroyed.
|
|
|
|
*/
|
2018-06-07 22:42:32 -07:00
|
|
|
export function destroyLView(view: LViewData) {
|
|
|
|
const renderer = view[RENDERER];
|
2018-05-31 15:45:46 +02:00
|
|
|
if (isProceduralRenderer(renderer) && renderer.destroyNode) {
|
2018-09-08 10:55:41 -07:00
|
|
|
walkTNodeTree(view, WalkTNodeTreeAction.Destroy, renderer);
|
2018-05-31 15:45:46 +02:00
|
|
|
}
|
|
|
|
destroyViewTree(view);
|
|
|
|
// Sets the destroyed flag
|
2018-06-07 22:42:32 -07:00
|
|
|
view[FLAGS] |= LViewFlags.Destroyed;
|
2018-05-31 15:45:46 +02:00
|
|
|
}
|
|
|
|
|
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-06-07 22:42:32 -07:00
|
|
|
export function getParentState(state: LViewData | LContainer, rootView: LViewData): LViewData|
|
|
|
|
LContainer|null {
|
2018-09-13 16:07:23 -07:00
|
|
|
let tNode;
|
|
|
|
if (state.length >= HEADER_OFFSET && (tNode = (state as LViewData) ![HOST_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-10-11 13:13:57 -07:00
|
|
|
return getLContainer(tNode as TViewNode, state as LViewData) as LContainer;
|
2017-12-01 14:23:03 -08:00
|
|
|
} else {
|
|
|
|
// otherwise, use parent view for containers or component views
|
2018-06-07 22:42:32 -07:00
|
|
|
return state[PARENT] === rootView ? null : state[PARENT];
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Removes all listeners and call all onDestroys in a given view.
|
|
|
|
*
|
2018-06-07 22:42:32 -07:00
|
|
|
* @param view The LViewData to clean up
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-06-07 22:42:32 -07:00
|
|
|
function cleanUpView(viewOrContainer: LViewData | LContainer): void {
|
2018-10-11 13:13:57 -07:00
|
|
|
if ((viewOrContainer as LViewData).length >= HEADER_OFFSET) {
|
2018-06-07 22:42:32 -07:00
|
|
|
const view = viewOrContainer as LViewData;
|
2018-06-05 15:28:15 -07:00
|
|
|
removeListeners(view);
|
|
|
|
executeOnDestroys(view);
|
|
|
|
executePipeOnDestroys(view);
|
|
|
|
// For component views only, the local renderer is destroyed as clean up time.
|
2018-06-07 22:42:32 -07:00
|
|
|
if (view[TVIEW].id === -1 && isProceduralRenderer(view[RENDERER])) {
|
2018-06-05 15:28:15 -07:00
|
|
|
ngDevMode && ngDevMode.rendererDestroy++;
|
2018-06-07 22:42:32 -07:00
|
|
|
(view[RENDERER] as ProceduralRenderer3).destroy();
|
2018-06-05 15:28:15 -07:00
|
|
|
}
|
2018-05-22 17:40:59 +02:00
|
|
|
}
|
2018-01-23 10:57:48 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/** Removes listeners and unsubscribes from output subscriptions */
|
2018-06-07 22:42:32 -07:00
|
|
|
function removeListeners(viewData: LViewData): void {
|
|
|
|
const cleanup = viewData[TVIEW].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') {
|
2018-06-05 15:28:15 -07:00
|
|
|
// This is a listener with the native renderer
|
2018-07-11 09:56:47 -07:00
|
|
|
const native = readElementValue(viewData[cleanup[i + 1]]).native;
|
2018-06-07 22:42:32 -07:00
|
|
|
const listener = viewData[CLEANUP] ![cleanup[i + 2]];
|
2018-06-05 15:28:15 -07:00
|
|
|
native.removeEventListener(cleanup[i], listener, cleanup[i + 3]);
|
2018-01-23 10:57:48 -08:00
|
|
|
i += 2;
|
2018-06-05 15:28:15 -07:00
|
|
|
} else if (typeof cleanup[i] === 'number') {
|
|
|
|
// This is a listener with renderer2 (cleanup fn can be found by index)
|
2018-06-07 22:42:32 -07:00
|
|
|
const cleanupFn = viewData[CLEANUP] ![cleanup[i]];
|
2018-06-05 15:28:15 -07:00
|
|
|
cleanupFn();
|
2018-01-23 10:57:48 -08:00
|
|
|
} else {
|
2018-06-05 15:28:15 -07:00
|
|
|
// This is a cleanup function that is grouped with the index of its context
|
2018-06-07 22:42:32 -07:00
|
|
|
const context = viewData[CLEANUP] ![cleanup[i + 1]];
|
2018-06-05 15:28:15 -07:00
|
|
|
cleanup[i].call(context);
|
2018-01-23 10:57:48 -08:00
|
|
|
}
|
|
|
|
}
|
2018-06-07 22:42:32 -07:00
|
|
|
viewData[CLEANUP] = null;
|
2018-01-23 10:57:48 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Calls onDestroy hooks for this view */
|
2018-06-07 22:42:32 -07:00
|
|
|
function executeOnDestroys(view: LViewData): void {
|
|
|
|
const tView = view[TVIEW];
|
2018-01-23 10:57:48 -08:00
|
|
|
let destroyHooks: HookData|null;
|
|
|
|
if (tView != null && (destroyHooks = tView.destroyHooks) != null) {
|
2018-10-08 16:04:46 -07:00
|
|
|
callHooks(view, destroyHooks);
|
2018-03-21 15:10:34 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Calls pipe destroy hooks for this view */
|
2018-06-07 22:42:32 -07:00
|
|
|
function executePipeOnDestroys(viewData: LViewData): void {
|
|
|
|
const pipeDestroyHooks = viewData[TVIEW] && viewData[TVIEW].pipeDestroyHooks;
|
2018-03-21 15:10:34 -07:00
|
|
|
if (pipeDestroyHooks) {
|
2018-06-07 22:42:32 -07:00
|
|
|
callHooks(viewData !, pipeDestroyHooks);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
export function getRenderParent(tNode: TNode, currentView: LViewData): LElementNode|null {
|
|
|
|
if (canInsertNativeNode(tNode, currentView)) {
|
|
|
|
const hostTNode = currentView[HOST_NODE];
|
|
|
|
return tNode.parent == null && hostTNode !.type === TNodeType.View ?
|
|
|
|
getContainerRenderParent(hostTNode as TViewNode, currentView) :
|
|
|
|
getParentLNode(tNode, currentView) as LElementNode;
|
2018-08-06 16:40:16 +02:00
|
|
|
}
|
2018-09-13 16:07:23 -07:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
function canInsertNativeChildOfElement(tNode: TNode): boolean {
|
|
|
|
// If the parent is null, then we are inserting across views. This happens when we
|
|
|
|
// insert a root element of the component view into the component host element and it
|
|
|
|
// should always be eager.
|
|
|
|
if (tNode.parent == null ||
|
|
|
|
// We should also eagerly insert if the parent is a regular, non-component element
|
|
|
|
// since we know that this relationship will never be broken.
|
|
|
|
tNode.parent.type === TNodeType.Element && !(tNode.parent.flags & TNodeFlags.isComponent)) {
|
2018-08-06 16:40:16 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Parent is 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 than move it to its final destination.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-08-06 17:46:49 +02:00
|
|
|
/**
|
|
|
|
* We might delay insertion of children for a given view if it is disconnected.
|
2018-09-13 16:07:23 -07:00
|
|
|
* This might happen for 2 main reasons:
|
|
|
|
* - view is not inserted into any container (view was created but not inserted yet)
|
2018-08-06 17:46:49 +02:00
|
|
|
* - view is inserted into a container but the container itself is not inserted into the DOM
|
|
|
|
* (container might be part of projection or child of a view that is not inserted yet).
|
|
|
|
*
|
2018-09-13 16:07:23 -07:00
|
|
|
* In other words we can insert children of a given view if this view was inserted into a container
|
|
|
|
* and
|
|
|
|
* the container itself has its render parent determined.
|
2018-08-06 17:46:49 +02:00
|
|
|
*/
|
2018-09-13 16:07:23 -07:00
|
|
|
function canInsertNativeChildOfView(viewTNode: TViewNode, view: LViewData): boolean {
|
2018-08-06 16:40:16 +02:00
|
|
|
// Because we are inserting into a `View` the `View` may be disconnected.
|
2018-10-11 13:13:57 -07:00
|
|
|
const container = getLContainer(viewTNode, view) !;
|
|
|
|
if (container == null || container[RENDER_PARENT] == null) {
|
2018-09-13 16:07:23 -07:00
|
|
|
// The `View` is not inserted into a `Container` or the parent `Container`
|
|
|
|
// itself is disconnected. So we have to delay.
|
2018-08-06 16:40:16 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The parent `Container` is in inserted state, so we can eagerly insert into
|
|
|
|
// this location.
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
2018-06-22 15:37:38 +02:00
|
|
|
* Returns whether a native element 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
|
|
|
|
* of a View which has not be inserted or is mare for projection but has not been inserted
|
2018-06-22 15:37:38 +02:00
|
|
|
* into destination.
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-06-22 15:37:38 +02:00
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
*
|
2018-06-22 15:37:38 +02:00
|
|
|
* @param parent The parent where the child will be inserted into.
|
|
|
|
* @param currentView Current LView being processed.
|
|
|
|
* @return boolean Whether the child should be inserted now (or delayed until later).
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-09-13 16:07:23 -07:00
|
|
|
export function canInsertNativeNode(tNode: TNode, currentView: LViewData): boolean {
|
|
|
|
let currentNode = tNode;
|
|
|
|
let parent: TNode|null = tNode.parent;
|
2018-06-22 15:37:38 +02:00
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
if (tNode.parent && tNode.parent.type === TNodeType.ElementContainer) {
|
|
|
|
currentNode = getHighestElementContainer(tNode);
|
|
|
|
parent = currentNode.parent;
|
|
|
|
}
|
|
|
|
if (parent === null) parent = currentView[HOST_NODE];
|
|
|
|
|
|
|
|
if (parent && parent.type === TNodeType.View) {
|
|
|
|
return canInsertNativeChildOfView(parent as TViewNode, currentView);
|
2018-06-22 15:37:38 +02:00
|
|
|
} else {
|
2018-09-13 16:07:23 -07:00
|
|
|
// Parent is a regular element or a component
|
|
|
|
return canInsertNativeChildOfElement(currentNode);
|
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.
|
|
|
|
*/
|
|
|
|
function nativeInsertBefore(
|
|
|
|
renderer: Renderer3, parent: RElement, child: RNode, beforeNode: RNode | null): void {
|
|
|
|
if (isProceduralRenderer(renderer)) {
|
|
|
|
renderer.insertBefore(parent, child, beforeNode);
|
|
|
|
} else {
|
|
|
|
parent.insertBefore(child, beforeNode, true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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-06-18 16:55:43 +02:00
|
|
|
* The element insertion might be delayed {@link canInsertNativeNode}.
|
2018-01-26 11:36:31 +01:00
|
|
|
*
|
2018-09-13 16:07:23 -07:00
|
|
|
* @param childEl The child that should be appended
|
|
|
|
* @param childTNode The TNode of the child element
|
2018-01-26 11:36:31 +01:00
|
|
|
* @param currentView The current LView
|
|
|
|
* @returns Whether or not the child was appended
|
|
|
|
*/
|
2018-09-13 16:07:23 -07:00
|
|
|
export function appendChild(
|
|
|
|
childEl: RNode | null, childTNode: TNode, currentView: LViewData): boolean {
|
|
|
|
const parentLNode = getParentLNode(childTNode, currentView);
|
|
|
|
const parentEl = parentLNode ? parentLNode.native : null;
|
|
|
|
|
|
|
|
if (childEl !== null && canInsertNativeNode(childTNode, currentView)) {
|
2018-06-07 22:42:32 -07:00
|
|
|
const renderer = currentView[RENDERER];
|
2018-09-13 16:07:23 -07:00
|
|
|
const parentTNode: TNode = childTNode.parent || currentView[HOST_NODE] !;
|
|
|
|
|
|
|
|
if (parentTNode.type === TNodeType.View) {
|
2018-10-11 13:13:57 -07:00
|
|
|
const lContainer = getLContainer(parentTNode as TViewNode, currentView) as LContainer;
|
|
|
|
const renderParent = lContainer[RENDER_PARENT];
|
|
|
|
const views = lContainer[VIEWS];
|
2018-09-12 08:47:03 -07:00
|
|
|
const index = views.indexOf(currentView);
|
2018-09-13 16:07:23 -07:00
|
|
|
nativeInsertBefore(
|
2018-10-11 13:13:57 -07:00
|
|
|
renderer, renderParent !.native, childEl,
|
|
|
|
getBeforeNodeForView(index, views, lContainer[NATIVE]));
|
2018-09-13 16:07:23 -07:00
|
|
|
} else if (parentTNode.type === TNodeType.ElementContainer) {
|
|
|
|
let elementContainer = getHighestElementContainer(childTNode);
|
|
|
|
let node: LElementNode = getRenderParent(elementContainer, currentView) !;
|
|
|
|
nativeInsertBefore(renderer, node.native, childEl, parentEl);
|
2018-06-22 15:37:38 +02:00
|
|
|
} else {
|
2018-09-13 16:07:23 -07:00
|
|
|
isProceduralRenderer(renderer) ? renderer.appendChild(parentEl !as RElement, childEl) :
|
|
|
|
parentEl !.appendChild(childEl);
|
2018-06-22 15:37:38 +02:00
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2018-09-13 16:07:23 -07:00
|
|
|
/**
|
|
|
|
* Gets the top-level ng-container if ng-containers are nested.
|
|
|
|
*
|
|
|
|
* @param ngContainer The TNode of the starting ng-container
|
|
|
|
* @returns tNode The TNode of the highest level ng-container
|
|
|
|
*/
|
|
|
|
function getHighestElementContainer(ngContainer: TNode): TNode {
|
|
|
|
while (ngContainer.parent != null && ngContainer.parent.type === TNodeType.ElementContainer) {
|
|
|
|
ngContainer = ngContainer.parent;
|
|
|
|
}
|
|
|
|
return ngContainer;
|
|
|
|
}
|
|
|
|
|
2018-10-11 13:13:57 -07:00
|
|
|
export function getBeforeNodeForView(index: number, views: LViewData[], containerNative: RComment) {
|
2018-09-13 16:07:23 -07:00
|
|
|
if (index + 1 < views.length) {
|
|
|
|
const view = views[index + 1] as LViewData;
|
|
|
|
const viewTNode = view[HOST_NODE] as TViewNode;
|
2018-10-11 13:13:57 -07:00
|
|
|
return viewTNode.child ? getLNode(viewTNode.child, view).native : containerNative;
|
2018-09-13 16:07:23 -07:00
|
|
|
} else {
|
2018-10-11 13:13:57 -07:00
|
|
|
return containerNative;
|
2018-09-13 16:07:23 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-06-18 16:55:43 +02:00
|
|
|
/**
|
|
|
|
* Removes the `child` element of the `parent` from the DOM.
|
|
|
|
*
|
2018-09-13 16:07:23 -07:00
|
|
|
* @param parentEl The parent element from which to remove the child
|
2018-06-18 16:55:43 +02:00
|
|
|
* @param child The child that should be removed
|
|
|
|
* @param currentView The current LView
|
|
|
|
* @returns Whether or not the child was removed
|
|
|
|
*/
|
2018-09-13 16:07:23 -07:00
|
|
|
export function removeChild(tNode: TNode, child: RNode | null, currentView: LViewData): boolean {
|
|
|
|
const parentNative = getParentLNode(tNode, currentView) !.native as RElement;
|
|
|
|
if (child !== null && canInsertNativeNode(tNode, currentView)) {
|
2018-06-18 16:55:43 +02:00
|
|
|
// We only remove the element if not in View or not projected.
|
|
|
|
const renderer = currentView[RENDERER];
|
2018-09-13 16:07:23 -07:00
|
|
|
isProceduralRenderer(renderer) ? renderer.removeChild(parentNative as RElement, child) :
|
|
|
|
parentNative !.removeChild(child);
|
2018-06-18 16:55:43 +02:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return 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
|
|
|
*
|
2018-10-11 13:13:57 -07:00
|
|
|
* @param projectedTNode The TNode to be projected
|
|
|
|
* @param tProjectionNode The projection (ng-content) TNode
|
2018-01-08 20:17:13 -08:00
|
|
|
* @param currentView Current LView
|
2018-10-11 13:13:57 -07:00
|
|
|
* @param projectionView Projection view (view above current)
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-26 11:36:31 +01:00
|
|
|
export function appendProjectedNode(
|
2018-09-13 16:07:23 -07:00
|
|
|
projectedTNode: TNode, tProjectionNode: TNode, currentView: LViewData,
|
|
|
|
projectionView: LViewData): void {
|
2018-10-11 13:13:57 -07:00
|
|
|
const native = getLNode(projectedTNode, projectionView).native;
|
|
|
|
appendChild(native, tProjectionNode, currentView);
|
2018-08-22 16:57:40 -07:00
|
|
|
|
|
|
|
// the projected contents are processed while in the shadow view (which is the currentView)
|
|
|
|
// therefore we need to extract the view where the host element lives since it's the
|
|
|
|
// logical container of the content projected views
|
2018-10-11 13:13:57 -07:00
|
|
|
attachPatchData(native, projectionView);
|
2018-09-13 16:07:23 -07:00
|
|
|
|
|
|
|
const renderParent = getRenderParent(tProjectionNode, currentView);
|
2018-08-22 16:57:40 -07:00
|
|
|
|
2018-10-11 13:13:57 -07:00
|
|
|
const nodeOrContainer = projectionView[projectedTNode.index];
|
2018-09-13 16:07:23 -07:00
|
|
|
if (projectedTNode.type === TNodeType.Container) {
|
2018-06-27 16:05:23 -07:00
|
|
|
// The node we are adding is a container and we are adding it to an element which
|
2017-12-08 11:48:54 -08:00
|
|
|
// 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).
|
2018-06-27 16:05:23 -07:00
|
|
|
// Assign the final projection location in those cases.
|
2018-10-11 13:13:57 -07:00
|
|
|
nodeOrContainer[RENDER_PARENT] = renderParent;
|
|
|
|
const views = nodeOrContainer[VIEWS];
|
2017-12-01 14:23:03 -08:00
|
|
|
for (let i = 0; i < views.length; i++) {
|
2018-10-11 13:13:57 -07:00
|
|
|
addRemoveViewFromContainer(views[i], true, nodeOrContainer[NATIVE]);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
2018-10-11 13:13:57 -07:00
|
|
|
} else {
|
|
|
|
if (projectedTNode.type === TNodeType.ElementContainer) {
|
|
|
|
let ngContainerChildTNode: TNode|null = projectedTNode.child as TNode;
|
|
|
|
while (ngContainerChildTNode) {
|
|
|
|
appendProjectedNode(ngContainerChildTNode, tProjectionNode, currentView, projectionView);
|
|
|
|
ngContainerChildTNode = ngContainerChildTNode.next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isLContainer(nodeOrContainer)) {
|
|
|
|
nodeOrContainer[RENDER_PARENT] = renderParent;
|
|
|
|
appendChild(nodeOrContainer[NATIVE], tProjectionNode, currentView);
|
2018-08-07 14:42:40 +02:00
|
|
|
}
|
2018-04-09 17:35:50 +02:00
|
|
|
}
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|