Simple Vue.js Form Validation with Vuelidate
Dobromir Hristov | August 27th, 2018 | 8 min read
Thanks to Vue's reactivity model, it's really easy to roll your own form validations. This can be done with a simple method call on the form submit, or a computed property evaluating input data on each change.
Using your form validation can quickly become cumbersome and annoying, however, especially when the number of inputs in the form increase, or the form structure gets more complicated e.g. multi-step forms.
Thankfully, there are great validation plugins for Vue like Vuelidate. In this article, we'll be looking at how Vuelidate can be used to simplify:
- Validation
- Multi-step form validation
- Child component validation
- Error messages
We'll also see how the Vuelidate-error-extractor plugin can be used to simplify error message display per input, or as an error summary above or below the form.
Basic validation with Vuelidate
Vuelidate is data-model oriented, meaning validation rules are added to a validations
object in the component definition, rather than being added directly to input elements in the DOM.
The structure must resemble that of the form object, but the number of validation rules can be dynamic and change depending on which fields need validation.
export default {
name: "FormComponent",
data() {
return {
form: {
name: "",
email: ""
}
};
},
validations: {
form: {
name: { required },
email: { required, email }
}
}
...
};
Here's a live example:
Defining custom validators
Out of the box validation rules in Vuetify will work for most cases, but every so often you'll need a custom validator.
With Vuelidate, each validation rule is a function that returns a Boolean
or Promise
resolving to a Boolean
. This means you can predefine your own validators in a validators.js
file and just import each validator when needed.
Custom validators receive the currently validated data as a first param, and the whole data context as a second. For example, if you have a form data object and you are validating the email property, the first param will be the email itself and the second will be the whole data object.
// validators.js
export function isNameJoe(value) {
if (!value) return true;
return value === "Joe";
}
export function notGmail(value = "") {
return !value.includes("gmail");
}
export function isEmailAvailable(value) {
if (value === "") return true;
return new Promise((resolve, reject) => {
setTimeout(() => {
resolve(value.length > 10);
}, 500);
});
}
// formComponent.vue
import { required, email } from "vuelidate/lib/validators";
import { isNameJoe, notGmail, isEmailAvailable } from "@/validators";
export default {
name: "FormComponent",
data() {
return {
form: {
name: "",
email: ""
}
};
},
validations: {
form: {
name: { required, isJoe: isNameJoe },
email: { required, email, notGmail, isEmailAvailable }
}
},
methods: {
submit() {
this.$v.form.$touch();
// if its still pending or an error is returned do not submit
if (this.$v.form.$pending || this.$v.form.$error) return;
// to form submit after this
alert("Form submitted");
}
}
};
You can also create custom validators with the help of a few special helpers that come packed with Vuelidate. Check out the Custom Validators section in the Vuelidate docs for examples.
Dynamically changing rules
Being able to change the validation rules on the fly can be a godsend with multi-step forms. Each step has its own rules that validate certain parts of the form data.
Vuelidate can use computed properties as validation rules. That means that you can return different rules for each step of a multi-step form.
In the example below, validations
is now a function returning an object, rather just an object. This means it will be called after the component is initialized and computed properties are run.
export default {
...
data() {
return {
step: 1,
maxSteps: 2,
form: {
name: "",
email: ""
}
};
},
computed: {
rules () {
return this.step === 1
? { name: { required } }
: { email: { required, email } }
}
},
validations() {
return {
form: this.rules
}
}
}
Breaking down large forms into child components
Once a form gets bigger, you might want split up your form into several smaller components to avoid having a mega component handling all the form validation.
However, storing form data in separate components rather than in a single place makes collecting the data harder. You could loop each child component via a ref
binding and get the data, either implementing a data fetcher method or naming the form data in a specific way.
Another way is to store form data in Vuex, where you define all the rules on the parent and create a computed property referencing the store’s form state. Pass down the validator to each component if needed.
Tip: if using Vuex for form data, try the vuex-map-fields plugin to reduce boilerplate by setting each field as a computed property.
For most cases, I keep all the data and validation rules on a parent wrapping component that passes down the relevant validator to each child as a prop and handles sending the data to the server.
Each smaller component then uses the $touch()
method on its validator prop to note that data is being changed and emits changed data via $emit('input', value)
for easy v-model
binding.
To make the validator available to all children, you have a few options:
- Pass it down as a prop to each component
- Use the Provide/Inject API
- Create a new Vue instance in the store itself. Check this gist on how that might work
Here's a live example of the first method i.e. passing the down as a prop to each component. This is the simplest to comprehend and in most situations will be the one you will want to use.
Once we've covered error message display, I'll show you an example using the Provide/Inject API.
Validation error display
So the form is in place, its validated on each key press, but what about displaying error messages to the users?
We can just check each validator for errors and color our inputs, but what if we wanted to show a message? What if there is a need to display more than one error at a time? If/else checks start flying around everywhere.
<div class="form-group" :class="{ 'hasError': v.$error }">
<label class="mr-2 font-bold text-grey">Email</label>
<input type="email" class="input" v-model="email" placeholder="user@yahoo.com" @input="v.$touch()">
<div class="text-sm mt-2 text-red" v-if="v.$error">
<div v-if="!v.required">Email is required</div>
<div v-if="!v.notGmail">Email should not be a Gmail one</div>
<div v-if="!v.isEmailAvailable">Email is not available (less than 10 char)</div>
<div v-if="!v.email">Email is not a properly formatted email address</div>
</div>
</div>
As you can see there is a lot of repetition there, a lot of checking, you need to know what validators each field has. Adding a new rule means you have to go and edit the template too.
Error display with Vuelidate-error-extractor
There is a better way! Vuelidate-error-extractor, which I wrote, does the heavy lifting for you by extracting all the errors for each field, finding the appropriate error message for each rule and displaying it. It gives the user a flexible way of displaying errors with minimal boilerplate and repetitiveness.
You can use one of the built-in templates for Bootstrap and Foundation or just as easily build our own to suit your needs. All you need to do is register the plugin, define an object containing common error messages and register the template you want to use. We will refer to the single input error display as singleErrorExtractor
Creating custom error display components
Vuelidate-error-extractor loops the validations for each form data and checks each rule if it's valid or not. The invalid ones are then extracted and a validation error message is assigned to them.
The bundled singleErrorExtractorMixin
provides a set of helper methods and computed properties to assist the developer with building their own input error display.
<template>
<div class="form-group" :class="{ hasError: hasErrors, hasSuccess: isValid }">
<div class="label">
{{ label }}
</div>
<div class="control"><slot/></div>
<div class="control-helper text-red mt-4 text-sm" v-if="hasErrors">
<div v-for="error in activeErrorMessages" :key="error">{{ error }}</div>
</div>
</div>
</template>
<script>
import { singleErrorExtractorMixin } from "vuelidate-error-extractor";
export default {
mixins: [singleErrorExtractorMixin]
};
</script>
Now just wrap your input with the new component:
<form @submit.prevent="handleForm">
<form-group :validator="v" label="Email">
<input
class="input"
type="email"
v-model="email"
placeholder="user@yahoo.com"
@input="v.$touch()"
>
</form-group>
</form>
Check out custom templates docs for a detailed explanation on how to make your own error display.
Being so flexible means that you can adapt it to any Vue UI framework you wish. Here is a list of popular UI frameworks and examples how to implement it for each one: https://dobromir-hristov.github.io/vuelidate-error-extractor/other_frameworks.html
Form error summary
Sometimes you need to have a summary of all the errors in a form, be it on top or at the bottom.
You can use the prebuilt components for Foundation or Bootstrap, use the baseMultiErrorExtractor
component or the multiErrorExtractor
mixin. For 90% of use cases, the baseMultiErrorExtractor
will suffice.
<template>
<base-errors v-bind="$attrs">
<div class="text-red" slot-scope="{ errorMessage }">{{ errorMessage }}</div>
</base-errors>
</template>
<script>
import { templates } from "vuelidate-error-extractor";
export default {
inheritAttrs: false,
components: {
baseErrors: templates.multiErrorExtractor.baseMultiErrorExtractor
}
};
</script>
It will reuse the same error messages that you defined up front for the singleErrorExtractor
to use. The $validator
must be passed as a prop.
To be able to assign a proper field label to each error, it requires an object called attributes
to be defined where the error messages were defined. This object represents a map annotating how each field should be called, i.e. { name: "Name", email: "Email" }
.
Reducing boilerplate by injecting the validator
Passing the validator
and attribute
prop to each form input and the multiErrorExtractor
can get annoying quite fast, not to mention the boilerplate.
To overcome this, you can use the provided form-wrapper
component to inject the validator down to all inputs.
<template>
<div class="form pt-6">
<form-wrapper :validator="$v.form">
<form-summary/>
<form @submit.prevent="submit" novalidate>
<div class="flex text-center my-6 text-left">
<div class="w-1/2">
<name-component v-model="form.name"/>
</div>
<div class="w-1/2">
<email-component v-model="form.email"/>
</div>
</div>
<div class="text-center">
<button type="submit" class="button">
Submit
</button>
</div>
</form>
</form-wrapper>
</div>
</template>
This method uses the Provide/Inject API to pass down the $validator
to all the components that need it. Each form-group
can then drop its validator
and attribute
props, to be replaced by a single name
prop, annotating which field in the form it represents.
It will also try to figure out the input's attribute
property by checking the attributes
object we defined earlier.
Summary
Handling and validating forms on the front end as you saw, can often become a pain in the bum, especially when forms become big and need to be split up.
Using Vuelidate makes the whole ordeal a lot more bearable. In tandem with Vuelidate-error-extractor, displaying error messages under each input can go from a tedious repetitive chore, to simply adding a wrapping component or two, that do it all for you.
Code examples
- Basic validation with Vuelidate
- Defining custom validators
- Dynamically changing rules
- Breaking down large forms into child components
- Validation error display
- Error display with Vuelidate-error-extractor
- Creating custom error display components
- Form error summary
- Reducing boilerplate by injecting the validator
About Dobromir Hristov
Enjoys writing articles, open sourcing code, authoring libraries and teaching people. Check out his work at https://dobromir.me
Click to load comments...