2020-03-19 22:54:03 +08:00
|
|
|
import { h, transformHArgs, resetTransformHArgs } from '../src/h'
|
2019-08-30 04:47:00 +08:00
|
|
|
import { createVNode } from '../src/vnode'
|
2020-03-19 22:54:03 +08:00
|
|
|
import { ComponentInternalInstance } from '@vue/runtime-core'
|
|
|
|
import { createApp } from '@vue/runtime-dom'
|
2019-08-30 04:47:00 +08:00
|
|
|
|
|
|
|
// Since h is a thin layer on top of createVNode, we are only testing its
|
|
|
|
// own logic here. Details of vnode creation is tested in vnode.spec.ts.
|
2020-03-19 22:54:03 +08:00
|
|
|
const testH = () => {
|
2019-08-30 04:47:00 +08:00
|
|
|
test('type only', () => {
|
|
|
|
expect(h('div')).toMatchObject(createVNode('div'))
|
|
|
|
})
|
|
|
|
|
|
|
|
test('type + props', () => {
|
|
|
|
expect(h('div', { id: 'foo' })).toMatchObject(
|
|
|
|
createVNode('div', { id: 'foo' })
|
|
|
|
)
|
|
|
|
})
|
|
|
|
|
|
|
|
test('type + omit props', () => {
|
|
|
|
// array
|
|
|
|
expect(h('div', ['foo'])).toMatchObject(createVNode('div', null, ['foo']))
|
|
|
|
// default slot
|
2019-10-10 22:17:16 +08:00
|
|
|
const Component = { template: '<br />' }
|
2019-08-30 04:47:00 +08:00
|
|
|
const slot = () => {}
|
2019-10-10 22:17:16 +08:00
|
|
|
expect(h(Component, slot)).toMatchObject(createVNode(Component, null, slot))
|
|
|
|
// single vnode
|
|
|
|
const vnode = h('div')
|
|
|
|
expect(h('div', vnode)).toMatchObject(createVNode('div', null, [vnode]))
|
2019-08-30 04:47:00 +08:00
|
|
|
// text
|
|
|
|
expect(h('div', 'foo')).toMatchObject(createVNode('div', null, 'foo'))
|
|
|
|
})
|
|
|
|
|
|
|
|
test('type + props + children', () => {
|
|
|
|
// array
|
|
|
|
expect(h('div', {}, ['foo'])).toMatchObject(createVNode('div', {}, ['foo']))
|
|
|
|
// default slot
|
2019-10-10 22:17:16 +08:00
|
|
|
const Component = { template: '<br />' }
|
2019-08-30 04:47:00 +08:00
|
|
|
const slot = () => {}
|
2019-10-10 22:17:16 +08:00
|
|
|
expect(h(Component, {}, slot)).toMatchObject(
|
|
|
|
createVNode(Component, {}, slot)
|
|
|
|
)
|
|
|
|
// single vnode
|
|
|
|
const vnode = h('div')
|
|
|
|
expect(h('div', {}, vnode)).toMatchObject(createVNode('div', {}, [vnode]))
|
2019-08-30 04:47:00 +08:00
|
|
|
// text
|
|
|
|
expect(h('div', {}, 'foo')).toMatchObject(createVNode('div', {}, 'foo'))
|
|
|
|
})
|
|
|
|
|
|
|
|
test('named slots with null props', () => {
|
2019-10-10 22:17:16 +08:00
|
|
|
const Component = { template: '<br />' }
|
2019-08-30 04:47:00 +08:00
|
|
|
const slot = () => {}
|
|
|
|
expect(
|
2019-10-10 22:17:16 +08:00
|
|
|
h(Component, null, {
|
2019-08-30 04:47:00 +08:00
|
|
|
foo: slot
|
|
|
|
})
|
|
|
|
).toMatchObject(
|
2019-10-10 22:17:16 +08:00
|
|
|
createVNode(Component, null, {
|
2019-08-30 04:47:00 +08:00
|
|
|
foo: slot
|
|
|
|
})
|
|
|
|
)
|
|
|
|
})
|
2020-03-19 22:54:03 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
describe('renderer: h', testH)
|
|
|
|
|
|
|
|
describe('renderer: transformHArgs', () => {
|
|
|
|
describe('no-op pass-through', () => {
|
|
|
|
beforeAll(() => {
|
|
|
|
transformHArgs((hArgs: unknown[]) => hArgs)
|
|
|
|
})
|
|
|
|
|
|
|
|
afterAll(resetTransformHArgs)
|
|
|
|
|
|
|
|
testH()
|
|
|
|
})
|
|
|
|
|
|
|
|
describe('args is used directly, without merging', () => {
|
|
|
|
beforeAll(() => {
|
|
|
|
transformHArgs(() => ['h1', 'Hello World'])
|
|
|
|
})
|
|
|
|
|
|
|
|
afterAll(resetTransformHArgs)
|
|
|
|
|
|
|
|
test('nodes become an h1 with text inside', () => {
|
|
|
|
expect(h('div')).toMatchObject(createVNode('h1', null, 'Hello World'))
|
|
|
|
})
|
|
|
|
|
|
|
|
test('resetting transformHArgs turns things back to normal', () => {
|
|
|
|
expect(h('div')).toMatchObject(createVNode('h1', null, 'Hello World'))
|
|
|
|
resetTransformHArgs()
|
|
|
|
expect(h('div')).toMatchObject(createVNode('div'))
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
test('receives component instance as the 2nd arg', () => {
|
|
|
|
transformHArgs((_: unknown[], instance: ComponentInternalInstance) => {
|
|
|
|
return ['h1', instance.type.name] // <h1>{{ name }}</h1>
|
|
|
|
})
|
|
|
|
|
|
|
|
const vm = createApp({
|
|
|
|
// this will be the name of the component in the h1
|
|
|
|
name: 'Root Component',
|
|
|
|
render() {
|
|
|
|
return h({
|
|
|
|
// this code will never execute,
|
|
|
|
// because it is overridden by the transformHArgs method
|
|
|
|
render() {
|
|
|
|
return h('h2', 'Stub Text')
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// we need to mount everything so that the instance passed to
|
|
|
|
// transformHArgs isn't null
|
|
|
|
vm.mount('body')
|
|
|
|
|
|
|
|
expect(document.body.outerHTML).toContain('<h1>Root Component</h1>')
|
|
|
|
})
|
2019-08-24 03:32:19 +08:00
|
|
|
})
|