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
|
|
|
|
*/
|
|
|
|
|
|
|
|
import {assertNotNull} from './assert';
|
2018-01-11 13:09:21 -08:00
|
|
|
import {LContainer, unusedValueExportToPlacateAjd as unused1} from './interfaces/container';
|
|
|
|
import {LContainerNode, LElementNode, LNode, LNodeFlags, LProjectionNode, LTextNode, LViewNode, unusedValueExportToPlacateAjd as unused2} from './interfaces/node';
|
|
|
|
import {LProjection, unusedValueExportToPlacateAjd as unused3} from './interfaces/projection';
|
|
|
|
import {ProceduralRenderer3, RComment, RElement, RNode, RText, unusedValueExportToPlacateAjd as unused4} from './interfaces/renderer';
|
|
|
|
import {LView, LViewOrLContainer, unusedValueExportToPlacateAjd as unused5} from './interfaces/view';
|
2017-12-01 14:23:03 -08:00
|
|
|
import {assertNodeType} from './node_assert';
|
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
|
|
|
/**
|
|
|
|
* Finds the closest DOM node above a given container in the hierarchy.
|
|
|
|
*
|
|
|
|
* This is necessary to add or remove elements from the DOM when a view
|
|
|
|
* is added or removed from the container. e.g. parent.removeChild(...)
|
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param containerNode The container node whose parent must be found
|
|
|
|
* @returns Closest DOM node above the container
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function findNativeParent(containerNode: LContainerNode): RNode|null {
|
|
|
|
let container: LContainerNode|null = containerNode;
|
2017-12-01 14:23:03 -08:00
|
|
|
while (container) {
|
|
|
|
ngDevMode && assertNodeType(container, LNodeFlags.Container);
|
|
|
|
const renderParent = container.data.renderParent;
|
|
|
|
if (renderParent !== null) {
|
|
|
|
return renderParent.native;
|
|
|
|
}
|
2018-01-08 20:17:13 -08:00
|
|
|
const viewOrElement: LViewNode|LElementNode = container.parent !;
|
2017-12-01 14:23:03 -08:00
|
|
|
ngDevMode && assertNotNull(viewOrElement, 'container.parent');
|
|
|
|
if ((viewOrElement.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Element) {
|
|
|
|
// we are an LElement, which means we are past the last LContainer.
|
|
|
|
// This means than we have not been projected so just ignore.
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
ngDevMode && assertNodeType(viewOrElement, LNodeFlags.View);
|
2018-01-08 20:17:13 -08:00
|
|
|
container = (viewOrElement as LViewNode).parent;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Finds the DOM element before which a certain view should be inserting its
|
|
|
|
* child elements.
|
|
|
|
*
|
|
|
|
* If the view has a next (e.g. for loop), elements should be inserted before
|
|
|
|
* the next view's first child element. Otherwise, the container's comment
|
|
|
|
* anchor is the marker.
|
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param index The index of the view to check
|
2018-01-08 20:17:13 -08:00
|
|
|
* @param lContainer parent LContainer
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param native Comment anchor for container
|
|
|
|
* @returns The DOM element for which the view should insert elements
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
function findBeforeNode(index: number, lContainer: LContainer, native: RNode): RNode {
|
|
|
|
const views = lContainer.views;
|
2017-12-01 14:23:03 -08:00
|
|
|
// Find the node to insert in front of
|
2017-12-13 19:34:46 -08:00
|
|
|
return index + 1 < views.length ?
|
2018-01-08 20:17:13 -08:00
|
|
|
(views[index + 1].child as LTextNode | LElementNode | LContainerNode).native :
|
2017-12-01 14:23:03 -08:00
|
|
|
native;
|
|
|
|
}
|
|
|
|
|
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 {
|
2017-12-01 14:23:03 -08:00
|
|
|
ngDevMode && assertNodeType(container, LNodeFlags.Container);
|
|
|
|
ngDevMode && assertNodeType(rootNode, LNodeFlags.View);
|
|
|
|
const parent = findNativeParent(container);
|
2017-12-08 11:48:54 -08:00
|
|
|
let node: LNode|null = rootNode.child;
|
2017-12-01 14:23:03 -08:00
|
|
|
if (parent) {
|
|
|
|
while (node) {
|
|
|
|
const type = node.flags & LNodeFlags.TYPE_MASK;
|
|
|
|
let nextNode: LNode|null = null;
|
|
|
|
const renderer = container.view.renderer;
|
2017-12-14 15:50:01 -08:00
|
|
|
const isFnRenderer = (renderer as ProceduralRenderer3).listen;
|
2017-12-01 14:23:03 -08:00
|
|
|
if (type === LNodeFlags.Element) {
|
2017-12-14 15:03:46 -08:00
|
|
|
insertMode ? (isFnRenderer ?
|
|
|
|
(renderer as ProceduralRenderer3)
|
|
|
|
.insertBefore !(parent, node.native !, beforeNode as RNode | null) :
|
|
|
|
parent.insertBefore(node.native !, beforeNode as RNode | null, true)) :
|
|
|
|
(isFnRenderer ?
|
|
|
|
(renderer as ProceduralRenderer3)
|
|
|
|
.removeChild !(parent as RElement, node.native !) :
|
|
|
|
parent.removeChild(node.native !));
|
2017-12-01 14:23:03 -08:00
|
|
|
nextNode = node.next;
|
|
|
|
} else if (type === LNodeFlags.Container) {
|
|
|
|
// 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;
|
2017-12-14 15:03:46 -08:00
|
|
|
insertMode ? (isFnRenderer ?
|
|
|
|
(renderer as ProceduralRenderer3)
|
|
|
|
.appendChild !(parent as RElement, node.native !) :
|
|
|
|
parent.appendChild(node.native !)) :
|
|
|
|
(isFnRenderer ?
|
|
|
|
(renderer as ProceduralRenderer3)
|
|
|
|
.removeChild !(parent as RElement, node.native !) :
|
|
|
|
parent.removeChild(node.native !));
|
2017-12-13 19:34:46 -08:00
|
|
|
nextNode = childContainerData.views.length ? childContainerData.views[0].child : null;
|
2017-12-01 14:23:03 -08:00
|
|
|
} else if (type === LNodeFlags.Projection) {
|
2018-01-08 20:17:13 -08:00
|
|
|
nextNode = (node as LProjectionNode).data[0];
|
2017-12-01 14:23:03 -08:00
|
|
|
} else {
|
2018-01-08 20:17:13 -08:00
|
|
|
nextNode = (node as LViewNode).child;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
if (nextNode === null) {
|
|
|
|
while (node && !node.next) {
|
|
|
|
node = node.parent;
|
|
|
|
if (node === rootNode) node = null;
|
|
|
|
}
|
|
|
|
node = node && node.next;
|
|
|
|
} else {
|
|
|
|
node = nextNode;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-12-13 11:04:46 -08:00
|
|
|
* Traverses the tree of component views and containers to remove listeners and
|
|
|
|
* 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 {
|
|
|
|
let viewOrContainer: LViewOrLContainer|null = rootView;
|
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) {
|
|
|
|
cleanUpView(viewOrContainer as LView);
|
|
|
|
next = viewOrContainer.next;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (next == null) {
|
2018-01-08 20:17:13 -08:00
|
|
|
while (viewOrContainer && !viewOrContainer !.next) {
|
|
|
|
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
|
|
|
|
* @param newView The view to insert
|
|
|
|
* @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(
|
|
|
|
container: LContainerNode, newView: 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.
|
2017-12-13 19:34:46 -08:00
|
|
|
setViewNext(views[index - 1], newView);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2017-12-13 19:34:46 -08:00
|
|
|
if (index < views.length && views[index].data.id !== newView.data.id) {
|
2017-12-01 14:23:03 -08:00
|
|
|
// View ID change replace the view.
|
2017-12-13 19:34:46 -08:00
|
|
|
setViewNext(newView, views[index]);
|
|
|
|
views.splice(index, 0, newView);
|
|
|
|
} else if (index >= views.length) {
|
|
|
|
views.push(newView);
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
if (state.nextIndex <= index) {
|
|
|
|
state.nextIndex++;
|
|
|
|
}
|
|
|
|
|
|
|
|
// 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) {
|
|
|
|
addRemoveViewFromContainer(
|
|
|
|
container, newView, true, findBeforeNode(index, state, container.native));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Notify query that view has been inserted
|
2017-12-13 19:34:46 -08:00
|
|
|
container.query && container.query.insertView(container, newView, index);
|
2017-12-01 14:23:03 -08:00
|
|
|
return newView;
|
|
|
|
}
|
|
|
|
|
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) {
|
2017-12-13 19:34:46 -08:00
|
|
|
setViewNext(views[removeIndex - 1], viewNode.next);
|
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
|
2017-12-13 19:34:46 -08:00
|
|
|
container.query && container.query.removeView(container, viewNode, removeIndex);
|
2017-12-01 14:23:03 -08:00
|
|
|
return viewNode;
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Sets a next on the view node, so views in for loops can easily jump from
|
2018-01-08 20:17:13 -08:00
|
|
|
* one view to the next to add/remove elements. Also adds the LView (view.data)
|
2017-12-13 11:04:46 -08:00
|
|
|
* to the view tree for easy traversal when cleaning up the view.
|
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param view The view to set up
|
|
|
|
* @param next The view's new next
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function setViewNext(view: LViewNode, next: LViewNode | null): void {
|
2017-12-01 14:23:03 -08:00
|
|
|
view.next = next;
|
|
|
|
view.data.next = next ? next.data : null;
|
|
|
|
}
|
|
|
|
|
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-01-08 20:17:13 -08:00
|
|
|
if ((node = (state as LView) !.node) && (node.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.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
|
|
|
|
return node.parent !.data as any;
|
|
|
|
} 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 {
|
|
|
|
if (!view.cleanup) return;
|
|
|
|
const cleanup = view.cleanup !;
|
2017-12-01 14:23:03 -08:00
|
|
|
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]);
|
|
|
|
}
|
|
|
|
}
|
2018-01-08 20:17:13 -08:00
|
|
|
view.cleanup = null;
|
2017-12-01 14:23:03 -08:00
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Appends the provided child element to the provided parent, if appropriate.
|
|
|
|
*
|
|
|
|
* If the parent is a view, the element will be appended as part of viewEnd(), so
|
|
|
|
* the element should not be appended now. Similarly, if the child is a content child
|
|
|
|
* of a parent component, the child will be appended to the right position later by
|
|
|
|
* the content projection system. Otherwise, append normally.
|
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param parent The parent to which to append the child
|
|
|
|
* @param child The child that should be appended
|
2018-01-08 20:17:13 -08:00
|
|
|
* @param currentView The current LView
|
2017-12-13 16:32:21 -08:00
|
|
|
* @returns Whether or not the child was appended
|
2017-12-13 11:04:46 -08:00
|
|
|
*/
|
2018-01-08 20:17:13 -08:00
|
|
|
export function appendChild(parent: LNode, child: RNode | null, currentView: LView): boolean {
|
2017-12-01 14:23:03 -08:00
|
|
|
// Only add native child element to parent element if the parent element is regular Element.
|
|
|
|
// If parent is:
|
|
|
|
// - Regular element => add child
|
|
|
|
// - Component host element =>
|
|
|
|
// - Current View, and parent view same => content => don't add -> parent component will
|
|
|
|
// re-project if needed.
|
|
|
|
// - Current View, and parent view different => view => add Child
|
|
|
|
// - View element => View's get added separately.
|
|
|
|
if (child !== null && (parent.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Element &&
|
|
|
|
(parent.view !==
|
|
|
|
currentView /* Crossing View Boundaries, it is Component, but add Element of View */
|
|
|
|
|| parent.data === null /* Regular Element. */)) {
|
|
|
|
// We only add element if not in View or not projected.
|
|
|
|
|
|
|
|
const renderer = currentView.renderer;
|
2017-12-14 15:50:01 -08:00
|
|
|
(renderer as ProceduralRenderer3).listen ?
|
|
|
|
(renderer as ProceduralRenderer3).appendChild !(parent.native !as RElement, child) :
|
2017-12-01 14:23:03 -08:00
|
|
|
parent.native !.appendChild(child);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-12-13 11:04:46 -08:00
|
|
|
/**
|
|
|
|
* Inserts the provided node before the correct element in the DOM, if appropriate.
|
|
|
|
*
|
|
|
|
* If the parent is a view, the element will be inserted as part of viewEnd(), so
|
|
|
|
* the element should not be inserted now. Similarly, if the child is a content child
|
|
|
|
* of a parent component, the child will be inserted to the right position later by
|
|
|
|
* the content projection system. Otherwise, insertBefore normally.
|
|
|
|
*
|
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 {
|
2017-12-01 14:23:03 -08:00
|
|
|
const parent = node.parent !;
|
|
|
|
// Only add child element to parent element if the parent element is regular Element.
|
|
|
|
// If parent is:
|
|
|
|
// - Normal element => add child
|
|
|
|
// - Component element =>
|
|
|
|
// - Current View, and parent view same => content don't add -> parent component will
|
|
|
|
// re-project if needed.
|
|
|
|
// - Current View, and parent view different => view => add Child
|
|
|
|
// - View element => View's get added separately.
|
|
|
|
if ((parent.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Element &&
|
|
|
|
(parent.view !==
|
|
|
|
currentView /* Crossing View Boundaries, its Component, but add Element of View */
|
|
|
|
|| parent.data === null /* Regular Element. */)) {
|
|
|
|
// We only add element if not in View or not projected.
|
|
|
|
|
|
|
|
let sibling = node.next;
|
|
|
|
let nativeSibling: RNode|null = null;
|
|
|
|
while (sibling && (nativeSibling = sibling.native) === null) {
|
|
|
|
sibling = sibling.next;
|
|
|
|
}
|
|
|
|
const renderer = currentView.renderer;
|
2017-12-14 15:50:01 -08:00
|
|
|
(renderer as ProceduralRenderer3).listen ?
|
2017-12-14 15:03:46 -08:00
|
|
|
(renderer as ProceduralRenderer3)
|
|
|
|
.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,
|
|
|
|
* appends the nodes from all of the container's active views to the DOM. Also stores the
|
|
|
|
* node in the given projectedNodes array.
|
|
|
|
*
|
2017-12-13 16:32:21 -08:00
|
|
|
* @param projectedNodes Array to store the projected node
|
|
|
|
* @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
|
|
|
*/
|
2017-12-01 14:23:03 -08:00
|
|
|
export function processProjectedNode(
|
2018-01-08 20:17:13 -08:00
|
|
|
projectedNodes: LProjection, node: LElementNode | LTextNode | LContainerNode,
|
|
|
|
currentParent: LViewNode | LElementNode, currentView: LView): void {
|
2017-12-01 14:23:03 -08:00
|
|
|
if ((node.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Container &&
|
|
|
|
(currentParent.flags & LNodeFlags.TYPE_MASK) === LNodeFlags.Element &&
|
2017-12-08 11:48:54 -08:00
|
|
|
(currentParent.data === null || currentParent.data === currentView)) {
|
|
|
|
// 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;
|
|
|
|
lContainer.renderParent = currentParent as LElementNode;
|
|
|
|
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
|
|
|
}
|
|
|
|
}
|
|
|
|
projectedNodes.push(node);
|
|
|
|
appendChild(currentParent, node.native, currentView);
|
|
|
|
}
|