perf: micro optimizations for vnode creation
This commit is contained in:
parent
40ccbdeaac
commit
8be578b6b6
@ -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,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user