perf: micro optimizations for vnode creation

This commit is contained in:
Evan You 2019-10-21 11:30:45 -04:00
parent 40ccbdeaac
commit 8be578b6b6

View File

@ -88,6 +88,7 @@ export interface VNode<HostNode = any, HostElement = any> {
// structure would be stable. This allows us to skip most children diffing // structure would be stable. This allows us to skip most children diffing
// and only worry about the dynamic nodes (indicated by patch flags). // and only worry about the dynamic nodes (indicated by patch flags).
const blockStack: (VNode[] | null)[] = [] const blockStack: (VNode[] | null)[] = []
let currentBlock: VNode[] | null = null
// Open a block. // Open a block.
// This must be called before `createBlock`. It cannot be part of `createBlock` // This must be called before `createBlock`. It cannot be part of `createBlock`
@ -101,7 +102,7 @@ const blockStack: (VNode[] | null)[] = []
// disableTracking is true when creating a fragment block, since a fragment // disableTracking is true when creating a fragment block, since a fragment
// always diffs its children. // always diffs its children.
export function openBlock(disableTracking?: boolean) { export function openBlock(disableTracking?: boolean) {
blockStack.push(disableTracking ? null : []) blockStack.push((currentBlock = disableTracking ? null : []))
} }
let shouldTrack = true let shouldTrack = true
@ -116,15 +117,20 @@ export function createBlock(
patchFlag?: number, patchFlag?: number,
dynamicProps?: string[] dynamicProps?: string[]
): VNode { ): VNode {
// avoid a block with optFlag tracking itself // avoid a block with patchFlag tracking itself
shouldTrack = false shouldTrack = false
const vnode = createVNode(type, props, children, patchFlag, dynamicProps) const vnode = createVNode(type, props, children, patchFlag, dynamicProps)
shouldTrack = true shouldTrack = true
const trackedNodes = blockStack.pop() // save current block children on the block vnode
vnode.dynamicChildren = vnode.dynamicChildren = currentBlock || EMPTY_ARR
trackedNodes && trackedNodes.length ? trackedNodes : EMPTY_ARR // close block
// a block is always going to be patched blockStack.pop()
trackDynamicNode(vnode) currentBlock = blockStack[blockStack.length - 1] || null
// a block is always going to be patched, so track it as a child of its
// parent block
if (currentBlock !== null) {
currentBlock.push(vnode)
}
return vnode return vnode
} }
@ -145,10 +151,10 @@ export function createVNode(
if (isReactive(props) || SetupProxySymbol in props) { if (isReactive(props) || SetupProxySymbol in props) {
props = extend({}, props) props = extend({}, props)
} }
if (props.class != null) { let { class: klass, style } = props
props.class = normalizeClass(props.class) if (klass != null && !isString(klass)) {
props.class = normalizeClass(klass)
} }
let { style } = props
if (style != null) { if (style != null) {
// reactive state objects need to be cloned since they are likely to be // reactive state objects need to be cloned since they are likely to be
// mutated // mutated
@ -172,8 +178,8 @@ export function createVNode(
_isVNode: true, _isVNode: true,
type, type,
props, props,
key: (props && props.key) || null, key: (props !== null && props.key) || null,
ref: (props && props.ref) || null, ref: (props !== null && props.ref) || null,
children: null, children: null,
component: null, component: null,
suspense: null, suspense: null,
@ -195,23 +201,17 @@ export function createVNode(
// the next vnode so that it can be properly unmounted later. // the next vnode so that it can be properly unmounted later.
if ( if (
shouldTrack && shouldTrack &&
(patchFlag || currentBlock !== null &&
(patchFlag > 0 ||
shapeFlag & ShapeFlags.STATEFUL_COMPONENT || shapeFlag & ShapeFlags.STATEFUL_COMPONENT ||
shapeFlag & ShapeFlags.FUNCTIONAL_COMPONENT) shapeFlag & ShapeFlags.FUNCTIONAL_COMPONENT)
) { ) {
trackDynamicNode(vnode) currentBlock.push(vnode)
} }
return vnode return vnode
} }
function trackDynamicNode(vnode: VNode) {
const currentBlockDynamicNodes = blockStack[blockStack.length - 1]
if (currentBlockDynamicNodes != null) {
currentBlockDynamicNodes.push(vnode)
}
}
export function cloneVNode(vnode: VNode): VNode { export function cloneVNode(vnode: VNode): VNode {
return { return {
_isVNode: true, _isVNode: true,