| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441 |
- import {
- createApp,
- h,
- nextTick,
- createComponent,
- vModelDynamic,
- applyDirectives,
- VNode
- } from '@vue/runtime-dom'
- const triggerEvent = (type: string, el: Element) => {
- const event = new Event(type)
- el.dispatchEvent(event)
- }
- const withVModel = (node: VNode, arg: any, mods?: any) =>
- applyDirectives(node, [[vModelDynamic, arg, '', mods]])
- const setValue = function(this: any, value: any) {
- this.value = value
- }
- let app: any, root: any
- beforeEach(() => {
- app = createApp()
- root = document.createElement('div') as any
- })
- describe('vModel', () => {
- it('should work with text input', async () => {
- const component = createComponent({
- data() {
- return { value: null }
- },
- render() {
- return [
- withVModel(
- h('input', {
- 'onUpdate:modelValue': setValue.bind(this)
- }),
- this.value
- )
- ]
- }
- })
- app.mount(component, root)
- const input = root.querySelector('input')
- const data = root._vnode.component.data
- input.value = 'foo'
- triggerEvent('input', input)
- await nextTick()
- expect(data.value).toEqual('foo')
- data.value = 'bar'
- await nextTick()
- expect(input.value).toEqual('bar')
- })
- it('should work with textarea', async () => {
- const component = createComponent({
- data() {
- return { value: null }
- },
- render() {
- return [
- withVModel(
- h('textarea', {
- 'onUpdate:modelValue': setValue.bind(this)
- }),
- this.value
- )
- ]
- }
- })
- app.mount(component, root)
- const input = root.querySelector('textarea')
- const data = root._vnode.component.data
- input.value = 'foo'
- triggerEvent('input', input)
- await nextTick()
- expect(data.value).toEqual('foo')
- data.value = 'bar'
- await nextTick()
- expect(input.value).toEqual('bar')
- })
- it('should support modifiers', async () => {
- const component = createComponent({
- data() {
- return { number: null, trim: null, lazy: null }
- },
- render() {
- return [
- withVModel(
- h('input', {
- class: 'number',
- 'onUpdate:modelValue': (val: any) => {
- this.number = val
- }
- }),
- this.number,
- {
- number: true
- }
- ),
- withVModel(
- h('input', {
- class: 'trim',
- 'onUpdate:modelValue': (val: any) => {
- this.trim = val
- }
- }),
- this.trim,
- {
- trim: true
- }
- ),
- withVModel(
- h('input', {
- class: 'lazy',
- 'onUpdate:modelValue': (val: any) => {
- this.lazy = val
- }
- }),
- this.lazy,
- {
- lazy: true
- }
- )
- ]
- }
- })
- app.mount(component, root)
- const number = root.querySelector('.number')
- const trim = root.querySelector('.trim')
- const lazy = root.querySelector('.lazy')
- const data = root._vnode.component.data
- number.value = '+01.2'
- triggerEvent('input', number)
- await nextTick()
- expect(data.number).toEqual(1.2)
- trim.value = ' hello, world '
- triggerEvent('input', trim)
- await nextTick()
- expect(data.trim).toEqual('hello, world')
- lazy.value = 'foo'
- triggerEvent('change', lazy)
- await nextTick()
- expect(data.lazy).toEqual('foo')
- })
- it('should work with checkbox', async () => {
- const component = createComponent({
- data() {
- return { value: null }
- },
- render() {
- return [
- withVModel(
- h('input', {
- type: 'checkbox',
- 'onUpdate:modelValue': setValue.bind(this)
- }),
- this.value
- )
- ]
- }
- })
- app.mount(component, root)
- const input = root.querySelector('input')
- const data = root._vnode.component.data
- input.checked = true
- triggerEvent('change', input)
- await nextTick()
- expect(data.value).toEqual(true)
- data.value = false
- await nextTick()
- expect(input.checked).toEqual(false)
- })
- it(`should support array as a checkbox model`, async () => {
- const component = createComponent({
- data() {
- return { value: [] }
- },
- render() {
- return [
- withVModel(
- h('input', {
- type: 'checkbox',
- class: 'foo',
- value: 'foo',
- 'onUpdate:modelValue': setValue.bind(this)
- }),
- this.value
- ),
- withVModel(
- h('input', {
- type: 'checkbox',
- class: 'bar',
- value: 'bar',
- 'onUpdate:modelValue': setValue.bind(this)
- }),
- this.value
- )
- ]
- }
- })
- app.mount(component, root)
- const foo = root.querySelector('.foo')
- const bar = root.querySelector('.bar')
- const data = root._vnode.component.data
- foo.checked = true
- triggerEvent('change', foo)
- await nextTick()
- expect(data.value).toMatchObject(['foo'])
- bar.checked = true
- triggerEvent('change', bar)
- await nextTick()
- expect(data.value).toMatchObject(['foo', 'bar'])
- bar.checked = false
- triggerEvent('change', bar)
- await nextTick()
- expect(data.value).toMatchObject(['foo'])
- foo.checked = false
- triggerEvent('change', foo)
- await nextTick()
- expect(data.value).toMatchObject([])
- data.value = ['foo']
- await nextTick()
- expect(bar.checked).toEqual(false)
- expect(foo.checked).toEqual(true)
- data.value = ['bar']
- await nextTick()
- expect(foo.checked).toEqual(false)
- expect(bar.checked).toEqual(true)
- data.value = []
- await nextTick()
- expect(foo.checked).toEqual(false)
- expect(bar.checked).toEqual(false)
- })
- it('should work with radio', async () => {
- const component = createComponent({
- data() {
- return { value: null }
- },
- render() {
- return [
- withVModel(
- h('input', {
- type: 'radio',
- class: 'foo',
- value: 'foo',
- 'onUpdate:modelValue': setValue.bind(this)
- }),
- this.value
- ),
- withVModel(
- h('input', {
- type: 'radio',
- class: 'bar',
- value: 'bar',
- 'onUpdate:modelValue': setValue.bind(this)
- }),
- this.value
- )
- ]
- }
- })
- app.mount(component, root)
- const foo = root.querySelector('.foo')
- const bar = root.querySelector('.bar')
- const data = root._vnode.component.data
- foo.checked = true
- triggerEvent('change', foo)
- await nextTick()
- expect(data.value).toEqual('foo')
- bar.checked = true
- triggerEvent('change', bar)
- await nextTick()
- expect(data.value).toEqual('bar')
- data.value = null
- await nextTick()
- expect(foo.checked).toEqual(false)
- expect(bar.checked).toEqual(false)
- data.value = 'foo'
- await nextTick()
- expect(foo.checked).toEqual(true)
- expect(bar.checked).toEqual(false)
- data.value = 'bar'
- await nextTick()
- expect(foo.checked).toEqual(false)
- expect(bar.checked).toEqual(true)
- })
- it('should work with single select', async () => {
- const component = createComponent({
- data() {
- return { value: null }
- },
- render() {
- return [
- withVModel(
- h(
- 'select',
- {
- value: null,
- 'onUpdate:modelValue': setValue.bind(this)
- },
- [h('option', { value: 'foo' }), h('option', { value: 'bar' })]
- ),
- this.value
- )
- ]
- }
- })
- app.mount(component, root)
- const input = root.querySelector('select')
- const foo = root.querySelector('option[value=foo]')
- const bar = root.querySelector('option[value=bar]')
- const data = root._vnode.component.data
- foo.selected = true
- triggerEvent('change', input)
- await nextTick()
- expect(data.value).toEqual('foo')
- foo.selected = false
- bar.selected = true
- triggerEvent('change', input)
- await nextTick()
- expect(data.value).toEqual('bar')
- foo.selected = false
- bar.selected = false
- data.value = 'foo'
- await nextTick()
- expect(input.value).toEqual('foo')
- expect(foo.selected).toEqual(true)
- expect(bar.selected).toEqual(false)
- foo.selected = true
- bar.selected = false
- data.value = 'bar'
- await nextTick()
- expect(input.value).toEqual('bar')
- expect(foo.selected).toEqual(false)
- expect(bar.selected).toEqual(true)
- })
- it('should work with multiple select', async () => {
- const component = createComponent({
- data() {
- return { value: [] }
- },
- render() {
- return [
- withVModel(
- h(
- 'select',
- {
- value: null,
- multiple: true,
- 'onUpdate:modelValue': setValue.bind(this)
- },
- [h('option', { value: 'foo' }), h('option', { value: 'bar' })]
- ),
- this.value
- )
- ]
- }
- })
- app.mount(component, root)
- const input = root.querySelector('select')
- const foo = root.querySelector('option[value=foo]')
- const bar = root.querySelector('option[value=bar]')
- const data = root._vnode.component.data
- foo.selected = true
- triggerEvent('change', input)
- await nextTick()
- expect(data.value).toMatchObject(['foo'])
- foo.selected = false
- bar.selected = true
- triggerEvent('change', input)
- await nextTick()
- expect(data.value).toMatchObject(['bar'])
- foo.selected = true
- bar.selected = true
- triggerEvent('change', input)
- await nextTick()
- expect(data.value).toMatchObject(['foo', 'bar'])
- foo.selected = false
- bar.selected = false
- data.value = ['foo']
- await nextTick()
- expect(input.value).toEqual('foo')
- expect(foo.selected).toEqual(true)
- expect(bar.selected).toEqual(false)
- foo.selected = false
- bar.selected = false
- data.value = ['foo', 'bar']
- await nextTick()
- expect(foo.selected).toEqual(true)
- expect(bar.selected).toEqual(true)
- })
- })
|