2019-08-26 22:08:56 +00:00
|
|
|
import { ref, reactive } from '@vue/reactivity'
|
|
|
|
import {
|
|
|
|
renderToString,
|
|
|
|
h,
|
|
|
|
nodeOps,
|
|
|
|
render,
|
|
|
|
serializeInner,
|
|
|
|
nextTick,
|
2020-02-22 07:19:10 +00:00
|
|
|
watchEffect,
|
2019-12-22 15:58:12 +00:00
|
|
|
defineComponent,
|
2019-08-26 22:08:56 +00:00
|
|
|
triggerEvent,
|
|
|
|
TestElement
|
|
|
|
} from '@vue/runtime-test'
|
|
|
|
|
2019-08-23 02:15:39 +00:00
|
|
|
// reference: https://vue-composition-api-rfc.netlify.com/api.html#setup
|
|
|
|
|
2019-08-23 19:32:19 +00:00
|
|
|
describe('api: setup context', () => {
|
2019-08-26 22:08:56 +00:00
|
|
|
it('should expose return values to template render context', () => {
|
2019-12-22 15:58:12 +00:00
|
|
|
const Comp = defineComponent({
|
2019-08-26 22:08:56 +00:00
|
|
|
setup() {
|
|
|
|
return {
|
|
|
|
// ref should auto-unwrap
|
|
|
|
ref: ref('foo'),
|
|
|
|
// object exposed as-is
|
|
|
|
object: reactive({ msg: 'bar' }),
|
|
|
|
// primitive value exposed as-is
|
|
|
|
value: 'baz'
|
|
|
|
}
|
|
|
|
},
|
|
|
|
render() {
|
|
|
|
return `${this.ref} ${this.object.msg} ${this.value}`
|
|
|
|
}
|
2019-09-05 22:48:49 +00:00
|
|
|
})
|
2019-08-26 22:08:56 +00:00
|
|
|
expect(renderToString(h(Comp))).toMatch(`foo bar baz`)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('should support returning render function', () => {
|
|
|
|
const Comp = {
|
|
|
|
setup() {
|
|
|
|
return () => {
|
|
|
|
return h('div', 'hello')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
expect(renderToString(h(Comp))).toMatch(`hello`)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('props', async () => {
|
|
|
|
const count = ref(0)
|
|
|
|
let dummy
|
|
|
|
|
|
|
|
const Parent = {
|
|
|
|
render: () => h(Child, { count: count.value })
|
|
|
|
}
|
|
|
|
|
2019-12-22 15:58:12 +00:00
|
|
|
const Child = defineComponent({
|
2020-04-03 01:51:01 +00:00
|
|
|
props: { count: Number },
|
|
|
|
setup(props) {
|
2020-02-22 07:19:10 +00:00
|
|
|
watchEffect(() => {
|
2019-08-26 22:08:56 +00:00
|
|
|
dummy = props.count
|
|
|
|
})
|
|
|
|
return () => h('div', props.count)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const root = nodeOps.createElement('div')
|
|
|
|
render(h(Parent), root)
|
|
|
|
expect(serializeInner(root)).toMatch(`<div>0</div>`)
|
|
|
|
expect(dummy).toBe(0)
|
|
|
|
|
|
|
|
// props should be reactive
|
|
|
|
count.value++
|
|
|
|
await nextTick()
|
|
|
|
expect(serializeInner(root)).toMatch(`<div>1</div>`)
|
|
|
|
expect(dummy).toBe(1)
|
|
|
|
})
|
|
|
|
|
2019-10-26 14:27:49 +00:00
|
|
|
it('setup props should resolve the correct types from props object', async () => {
|
2019-10-05 19:37:55 +00:00
|
|
|
const count = ref(0)
|
|
|
|
let dummy
|
|
|
|
|
|
|
|
const Parent = {
|
|
|
|
render: () => h(Child, { count: count.value })
|
|
|
|
}
|
|
|
|
|
2019-12-22 15:58:12 +00:00
|
|
|
const Child = defineComponent({
|
2019-10-05 19:37:55 +00:00
|
|
|
props: {
|
|
|
|
count: Number
|
|
|
|
},
|
|
|
|
|
|
|
|
setup(props) {
|
2020-02-22 07:19:10 +00:00
|
|
|
watchEffect(() => {
|
2019-10-05 19:37:55 +00:00
|
|
|
dummy = props.count
|
|
|
|
})
|
|
|
|
return () => h('div', props.count)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const root = nodeOps.createElement('div')
|
|
|
|
render(h(Parent), root)
|
|
|
|
expect(serializeInner(root)).toMatch(`<div>0</div>`)
|
|
|
|
expect(dummy).toBe(0)
|
|
|
|
|
|
|
|
// props should be reactive
|
|
|
|
count.value++
|
|
|
|
await nextTick()
|
|
|
|
expect(serializeInner(root)).toMatch(`<div>1</div>`)
|
|
|
|
expect(dummy).toBe(1)
|
|
|
|
})
|
|
|
|
|
2019-08-26 22:08:56 +00:00
|
|
|
it('context.attrs', async () => {
|
|
|
|
const toggle = ref(true)
|
|
|
|
|
|
|
|
const Parent = {
|
|
|
|
render: () => h(Child, toggle.value ? { id: 'foo' } : { class: 'baz' })
|
|
|
|
}
|
|
|
|
|
|
|
|
const Child = {
|
|
|
|
// explicit empty props declaration
|
|
|
|
// puts everything received in attrs
|
2019-10-26 14:31:36 +00:00
|
|
|
// disable implicit fallthrough
|
|
|
|
inheritAttrs: false,
|
2019-08-26 22:08:56 +00:00
|
|
|
setup(props: any, { attrs }: any) {
|
|
|
|
return () => h('div', attrs)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const root = nodeOps.createElement('div')
|
|
|
|
render(h(Parent), root)
|
|
|
|
expect(serializeInner(root)).toMatch(`<div id="foo"></div>`)
|
|
|
|
|
|
|
|
// should update even though it's not reactive
|
|
|
|
toggle.value = false
|
|
|
|
await nextTick()
|
|
|
|
expect(serializeInner(root)).toMatch(`<div class="baz"></div>`)
|
|
|
|
})
|
|
|
|
|
2021-07-21 21:31:00 +00:00
|
|
|
// #4161
|
|
|
|
it('context.attrs in child component slots', async () => {
|
|
|
|
const toggle = ref(true)
|
|
|
|
|
|
|
|
const Parent = {
|
|
|
|
render: () => h(Child, toggle.value ? { id: 'foo' } : { class: 'baz' })
|
|
|
|
}
|
|
|
|
|
|
|
|
const Wrapper = {
|
|
|
|
render(this: any) {
|
|
|
|
return this.$slots.default()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const Child = {
|
|
|
|
inheritAttrs: false,
|
|
|
|
setup(_: any, { attrs }: any) {
|
|
|
|
return () => {
|
|
|
|
const vnode = h(Wrapper, null, {
|
|
|
|
default: () => [h('div', attrs)],
|
|
|
|
_: 1 // mark stable slots
|
|
|
|
})
|
|
|
|
vnode.dynamicChildren = [] // force optimized mode
|
|
|
|
return vnode
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const root = nodeOps.createElement('div')
|
|
|
|
render(h(Parent), root)
|
|
|
|
expect(serializeInner(root)).toMatch(`<div id="foo"></div>`)
|
|
|
|
|
|
|
|
// should update even though it's not reactive
|
|
|
|
toggle.value = false
|
|
|
|
await nextTick()
|
|
|
|
expect(serializeInner(root)).toMatch(`<div class="baz"></div>`)
|
|
|
|
})
|
|
|
|
|
2019-08-26 22:08:56 +00:00
|
|
|
it('context.slots', async () => {
|
|
|
|
const id = ref('foo')
|
|
|
|
|
|
|
|
const Parent = {
|
|
|
|
render: () =>
|
|
|
|
h(Child, null, {
|
|
|
|
foo: () => id.value,
|
|
|
|
bar: () => 'bar'
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
const Child = {
|
|
|
|
setup(props: any, { slots }: any) {
|
|
|
|
return () => h('div', [...slots.foo(), ...slots.bar()])
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const root = nodeOps.createElement('div')
|
|
|
|
render(h(Parent), root)
|
|
|
|
expect(serializeInner(root)).toMatch(`<div>foobar</div>`)
|
|
|
|
|
|
|
|
// should update even though it's not reactive
|
|
|
|
id.value = 'baz'
|
|
|
|
await nextTick()
|
|
|
|
expect(serializeInner(root)).toMatch(`<div>bazbar</div>`)
|
|
|
|
})
|
|
|
|
|
|
|
|
it('context.emit', async () => {
|
|
|
|
const count = ref(0)
|
|
|
|
const spy = jest.fn()
|
|
|
|
|
|
|
|
const Parent = {
|
|
|
|
render: () =>
|
|
|
|
h(Child, {
|
|
|
|
count: count.value,
|
|
|
|
onInc: (newVal: number) => {
|
|
|
|
spy()
|
|
|
|
count.value = newVal
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-12-22 15:58:12 +00:00
|
|
|
const Child = defineComponent({
|
2019-08-26 22:08:56 +00:00
|
|
|
props: {
|
|
|
|
count: {
|
|
|
|
type: Number,
|
|
|
|
default: 1
|
|
|
|
}
|
|
|
|
},
|
|
|
|
setup(props, { emit }) {
|
|
|
|
return () =>
|
|
|
|
h(
|
|
|
|
'div',
|
|
|
|
{
|
|
|
|
onClick: () => emit('inc', props.count + 1)
|
|
|
|
},
|
|
|
|
props.count
|
|
|
|
)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
const root = nodeOps.createElement('div')
|
|
|
|
render(h(Parent), root)
|
|
|
|
expect(serializeInner(root)).toMatch(`<div>0</div>`)
|
|
|
|
|
|
|
|
// emit should trigger parent handler
|
|
|
|
triggerEvent(root.children[0] as TestElement, 'click')
|
|
|
|
expect(spy).toHaveBeenCalled()
|
|
|
|
await nextTick()
|
|
|
|
expect(serializeInner(root)).toMatch(`<div>1</div>`)
|
|
|
|
})
|
2019-08-23 19:32:19 +00:00
|
|
|
})
|