2019-10-07 10:24:30 -04:00
|
|
|
|
import Vue from 'vue'
|
2020-02-25 17:32:40 -05:00
|
|
|
|
import flushPromises from 'flush-promises'
|
2019-10-07 10:24:30 -04:00
|
|
|
|
import { mount } from '@vue/test-utils'
|
2020-03-07 09:03:59 -05:00
|
|
|
|
import Formulate from '../../src/Formulate.js'
|
2019-11-01 16:01:42 -04:00
|
|
|
|
import FormulateInput from '@/FormulateInput.vue'
|
|
|
|
|
import FormulateInputText from '@/inputs/FormulateInputText.vue'
|
2019-10-08 13:50:53 -04:00
|
|
|
|
import { doesNotReject } from 'assert';
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
|
|
|
|
Vue.use(Formulate)
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Test each type of text element
|
|
|
|
|
*/
|
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
describe('FormulateInputText', () => {
|
|
|
|
|
it('renders text input when type is "text"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders search input when type is "search"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'search' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders email input when type is "email"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'email' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders number input when type is "number"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'number' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders color input when type is "color"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'color' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders date input when type is "date"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'date' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders month input when type is "month"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'month' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders password input when type is "password"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'password' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders tel input when type is "tel"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'tel' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders time input when type is "time"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'time' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders url input when type is "url"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'url' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders week input when type is "week"', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'week' } })
|
|
|
|
|
expect(wrapper.contains(FormulateInputText)).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
/**
|
|
|
|
|
* Test rendering functionality to text inputs
|
|
|
|
|
*/
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('automatically assigns an id', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text' } })
|
|
|
|
|
expect(wrapper.vm.context).toHaveProperty('id')
|
|
|
|
|
expect(wrapper.find(`input[id="${wrapper.vm.context.attributes.id}"]`).exists()).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('doesn’t automatically add a label', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text' } })
|
|
|
|
|
expect(wrapper.find('label').exists()).toBe(false)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders labels', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', label: 'Field label' } })
|
|
|
|
|
expect(wrapper.find(`label[for="${wrapper.vm.context.attributes.id}"]`).exists()).toBe(true)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('doesn’t automatically render help text', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text' } })
|
|
|
|
|
expect(wrapper.find(`.formulate-input-help`).exists()).toBe(false)
|
|
|
|
|
})
|
2019-10-07 10:24:30 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('renders help text', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', help: 'This is some help text' } })
|
|
|
|
|
expect(wrapper.find(`.formulate-input-help`).exists()).toBe(true)
|
|
|
|
|
})
|
2019-10-08 13:50:53 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
/**
|
|
|
|
|
* Test data binding
|
|
|
|
|
*/
|
|
|
|
|
it('emits input (vmodel) event with value when edited', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text' } })
|
|
|
|
|
wrapper.find('input').setValue('Updated Value')
|
|
|
|
|
expect(wrapper.emitted().input).toBeTruthy()
|
|
|
|
|
expect(wrapper.emitted().input[0]).toEqual(['Updated Value'])
|
|
|
|
|
})
|
2019-10-08 13:50:53 -04:00
|
|
|
|
|
|
|
|
|
|
2020-02-25 17:32:40 -05:00
|
|
|
|
it('doesn’t re-context itself if there were no changes', async () => {
|
2019-11-06 17:17:19 -05:00
|
|
|
|
const wrapper = mount({
|
|
|
|
|
data () {
|
|
|
|
|
return {
|
|
|
|
|
valueA: 'first value',
|
|
|
|
|
valueB: 'second value'
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
template: `
|
|
|
|
|
<div>
|
|
|
|
|
<FormulateInput type="text" ref="first" v-model="valueA" :placeholder="valueA" />
|
|
|
|
|
<FormulateInput type="text" ref="second" v-model="valueB" :placeholder="valueB" />
|
|
|
|
|
</div>
|
|
|
|
|
`
|
|
|
|
|
})
|
2020-02-25 17:32:40 -05:00
|
|
|
|
await flushPromises()
|
2019-11-06 17:17:19 -05:00
|
|
|
|
const firstContext = wrapper.find({ref: "first"}).vm.context
|
|
|
|
|
const secondContext = wrapper.find({ref: "second"}).vm.context
|
|
|
|
|
wrapper.find('input').setValue('new value')
|
2020-02-25 17:32:40 -05:00
|
|
|
|
await flushPromises()
|
2019-11-06 17:17:19 -05:00
|
|
|
|
expect(firstContext).toBeTruthy()
|
|
|
|
|
expect(wrapper.vm.valueA === 'new value').toBe(true)
|
|
|
|
|
expect(wrapper.vm.valueB === 'second value').toBe(true)
|
|
|
|
|
expect(wrapper.find({ref: "first"}).vm.context === firstContext).toBe(false)
|
|
|
|
|
expect(wrapper.find({ref: "second"}).vm.context === secondContext).toBe(true)
|
|
|
|
|
})
|
2019-10-29 23:33:31 -04:00
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('uses the v-model value as the initial value', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', formulateValue: 'initial val' } })
|
|
|
|
|
expect(wrapper.find('input').element.value).toBe('initial val')
|
|
|
|
|
})
|
|
|
|
|
|
2020-01-31 22:50:18 -05:00
|
|
|
|
it('uses the value as the initial value', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', value: 'initial val' } })
|
|
|
|
|
expect(wrapper.find('input').element.value).toBe('initial val')
|
|
|
|
|
})
|
|
|
|
|
|
2020-03-01 22:29:54 -05:00
|
|
|
|
it('uses the value prop as the initial value when v-model provided', () => {
|
2020-01-31 22:50:18 -05:00
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', formulateValue: 'initial val', value: 'initial other val' } })
|
2020-03-01 22:29:54 -05:00
|
|
|
|
expect(wrapper.find('input').element.value).toBe('initial other val')
|
2020-01-31 22:50:18 -05:00
|
|
|
|
})
|
|
|
|
|
|
2019-11-06 17:17:19 -05:00
|
|
|
|
it('uses a proxy model internally if it doesnt have a v-model', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'textarea' } })
|
|
|
|
|
const input = wrapper.find('textarea')
|
|
|
|
|
input.setValue('changed value')
|
|
|
|
|
expect(wrapper.vm.internalModelProxy).toBe('changed value')
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Test error handling
|
|
|
|
|
*/
|
|
|
|
|
it('doesn’t automatically render errors', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text' } })
|
|
|
|
|
expect(wrapper.find('.formulate-input-errors').exists()).toBe(false)
|
|
|
|
|
})
|
2019-10-31 10:24:18 -04:00
|
|
|
|
|
2020-03-06 16:10:25 -05:00
|
|
|
|
it('accepts a single string as an error prop', async () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', error: 'This is an error' } })
|
2019-11-06 17:17:19 -05:00
|
|
|
|
expect(wrapper.find('.formulate-input-errors').exists()).toBe(true)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
it('accepts an array as errors prop', () => {
|
2019-11-13 16:10:17 -05:00
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', errorBehavior: 'live', errors: ['This is an error', 'this is another'] } })
|
2019-11-06 17:17:19 -05:00
|
|
|
|
expect(wrapper.findAll('.formulate-input-error').length).toBe(2)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
it('removes any duplicate errors', () => {
|
2019-11-13 16:10:17 -05:00
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', errorBehavior: 'live', errors: ['This is an error', 'This is an error'] } })
|
2019-11-06 17:17:19 -05:00
|
|
|
|
expect(wrapper.findAll('.formulate-input-error').length).toBe(1)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
it('adds data-has-errors when there are errors', () => {
|
2019-11-13 16:10:17 -05:00
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', errorBehavior: 'live', errors: ['This is an error', 'This is an error'] } })
|
2019-11-06 17:17:19 -05:00
|
|
|
|
expect(wrapper.find('[data-has-errors]').exists()).toBe(true)
|
|
|
|
|
})
|
2019-11-13 16:10:17 -05:00
|
|
|
|
|
2020-03-06 16:10:25 -05:00
|
|
|
|
it('Should always show explicitly set errors, but not validation errors', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', validation: 'required', errorBehavior: 'blur', errors: ['Bad input'] } })
|
2019-11-13 16:10:17 -05:00
|
|
|
|
expect(wrapper.find('[data-has-errors]').exists()).toBe(true)
|
2020-03-06 16:10:25 -05:00
|
|
|
|
expect(wrapper.find('[data-is-showing-errors]').exists()).toBe(true)
|
|
|
|
|
expect(wrapper.findAll('.formulate-input-error').length).toBe(1)
|
2019-11-13 16:10:17 -05:00
|
|
|
|
})
|
|
|
|
|
|
2020-03-06 16:10:25 -05:00
|
|
|
|
it('Should show no errors when there are no errors', () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text' } })
|
|
|
|
|
expect(wrapper.find('[data-has-errors]').exists()).toBe(false)
|
|
|
|
|
expect(wrapper.find('[data-is-showing-errors]').exists()).toBe(false)
|
|
|
|
|
expect(wrapper.findAll('.formulate-input-error').exists()).toBe(false)
|
|
|
|
|
})
|
2019-11-13 16:10:17 -05:00
|
|
|
|
|
2020-03-06 16:10:25 -05:00
|
|
|
|
it('allows error-behavior blur to be overridden with show-errors', async () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', errorBehavior: 'blur', showErrors: true, validation: 'required' } })
|
|
|
|
|
await flushPromises()
|
2019-11-13 16:10:17 -05:00
|
|
|
|
expect(wrapper.find('[data-has-errors]').exists()).toBe(true)
|
|
|
|
|
expect(wrapper.find('[data-is-showing-errors]').exists()).toBe(true)
|
|
|
|
|
expect(wrapper.findAll('.formulate-input-errors').exists()).toBe(true)
|
|
|
|
|
expect(wrapper.findAll('.formulate-input-error').length).toBe(1)
|
|
|
|
|
})
|
|
|
|
|
|
2020-02-25 17:32:40 -05:00
|
|
|
|
it('shows errors on blur with error-behavior blur', async () => {
|
2020-03-06 16:10:25 -05:00
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'text', errorBehavior: 'blur', validation: 'required' } })
|
|
|
|
|
await wrapper.vm.$nextTick()
|
|
|
|
|
await flushPromises()
|
2019-11-13 16:10:17 -05:00
|
|
|
|
expect(wrapper.find('[data-has-errors]').exists()).toBe(true)
|
|
|
|
|
expect(wrapper.find('[data-is-showing-errors]').exists()).toBe(false)
|
2020-03-06 16:10:25 -05:00
|
|
|
|
expect(wrapper.findAll('.formulate-input-error').exists()).toBe(false)
|
|
|
|
|
expect(wrapper.vm.showValidationErrors).toBe(false)
|
2019-11-13 16:10:17 -05:00
|
|
|
|
wrapper.find('input').trigger('blur')
|
2020-02-25 17:32:40 -05:00
|
|
|
|
await flushPromises()
|
2020-03-06 16:10:25 -05:00
|
|
|
|
expect(wrapper.vm.showValidationErrors).toBe(true)
|
2019-11-13 16:10:17 -05:00
|
|
|
|
expect(wrapper.find('[data-is-showing-errors]').exists()).toBe(true)
|
2020-03-06 16:10:25 -05:00
|
|
|
|
// expect(wrapper.findAll('.formulate-input-errors').exists()).toBe(true)
|
|
|
|
|
// expect(wrapper.findAll('.formulate-input-error').length).toBe(1)
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
it('continues to show errors if validation fires more than one time', async () => {
|
|
|
|
|
const wrapper = mount(FormulateInput, { propsData: { type: 'date', errorBehavior: 'live', validation: [['after', '01/01/2021']] , value: '01/01/1999' } })
|
|
|
|
|
await wrapper.vm.$nextTick()
|
|
|
|
|
await flushPromises()
|
|
|
|
|
expect(wrapper.find('[data-has-errors]').exists()).toBe(true)
|
|
|
|
|
wrapper.find('input').trigger('input')
|
|
|
|
|
await wrapper.vm.$nextTick()
|
|
|
|
|
await flushPromises()
|
|
|
|
|
expect(wrapper.find('[data-has-errors]').exists()).toBe(true)
|
2019-11-13 16:10:17 -05:00
|
|
|
|
})
|
2019-10-31 10:24:18 -04:00
|
|
|
|
})
|