import { ref, h, render, nodeOps, serializeInner, nextTick, VNode, provide, inject, Ref, watch, SetupContext } from '@vue/runtime-test' describe('renderer: component', () => { test('should update parent(hoc) component host el when child component self update', async () => { const value = ref(true) let parentVnode: VNode let childVnode1: VNode let childVnode2: VNode const Parent = { render: () => { // let Parent first rerender return (parentVnode = h(Child)) } } const Child = { render: () => { return value.value ? (childVnode1 = h('div')) : (childVnode2 = h('span')) } } const root = nodeOps.createElement('div') render(h(Parent), root) expect(serializeInner(root)).toBe(`
`) expect(parentVnode!.el).toBe(childVnode1!.el) value.value = false await nextTick() expect(serializeInner(root)).toBe(``) expect(parentVnode!.el).toBe(childVnode2!.el) }) it('should create an Component with props', () => { const Comp = { render: () => { return h('div') } } const root = nodeOps.createElement('div') render(h(Comp, { id: 'foo', class: 'bar' }), root) expect(serializeInner(root)).toBe(`
`) }) it('should create an Component with direct text children', () => { const Comp = { render: () => { return h('div', 'test') } } const root = nodeOps.createElement('div') render(h(Comp, { id: 'foo', class: 'bar' }), root) expect(serializeInner(root)).toBe(`
test
`) }) it('should update an Component tag which is already mounted', () => { const Comp1 = { render: () => { return h('div', 'foo') } } const root = nodeOps.createElement('div') render(h(Comp1), root) expect(serializeInner(root)).toBe('
foo
') const Comp2 = { render: () => { return h('span', 'foo') } } render(h(Comp2), root) expect(serializeInner(root)).toBe('foo') }) // #2072 it('should not update Component if only changed props are declared emit listeners', () => { const Comp1 = { emits: ['foo'], updated: jest.fn(), render: () => null } const root = nodeOps.createElement('div') render( h(Comp1, { onFoo: () => {} }), root ) render( h(Comp1, { onFoo: () => {} }), root ) expect(Comp1.updated).not.toHaveBeenCalled() }) // #2043 test('component child synchronously updating parent state should trigger parent re-render', async () => { const App = { setup() { const n = ref(0) provide('foo', n) return () => { return [h('div', n.value), h(Child)] } } } const Child = { setup() { const n = inject>('foo')! n.value++ return () => { return h('div', n.value) } } } const root = nodeOps.createElement('div') render(h(App), root) expect(serializeInner(root)).toBe(`
0
1
`) await nextTick() expect(serializeInner(root)).toBe(`
1
1
`) }) // #2170 test('should have access to instance’s “$el” property in watcher when rendereing with watched prop', async () => { function returnThis(this: any) { return this } const propWatchSpy = jest.fn(returnThis) let instance: any const Comp = { props: { testProp: String }, watch: { testProp() { // @ts-ignore propWatchSpy(this.$el) } }, created() { instance = this }, render() { return h('div') } } const root = nodeOps.createElement('div') render(h(Comp), root) await nextTick() expect(propWatchSpy).not.toHaveBeenCalled() render(h(Comp, { testProp: 'prop ' }), root) await nextTick() expect(propWatchSpy).toHaveBeenCalledWith(instance.$el) }) // #2200 test('component child updating parent state in pre-flush should trigger parent re-render', async () => { const outer = ref(0) const App = { setup() { const inner = ref(0) return () => { return [ h('div', inner.value), h(Child, { value: outer.value, onUpdate: (val: number) => (inner.value = val) }) ] } } } const Child = { props: ['value'], setup(props: any, { emit }: SetupContext) { watch(() => props.value, (val: number) => emit('update', val)) return () => { return h('div', props.value) } } } const root = nodeOps.createElement('div') render(h(App), root) expect(serializeInner(root)).toBe(`
0
0
`) outer.value++ await nextTick() expect(serializeInner(root)).toBe(`
1
1
`) }) })