perf(ivy): remove firstTemplatePass check for component view creation (#31946)

PR Close #31946
This commit is contained in:
Pawel Kozlowski 2019-08-01 14:00:23 +02:00 committed by Alex Rickabaugh
parent f2d47c96c4
commit a7c71d1a57
6 changed files with 33 additions and 37 deletions

View File

@ -70,6 +70,7 @@ export function assertLView(value: any) {
assertEqual(isLView(value), true, 'Expecting LView'); assertEqual(isLView(value), true, 'Expecting LView');
} }
export function assertFirstTemplatePass(tView: TView, errMessage: string) { export function assertFirstTemplatePass(tView: TView, errMessage?: string) {
assertEqual(tView.firstTemplatePass, true, errMessage); assertEqual(
tView.firstTemplatePass, true, errMessage || 'Should only be called in first template pass.');
} }

View File

@ -17,9 +17,9 @@ import {assertComponentType} from './assert';
import {getComponentDef} from './definition'; import {getComponentDef} from './definition';
import {diPublicInInjector, getOrCreateNodeInjectorForNode} from './di'; import {diPublicInInjector, getOrCreateNodeInjectorForNode} from './di';
import {registerPostOrderHooks, registerPreOrderHooks} from './hooks'; import {registerPostOrderHooks, registerPreOrderHooks} from './hooks';
import {CLEAN_PROMISE, addToViewTree, createLView, createTView, getOrCreateTNode, getOrCreateTView, initNodeFlags, instantiateRootComponent, invokeHostBindingsInCreationMode, locateHostElement, queueComponentIndexForCheck, refreshDescendantViews} from './instructions/shared'; import {CLEAN_PROMISE, addToViewTree, createLView, createTView, getOrCreateTNode, getOrCreateTView, initNodeFlags, instantiateRootComponent, invokeHostBindingsInCreationMode, locateHostElement, markAsComponentHost, refreshDescendantViews} from './instructions/shared';
import {ComponentDef, ComponentType, RenderFlags} from './interfaces/definition'; import {ComponentDef, ComponentType, RenderFlags} from './interfaces/definition';
import {TElementNode, TNode, TNodeFlags, TNodeType} from './interfaces/node'; import {TElementNode, TNode, TNodeType} from './interfaces/node';
import {PlayerHandler} from './interfaces/player'; import {PlayerHandler} from './interfaces/player';
import {RElement, Renderer3, RendererFactory3, domRendererFactory3} from './interfaces/renderer'; import {RElement, Renderer3, RendererFactory3, domRendererFactory3} from './interfaces/renderer';
import {CONTEXT, FLAGS, HEADER_OFFSET, LView, LViewFlags, RootContext, RootContextFlags, TVIEW} from './interfaces/view'; import {CONTEXT, FLAGS, HEADER_OFFSET, LView, LViewFlags, RootContext, RootContextFlags, TVIEW} from './interfaces/view';
@ -184,9 +184,8 @@ export function createRootComponentView(
if (tView.firstTemplatePass) { if (tView.firstTemplatePass) {
diPublicInInjector(getOrCreateNodeInjectorForNode(tNode, rootView), tView, def.type); diPublicInInjector(getOrCreateNodeInjectorForNode(tNode, rootView), tView, def.type);
tNode.flags = TNodeFlags.isComponent; markAsComponentHost(tView, tNode);
initNodeFlags(tNode, rootView.length, 1); initNodeFlags(tNode, rootView.length, 1);
queueComponentIndexForCheck(tNode);
} }
// Store component view at node index, with node as the HOST // Store component view at node index, with node as the HOST

View File

@ -14,7 +14,7 @@ import {Sanitizer} from '../../sanitization/security';
import {assertDataInRange, assertDefined, assertDomNode, assertEqual, assertGreaterThan, assertNotEqual, assertNotSame} from '../../util/assert'; import {assertDataInRange, assertDefined, assertDomNode, assertEqual, assertGreaterThan, assertNotEqual, assertNotSame} from '../../util/assert';
import {createNamedArrayType} from '../../util/named_array_type'; import {createNamedArrayType} from '../../util/named_array_type';
import {normalizeDebugBindingName, normalizeDebugBindingValue} from '../../util/ng_reflect'; import {normalizeDebugBindingName, normalizeDebugBindingValue} from '../../util/ng_reflect';
import {assertLView, assertPreviousIsParent} from '../assert'; import {assertFirstTemplatePass, assertLView, assertPreviousIsParent} from '../assert';
import {attachPatchData, getComponentViewByInstance} from '../context_discovery'; import {attachPatchData, getComponentViewByInstance} from '../context_discovery';
import {diPublicInInjector, getNodeInjectable, getOrCreateNodeInjectorForNode} from '../di'; import {diPublicInInjector, getNodeInjectable, getOrCreateNodeInjectorForNode} from '../di';
import {throwMultipleComponentError} from '../errors'; import {throwMultipleComponentError} from '../errors';
@ -1207,7 +1207,7 @@ function postProcessBaseDirective<T>(
function findDirectiveMatches( function findDirectiveMatches(
tView: TView, viewData: LView, tView: TView, viewData: LView,
tNode: TElementNode | TContainerNode | TElementContainerNode): DirectiveDef<any>[]|null { tNode: TElementNode | TContainerNode | TElementContainerNode): DirectiveDef<any>[]|null {
ngDevMode && assertEqual(tView.firstTemplatePass, true, 'should run on first template pass only'); ngDevMode && assertFirstTemplatePass(tView);
const registry = tView.directiveRegistry; const registry = tView.directiveRegistry;
let matches: any[]|null = null; let matches: any[]|null = null;
if (registry) { if (registry) {
@ -1219,8 +1219,7 @@ function findDirectiveMatches(
if (isComponentDef(def)) { if (isComponentDef(def)) {
if (tNode.flags & TNodeFlags.isComponent) throwMultipleComponentError(tNode); if (tNode.flags & TNodeFlags.isComponent) throwMultipleComponentError(tNode);
tNode.flags = TNodeFlags.isComponent; markAsComponentHost(tView, tNode);
// The component is always stored first with directives after. // The component is always stored first with directives after.
matches.unshift(def); matches.unshift(def);
} else { } else {
@ -1232,13 +1231,16 @@ function findDirectiveMatches(
return matches; return matches;
} }
/** Stores index of component's host element so it will be queued for view refresh during CD. */ /**
export function queueComponentIndexForCheck(previousOrParentTNode: TNode): void { * Marks a given TNode as a component's host. This consists of:
const tView = getLView()[TVIEW]; * - setting appropriate TNode flags;
ngDevMode && * - storing index of component's host element so it will be queued for view refresh during CD.
assertEqual(tView.firstTemplatePass, true, 'Should only be called in first template pass.'); */
export function markAsComponentHost(tView: TView, hostTNode: TNode): void {
ngDevMode && assertFirstTemplatePass(tView);
hostTNode.flags = TNodeFlags.isComponent;
(tView.components || (tView.components = ngDevMode ? new TViewComponents !() : [ (tView.components || (tView.components = ngDevMode ? new TViewComponents !() : [
])).push(previousOrParentTNode.index); ])).push(hostTNode.index);
} }
@ -1306,10 +1308,8 @@ function baseResolveDirective<T>(
viewData.push(nodeInjectorFactory); viewData.push(nodeInjectorFactory);
} }
function addComponentLogic<T>( function addComponentLogic<T>(lView: LView, hostTNode: TNode, def: ComponentDef<T>): void {
lView: LView, previousOrParentTNode: TNode, def: ComponentDef<T>): void { const native = getNativeByTNode(hostTNode, lView);
const native = getNativeByTNode(previousOrParentTNode, lView);
const tView = getOrCreateTView(def); const tView = getOrCreateTView(def);
// Only component views should be added to the view tree directly. Embedded views are // Only component views should be added to the view tree directly. Embedded views are
@ -1318,18 +1318,14 @@ function addComponentLogic<T>(
const componentView = addToViewTree( const componentView = addToViewTree(
lView, createLView( lView, createLView(
lView, tView, null, def.onPush ? LViewFlags.Dirty : LViewFlags.CheckAlways, lView, tView, null, def.onPush ? LViewFlags.Dirty : LViewFlags.CheckAlways,
lView[previousOrParentTNode.index], previousOrParentTNode as TElementNode, lView[hostTNode.index], hostTNode as TElementNode, rendererFactory,
rendererFactory, rendererFactory.createRenderer(native as RElement, def))); rendererFactory.createRenderer(native as RElement, def)));
componentView[T_HOST] = previousOrParentTNode as TElementNode; componentView[T_HOST] = hostTNode as TElementNode;
// Component view will always be created before any injected LContainers, // Component view will always be created before any injected LContainers,
// so this is a regular element, wrap it with the component view // so this is a regular element, wrap it with the component view
lView[previousOrParentTNode.index] = componentView; lView[hostTNode.index] = componentView;
if (lView[TVIEW].firstTemplatePass) {
queueComponentIndexForCheck(previousOrParentTNode);
}
} }
export function elementAttributeInternal( export function elementAttributeInternal(

View File

@ -515,6 +515,9 @@
{ {
"name": "locateHostElement" "name": "locateHostElement"
}, },
{
"name": "markAsComponentHost"
},
{ {
"name": "matchTemplateAttribute" "name": "matchTemplateAttribute"
}, },
@ -545,9 +548,6 @@
{ {
"name": "postProcessDirective" "name": "postProcessDirective"
}, },
{
"name": "queueComponentIndexForCheck"
},
{ {
"name": "readPatchedData" "name": "readPatchedData"
}, },

View File

@ -386,6 +386,9 @@
{ {
"name": "locateHostElement" "name": "locateHostElement"
}, },
{
"name": "markAsComponentHost"
},
{ {
"name": "namespaceHTMLInternal" "name": "namespaceHTMLInternal"
}, },
@ -410,9 +413,6 @@
{ {
"name": "postProcessBaseDirective" "name": "postProcessBaseDirective"
}, },
{
"name": "queueComponentIndexForCheck"
},
{ {
"name": "readPatchedData" "name": "readPatchedData"
}, },

View File

@ -1118,6 +1118,9 @@
{ {
"name": "makeParamDecorator" "name": "makeParamDecorator"
}, },
{
"name": "markAsComponentHost"
},
{ {
"name": "markContextToPersistState" "name": "markContextToPersistState"
}, },
@ -1178,9 +1181,6 @@
{ {
"name": "postProcessDirective" "name": "postProcessDirective"
}, },
{
"name": "queueComponentIndexForCheck"
},
{ {
"name": "readPatchedData" "name": "readPatchedData"
}, },