vue-form
Form validation for Vue.js 2.0+
Install
Available through npm as vue-form
.
import vueForm from 'vue-form';
Vue.use(vueForm);
...
mixins: [vueForm.mixin]
...
Usage
Once installed you have access to four components (vue-form
, validate
, field
, field-messages
) for managing form state, validating form fields and displaying validation messages.
Live examples
Example
<div id="app">
<vue-form :state="formstate" @submit.prevent="onSubmit">
<validate tag="label">
<span>Name *</span>
<input v-model="model.name" required name="name" />
<field-messages name="name">
<div>Success!</div>
<div slot="required">Name is a required field</div>
</field-messages>
</validate>
<validate tag="label">
<span>Email</span>
<input v-model="model.email" name="email" type="email" required />
<field-messages name="email">
<div slot="required">Email is a required field</div>
<div slot="email">Email is not valid</div>
</field-messages>
</validate>
<button type="submit">Submit</button>
</vue-form>
<pre>{{ formstate }}</pre>
</div>
Vue.use(vueForm);
new Vue({
el: '#app',
data: {
formstate: {},
model: {
name: '',
email: 'invalid-email'
}
},
methods: {
onSubmit: function () {
if(this.formstate.$invalid) {
return;
}
}
}
});
The output of formstate
will be:
{
"$dirty": false,
"$pristine": true,
"$valid": false,
"$invalid": true,
"$submitted": false,
"$touched": false,
"$untouched": true,
"$pending": false,
"$error": {
},
"$submittedState": {
},
"name": {
"$name": "name",
"$dirty": false,
"$pristine": true,
"$valid": false,
"$invalid": true,
"$touched": false,
"$untouched": true,
"$pending": false,
"$error": {
"required": true
}
},
"email": {
"$name": "email",
"$dirty": false,
"$pristine": true,
"$valid": false,
"$invalid": true,
"$touched": false,
"$untouched": true,
"$pending": false,
"$error": {
"email": true
}
}
}
Displaying messages
Display validation errors or success messages with field-messages
.
The show
prop supports simple expressions which specifiy when erros should be displayed based on the current state of the field, e.g: $dirty
, $dirty && $touched
, $dirty || $touched
<form-error field="fieldName" error="errorKey" show="$dirty">Error message</form-error>
<form-errors field="fieldName" show="$dirty && $touched">
<div slot="errorKeyA">Error message A</div>
<div slot="errorKeyB">Error message B</div>
</form-errors>
Validators
type="email"
type="url"
type="number"
required
minlength
maxlength
pattern
min (for type="number")
max (for type="number")
You can use static validation attributes or bindings. If it is a binding, the input will be re-validated every binding update meaning you can have inputs which are conditionally required etc.
<validate>
<input type="email" name="email" v-model="model.email" required />
</validate>
<validate>
<input type="text" name="name" v-model="model.name" maxlength="25" minlength="5" />
</validate>
<validate>
<input type="email" name="email" v-model="model.email" :required="isRequired" />
</validate>
<validate>
<input type="text" name="name" v-model="model.name" :maxlength="maxLen" :minlength="minLen" />
</validate>
Custom validators
You can register global and local custom validators.
Global custom validator
vueForm.addValidator('my-custom-validator', function (value, attrValue, vnode) {
return value === 'custom';
});
<validate>
<input v-model="something" name="something" my-custom-validator />
</validate>
Local custom validator
<validate :custom="{customValidator: customValidator}">
<input v-model="something" name="something" />
</validate>
methods: {
customValidator: function (value) {
return value === 'custom';
}
}
Async validators:
Async validators are custom validators which return a Promise. resolve()
true
or false
to set field vadility.
methods: {
customValidator (value) {
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve(value === 'ajax');
}, 100);
});
}
}
Async validator with debounce (example uses lodash debounce)
methods: {
debounced: _.debounce(function (value, resolve, reject) {
fetch('https://httpbin.org/get').then(function(response){
resolve(response.isValid);
});
}, 500),
customValidator (value) {
return new Promise((resolve, reject) => {
this.debounced(value, resolve, reject);
});
}
}
State classes
As form and input validation states change, state classes are added and removed
Possible form classes:
vf-form-dirty, vf-form-pristine, vf-form-valid, vf-form-invalid, vf-form-submitted
Possible input classes:
vf-dirty, vf-pristine, vf-valid, vf-invalid
// also for every validation error, a class will be added, e.g.
vf-invalid-required, vf-invalid-minlength, vf-invalid-max, etc
Input wrappers (e.g. the tag the validate
component renders) will also get state classes, but with the field
prefix, e.g.
vf-field-dirty, vf-field-pristine, vf-field-valid, vf-field-invalid
Custom components
When writing custom form field components, e.g. <my-checkbox v-model="foo"></my-checkbox>
you should trigger the focus
and blur
events after user interaction either by triggering native dom events on the root node of your component, or emitting Vue events (this.$emit('focus)
) so the validate
component can detect and set the $dirty
and $touched
states on the field.
Component props
vue-form
state
Object on which form state is set
validate
state
Optional way of passing in the form state. If omitted form state will be found in the $parentcustom
Object containing one or many custom validators. {validatorName: validatorFunction}
tag
String which specifies what element tag should be rendered by the validate
component, defaults to span
auto-label
: Automatically set for
and id
attributes of label and input elements found inside the validate
component
field-messages
state
Optional way of passing in the form state. If omitted form state will be found in the $parentname
String which specifies the related field nametag
String, defaults to div
show
: String, show error dependant on form field state e.g. $touched
, $dirty || $touched
, '$touched || $submitted'auto-label
: Automatically set the for
attribute of labels found inside the field-messages
component
Config
Set config options using vueForm.config
, defaults:
{
formComponent: 'vueForm',
messagesComponent: 'fieldMessages',
validateComponent: 'validate',
fieldComponent: 'field',
messagesTag: 'div',
fieldTag: 'div',
classes: {
form: {
dirty: 'vf-form-dirty',
pristine: 'vf-form-pristine',
valid: 'vf-form-valid',
invalid: 'vf-form-invalid',
touched: 'vf-form-touched',
untouched: 'vf-form-untouched',
submitted: 'vf-form-submitted',
pending: 'vf-form-pending'
},
validate: {
dirty: 'vf-field-dirty',
pristine: 'vf-field-pristine',
valid: 'vf-field-valid',
invalid: 'vf-field-invalid',
touched: 'vf-field-touched',
untouched: 'vf-field-untouched',
submitted: 'vf-field-submitted',
pending: 'vf-field-pending'
},
input: {
dirty: 'vf-dirty',
pristine: 'vf-pristine',
valid: 'vf-valid',
invalid: 'vf-invalid',
touched: 'vf-touched',
untouched: 'vf-untouched',
submitted: 'vf-submitted',
pending: 'vf-pending'
}
},
Promise: window.Promise
}