test: basic test for suspense
This commit is contained in:
parent
e688e69b1f
commit
bd15138ef5
64
packages/runtime-core/__tests__/rendererSuspense.spec.ts
Normal file
64
packages/runtime-core/__tests__/rendererSuspense.spec.ts
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
import {
|
||||||
|
h,
|
||||||
|
ref,
|
||||||
|
Suspense,
|
||||||
|
ComponentOptions,
|
||||||
|
render,
|
||||||
|
nodeOps,
|
||||||
|
serializeInner,
|
||||||
|
nextTick
|
||||||
|
} from '@vue/runtime-test'
|
||||||
|
|
||||||
|
describe('renderer: suspense', () => {
|
||||||
|
it('should work', async () => {
|
||||||
|
const msg = ref('hello')
|
||||||
|
const deps: Promise<any>[] = []
|
||||||
|
|
||||||
|
const createAsyncComponent = (loader: () => Promise<ComponentOptions>) => ({
|
||||||
|
async setup(props: any, { slots }: any) {
|
||||||
|
const p = loader()
|
||||||
|
deps.push(p)
|
||||||
|
const Inner = await p
|
||||||
|
return () => h(Inner, props, slots)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
const AsyncChild = createAsyncComponent(
|
||||||
|
() =>
|
||||||
|
new Promise(resolve => {
|
||||||
|
setTimeout(() => {
|
||||||
|
resolve({
|
||||||
|
setup(props: { msg: string }) {
|
||||||
|
return () => h('div', props.msg)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
}, 0)
|
||||||
|
})
|
||||||
|
)
|
||||||
|
|
||||||
|
const Mid = {
|
||||||
|
setup() {
|
||||||
|
return () =>
|
||||||
|
h(AsyncChild, {
|
||||||
|
msg: msg.value
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const Comp = {
|
||||||
|
name: 'root',
|
||||||
|
setup() {
|
||||||
|
// TODO test fallback
|
||||||
|
return () => h(Suspense, [msg.value, h(Mid)])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const root = nodeOps.createElement('div')
|
||||||
|
render(h(Comp), root)
|
||||||
|
expect(serializeInner(root)).toBe(`<!---->`)
|
||||||
|
|
||||||
|
await Promise.all(deps)
|
||||||
|
await nextTick()
|
||||||
|
expect(serializeInner(root)).toBe(`<!---->hello<div>hello</div><!---->`)
|
||||||
|
})
|
||||||
|
})
|
@ -634,10 +634,10 @@ export function createRenderer<
|
|||||||
})
|
})
|
||||||
|
|
||||||
suspense.onResolve(() => {
|
suspense.onResolve(() => {
|
||||||
|
// unmount fallback tree
|
||||||
|
unmount(suspense.fallbackTree as HostVNode, parentComponent, true)
|
||||||
// move content from off-dom container to actual container
|
// move content from off-dom container to actual container
|
||||||
;(suspense.subTree as any).children.forEach((vnode: HostVNode) => {
|
move(suspense.subTree as HostVNode, container, anchor)
|
||||||
move(vnode, container, anchor)
|
|
||||||
})
|
|
||||||
suspense.vnode.el = (suspense.subTree as HostVNode).el
|
suspense.vnode.el = (suspense.subTree as HostVNode).el
|
||||||
// check if there is a pending parent suspense
|
// check if there is a pending parent suspense
|
||||||
let parent = suspense.parent
|
let parent = suspense.parent
|
||||||
@ -682,7 +682,12 @@ export function createRenderer<
|
|||||||
// now check if we have encountered any async deps
|
// now check if we have encountered any async deps
|
||||||
if (suspense.deps > 0) {
|
if (suspense.deps > 0) {
|
||||||
// TODO mount the fallback tree.
|
// TODO mount the fallback tree.
|
||||||
console.log('fallback')
|
processEmptyNode(
|
||||||
|
null,
|
||||||
|
(suspense.fallbackTree = createVNode(Empty)),
|
||||||
|
container,
|
||||||
|
anchor
|
||||||
|
)
|
||||||
} else {
|
} else {
|
||||||
suspense.resolve()
|
suspense.resolve()
|
||||||
}
|
}
|
||||||
@ -815,6 +820,10 @@ export function createRenderer<
|
|||||||
suspense.deps--
|
suspense.deps--
|
||||||
suspense.retry()
|
suspense.retry()
|
||||||
})
|
})
|
||||||
|
// give it a placeholder
|
||||||
|
const placeholder = (instance.subTree = createVNode(Empty))
|
||||||
|
processEmptyNode(null, placeholder, container, anchor)
|
||||||
|
initialVNode.el = placeholder.el
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
x
Reference in New Issue
Block a user