fix(runtime-core): enable block tracking when normalizing plain element with slot children (#1987)

fix #1980
This commit is contained in:
HcySunYang 2020-09-02 00:38:47 +08:00 committed by GitHub
parent 706b52aadd
commit 5b82c48c7b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 71 additions and 12 deletions

View File

@ -11,9 +11,12 @@ import {
serializeInner as inner, serializeInner as inner,
VNode, VNode,
ref, ref,
nextTick nextTick,
defineComponent,
withCtx,
renderSlot
} from '@vue/runtime-test' } from '@vue/runtime-test'
import { PatchFlags } from '@vue/shared' import { PatchFlags, SlotFlags } from '@vue/shared'
describe('renderer: optimized mode', () => { describe('renderer: optimized mode', () => {
let root: TestElement let root: TestElement
@ -398,4 +401,52 @@ describe('renderer: optimized mode', () => {
expect(inner(root)).toBe('<div><i>bar</i></div>') expect(inner(root)).toBe('<div><i>bar</i></div>')
expect(block!.dynamicChildren).toBe(null) expect(block!.dynamicChildren).toBe(null)
}) })
// #1980
test('dynamicChildren should be tracked correctly when normalizing slots to plain children', async () => {
let block: VNode
const Comp = defineComponent({
setup(_props, { slots }) {
return () => {
const vnode = (openBlock(),
(block = createBlock('div', null, {
default: withCtx(() => [renderSlot(slots, 'default')]),
_: SlotFlags.FORWARDED
})))
return vnode
}
}
})
const foo = ref(0)
const App = {
setup() {
return () => {
return createVNode(Comp, null, {
default: withCtx(() => [
createVNode('p', null, foo.value, PatchFlags.TEXT)
]),
// Indicates that this is a stable slot to avoid bail out
_: SlotFlags.STABLE
})
}
}
}
render(h(App), root)
expect(inner(root)).toBe('<div><p>0</p></div>')
expect(block!.dynamicChildren!.length).toBe(1)
expect(block!.dynamicChildren![0].type).toBe(Fragment)
expect(block!.dynamicChildren![0].dynamicChildren!.length).toBe(1)
expect(
serialize(block!.dynamicChildren![0].dynamicChildren![0]
.el as TestElement)
).toBe('<p>0</p>')
foo.value++
await nextTick()
expect(inner(root)).toBe('<div><p>1</p></div>')
})
}) })

View File

@ -130,10 +130,10 @@ describe('vnode', () => {
}) })
test('object', () => { test('object', () => {
const vnode = createVNode('p', null, { foo: 'foo' }) const vnode = createVNode({}, null, { foo: 'foo' })
expect(vnode.children).toMatchObject({ foo: 'foo' }) expect(vnode.children).toMatchObject({ foo: 'foo' })
expect(vnode.shapeFlag).toBe( expect(vnode.shapeFlag).toBe(
ShapeFlags.ELEMENT | ShapeFlags.SLOTS_CHILDREN ShapeFlags.STATEFUL_COMPONENT | ShapeFlags.SLOTS_CHILDREN
) )
}) })

View File

@ -11,6 +11,8 @@ import { PatchFlags, SlotFlags } from '@vue/shared'
import { warn } from '../warning' import { warn } from '../warning'
export let isRenderingCompiledSlot = 0 export let isRenderingCompiledSlot = 0
export const setCompiledSlotRendering = (n: number) =>
(isRenderingCompiledSlot += n)
/** /**
* Compiler runtime helper for rendering `<slot/>` * Compiler runtime helper for rendering `<slot/>`

View File

@ -16,7 +16,7 @@ export function withCtx(
ctx: ComponentInternalInstance | null = currentRenderingInstance ctx: ComponentInternalInstance | null = currentRenderingInstance
) { ) {
if (!ctx) return fn if (!ctx) return fn
return function renderFnWithContext() { const renderFnWithContext = (...args: any[]) => {
// If a user calls a compiled slot inside a template expression (#1745), it // If a user calls a compiled slot inside a template expression (#1745), it
// can mess up block tracking, so by default we need to push a null block to // can mess up block tracking, so by default we need to push a null block to
// avoid that. This isn't necessary if rendering a compiled `<slot>`. // avoid that. This isn't necessary if rendering a compiled `<slot>`.
@ -25,11 +25,13 @@ export function withCtx(
} }
const owner = currentRenderingInstance const owner = currentRenderingInstance
setCurrentRenderingInstance(ctx) setCurrentRenderingInstance(ctx)
const res = fn.apply(null, arguments as any) const res = fn(...args)
setCurrentRenderingInstance(owner) setCurrentRenderingInstance(owner)
if (!isRenderingCompiledSlot) { if (!isRenderingCompiledSlot) {
closeBlock() closeBlock()
} }
return res return res
} }
renderFnWithContext._c = true
return renderFnWithContext
} }

View File

@ -36,6 +36,7 @@ import { currentRenderingInstance } from './componentRenderUtils'
import { RendererNode, RendererElement } from './renderer' import { RendererNode, RendererElement } from './renderer'
import { NULL_DYNAMIC_COMPONENT } from './helpers/resolveAssets' import { NULL_DYNAMIC_COMPONENT } from './helpers/resolveAssets'
import { hmrDirtyComponents } from './hmr' import { hmrDirtyComponents } from './hmr'
import { setCompiledSlotRendering } from './helpers/renderSlot'
export const Fragment = (Symbol(__DEV__ ? 'Fragment' : undefined) as any) as { export const Fragment = (Symbol(__DEV__ ? 'Fragment' : undefined) as any) as {
__isFragment: true __isFragment: true
@ -539,12 +540,15 @@ export function normalizeChildren(vnode: VNode, children: unknown) {
} else if (isArray(children)) { } else if (isArray(children)) {
type = ShapeFlags.ARRAY_CHILDREN type = ShapeFlags.ARRAY_CHILDREN
} else if (typeof children === 'object') { } else if (typeof children === 'object') {
// Normalize slot to plain children if (shapeFlag & ShapeFlags.ELEMENT || shapeFlag & ShapeFlags.TELEPORT) {
if ( // Normalize slot to plain children for plain element and Teleport
(shapeFlag & ShapeFlags.ELEMENT || shapeFlag & ShapeFlags.TELEPORT) && const slot = (children as any).default
(children as any).default if (slot) {
) { // _c marker is added by withCtx() indicating this is a compiled slot
normalizeChildren(vnode, (children as any).default()) slot._c && setCompiledSlotRendering(1)
normalizeChildren(vnode, slot())
slot._c && setCompiledSlotRendering(-1)
}
return return
} else { } else {
type = ShapeFlags.SLOTS_CHILDREN type = ShapeFlags.SLOTS_CHILDREN