## Form > New in 1.7.0+ > > 1.8.0+ supported trigger validating when blur and debounce. Supported async validate too because of Validator supported it. CubeForm is a schema-based form generator component. ### Example - default usage This demo includes all cube-ui form fields. ```html ``` ```js export default { data() { return { validity: {}, valid: undefined, model: { checkboxValue: false, checkboxGroupValue: [], inputValue: '', radioValue: '', rateValue: 0, selectValue: 2018, switchValue: true, textareaValue: '', uploadValue: [] }, schema: { groups: [ { legend: '基础', fields: [ { type: 'checkbox', modelKey: 'checkboxValue', props: { option: { label: 'Checkbox', value: true } }, rules: { required: true }, messages: { required: 'Please check this field' } }, { type: 'checkbox-group', modelKey: 'checkboxGroupValue', label: 'CheckboxGroup', props: { options: ['1', '2', '3'] }, rules: { required: true } }, { type: 'input', modelKey: 'inputValue', label: 'Input', props: { placeholder: '请输入' }, rules: { required: true }, // validating when blur trigger: 'blur' }, { type: 'radio-group', modelKey: 'radioValue', label: 'Radio', props: { options: ['1', '2', '3'] }, rules: { required: true } }, { type: 'select', modelKey: 'selectValue', label: 'Select', props: { options: [2015, 2016, 2017, 2018, 2019, 2020] }, rules: { required: true } }, { type: 'switch', modelKey: 'switchValue', label: 'Switch', rules: { required: true } }, { type: 'textarea', modelKey: 'textareaValue', label: 'Textarea', rules: { required: true }, // debounce validate // if set to true, the default debounce time will be 200(ms) debounce: 100 } ] }, { legend: '高级', fields: [ { type: 'rate', modelKey: 'rateValue', label: 'Rate', rules: { required: true } }, { type: 'upload', modelKey: 'uploadValue', label: 'Upload', events: { 'file-removed': (...args) => { console.log('file removed', args) } }, rules: { required: true, uploaded: (val, config) => { return Promise.all(val.map((file, i) => { return new Promise((resolve, reject) => { if (file.uploadedUrl) { return resolve() } // fake request setTimeout(() => { if (i % 2) { reject(new Error()) } else { file.uploadedUrl = 'uploaded/url' resolve() } }, 1000) }) })).then(() => { return true }) } }, messages: { uploaded: '上传失败' } } ] }, { fields: [ { type: 'submit', label: 'Submit' }, { type: 'reset', label: 'Reset' } ] } ] }, options: { scrollToInvalidField: true, layout: 'standard' // classic fresh } } }, methods: { submitHandler(e) { e.preventDefault() console.log('submit', e) }, validateHandler(result) { this.validity = result.validity this.valid = result.valid console.log('validity', result.validity, result.valid, result.dirty, result.firstInvalidFieldIndex) }, resetHandler(e) { console.log('reset', e) } } } ``` `model` is the model/target JSON object, `schema` is the schema object with fields, if `immediate-validate` is true, we will run validation after load. `submit` is the submited event, the `validate` is the form validate event and the `reset` is the form reset event. - Custom usage You can choose to use your own custom components, or use slots to custom your logic. ```html {{model.dateValue || 'Please select date'}} Submit ``` ```js // province, city, area // select component const PCA = { props: { value: { type: Array, default() { return [] } } }, data() { return { selected: [] } }, render(createElement) { return createElement('cube-button', { on: { click: this.showPicker } }, this.selected.length ? this.selected.join(' ') : 'placeholder') }, mounted() { this.picker = this.$createCascadePicker({ title: 'PCA Select', data: cityData, selectedIndex: this.value, onSelect: this.selectHandler }) }, methods: { showPicker() { this.picker.show() }, selectHandler(selectedVal, selectedIndex, selectedTxt) { this.selected = selectedTxt this.$emit('input', selectedVal) } } } export default { data() { return { validity: {}, valid: undefined, model: { inputValue: '', pcaValue: [], dateValue: '' }, fields: [ { type: 'input', modelKey: 'inputValue', label: 'Input', props: { placeholder: '请输入' }, rules: { required: true } }, { component: PCA, modelKey: 'pcaValue', label: 'PCASelect', rules: { required: true }, messages: { required: '请选择' } }, { modelKey: 'dateValue', label: 'Date', rules: { required: true } } ] } }, methods: { submitHandler(e) { console.log('submit') }, validateHandler(result) { this.validity = result.validity this.valid = result.valid console.log('validity', result.validity, result.valid, result.dirty, result.firstInvalidFieldIndex) }, showDatePicker() { this.$refs.datePicker.show() }, dateSelectHandler(selectedVal) { this.model.dateValue = new Date(selectedVal[0], selectedVal[1] - 1, selectedVal[2]).toDateString() } }, components: { DatePicker } } ``` Like the `PCA` component, you can use `component` to define your custom component. Or use slots, like use DatePicker component. - Questionnaire You can use form features to build your own app, like questionnaire. Code source: https://github.com/didi/cube-ui/tree/dev/example/components/questionnaire ```html ``` ```js // source // https://github.com/didi/cube-ui/tree/dev/example/components/questionnaire/questionnaire.vue import DemoQuestionnaire from 'example/components/questionnaire/questionnaire.vue' export default { data() { return { tip: '请配合如实填写问卷,确保xxxx相关文案', questions: [ { type: 'switch', model: 'switch', title: '询问是否?' // required: true }, { type: 'input', model: 'input', title: '输入', options: { placeholder: '请输入' }, on: 'switch', required: true }, { type: 'date', model: 'date', title: '日期', options: { // min: '2020-01-01', // max: '2020-02-18' }, required: true }, { type: 'time', model: 'time', title: '时间', options: { min: '01:00', max: '23:59' }, required: true }, { type: 'select', model: 'select', title: '选择', options: [ 'option1', 'option2', 'option3' ], required: true }, { type: 'radio', model: 'radio', title: '单选', options: [ '单选1', '单选2', '单选3' ], required: true }, { type: 'checkbox', model: 'checkbox', title: '多选', options: [ '多选1', '多选2', '多选3' ], required: true }, { type: 'textarea', model: 'textarea', title: '多行文本', on: { model: 'checkbox', options: ['多选1', '多选3'] }, required: true }, { type: 'checkbox', row: true, model: 'checkbox2', title: '多选-横', options: [ '多选-横1', '多选-横2', '多选-横3' ], required: true }, { type: 'tel', model: 'tel', title: '手机号', options: { placeholder: '请输入手机号' }, required: true }, { type: 'rate', model: 'rate', title: '级别', options: { max: 10 }, required: true }, { type: 'city', model: 'city', title: '城市', required: true }, { type: 'upload', model: 'upload', title: '上传', options: { action: '//jsonplaceholder.typicode.com/photos/', max: 2 }, required: true }, { type: 'agreement', model: 'agreement', options: { text: '请同意', link: { text: '《xx协议》', href: 'https://github.com/didi/cube-ui' }, desc: '说明:本人承诺xx xxxxx xxx xx。' }, required: true, errMsg: '请同意协议' } ], submit: { text: 'Submit' } } }, components: { DemoQuestionnaire }, methods: { submitHandler(model) { console.log('submit', model) } } } ``` ### Props #### Form | Attribute | Description | Type | Accepted Values | Default | | - | - | - | - | - | | model | The model/target JSON object | Object | - | {} | | schema | the schema object with fields | Object | - | {} | | immediateValidate | If true, we will run validation after load | Boolean | true/false | false | | action | Form action value | String | - | undefined | | options | Options for CubeForm | Object | - | {
scrollToInvalidField: false,
layout: 'standard' // or: classic|fresh
} | | submitAlwaysValidate1.12.36+ | If true, we will always validate all fields when submit | Boolean | true/false | false | - `schema` sub configuration The schema contains the fields of the form, optionally in groups. - Without groups Use `fields`: ```js { fields: [ { type: 'input', modelKey: 'inputValue', label: 'Input' }, // ... ] } ``` - With groups Use `groups`: ```js { groups: [ { legend: 'Group 1' fields: [ { type: 'input', modelKey: 'inputValue', label: 'Input' }, // ... ] }, { legend: 'Group 2' fields: [ { type: 'input', modelKey: 'inputValue', label: 'Input' }, // ... ] } ] } ``` A field in the schema defines an input element to a value of model. The common properties of field: | Attribute | Description | Type | Accepted Values | Default | | - | - | - | - | - | | type | Type of field | String | Default fields: `button`, `checkbox`, `checkbox-group`, `input`, `radio`, `radio-group`, `rate`, `select`, `switch`, `textarea`, `upload`. There are two special types: `submit` and `reset`, they will be transformed to the corresponding type of `button`. | - | | component | Custom filed component to replace `type`, the component should to work with `v-model` | Object/String | - | - | | modelKey | Name of property in the `form` model | String | - | - | | label | Label of field | String | - | - | | props | This value will be the `type` or `component` props | Object | - | - | | events1.8.0+ | This value will be the `type` or `component` custom events | Object | - | - | | rules | Validator rules, see Validator | Object | - | - | | trigger1.8.0+ | If set to 'blur' then will be validate this filed when blur | String | blur/change | - | | debounce1.8.0+ | Debounce validating time(ms). If `trigger` is 'blur' then the debounce will be ignored | Number/Boolean | >= 0, if set to true the time will be 200(ms) | - | | messages | Validator messages, see Validator | String | - | - | | key1.12.36+ | field union key | String | - | - | - `options` sub configuration | Attribute | Description | Type | Accepted Values | Default | | - | - | - | - | - | | scrollToInvalidField | Whether to scroll to the first invalid fileld by default | Boolean | true/false | false | | layout | Form layout style | String | standard/classic/fresh | standard | #### FormGroup | Attribute | Description | Type | Accepted Values | Default | | - | - | - | - | - | | legend | Group legend | String | - | '' | | fields | Group fields | Array | - | [] | #### FormItem | Attribute | Description | Type | Accepted Values | Default | | - | - | - | - | - | | field | Field value | Object | - | - | ### Events | Event Name | Description | Parameters 1 | Parameters 2 | Parameters 3 | | - | - | - | - | - | | submit | Form submit, only trigged when the form's validation is ok. If only have sync validators, this event will not be prevented by default. If have async validators, this event will be prevented by default. | e - event | form model value | form model value with only fields `modelKey`1.12.30+ | | reset | Form reset | e - event | - | - | | validate | Form validated | Properties:
{
validity,
valid,
invalid,
dirty,
firstInvalidFieldIndex
} | - | - | | valid | Form valid | Validity result | - | - | | invalid | Form invalid | Validity result | - | - | - `validate` event parameters | Attribute | Description | Type | | - | - | - | | validity | Validity result | Object | | valid | Validity check, if not verified this value will be undefined, once checked this value will be true or false | Boolean/Undefined | | invalid | Validity check, if not verified this value will be undefined, once checked this value will be true or false | Boolean | | dirty | When the form model changed then the form's dirty is true | Boolean | | firstInvalidFieldIndex | First invalid field index | Number | - Validity result `validity` | Attribute | Description | Type | | - | - | - | | valid | Is valid or not | Boolean/Undefined | | result | Result, eg:
{
 required: {
  valid: false,
  invalid: true,
  message: 'Required.'
 }
} | Object | | dirty | The form model is changed or not | Boolean | ### Instance methods | Method name | Description | Parameters | Returned value | | - | - | - | - | | submit | submit form | skipValidate: default `false`, if skipValidate is `true` then will be trigged submit event without validating1.12.2+ | - | | reset | reset form | - | - | | validate(cb) | validate form | cb: validated callback function, used to async validating cases normally. The arguments is the `valid` value | If supported Promise then the returned value will be Promise instance(Only have resolved state, the resolved value is `valid`), otherwise the returned value is `undefined` |