Form
Form wrapper that provides better security practices and automatic input validation based on HTML5 form validation API.
Features:
- Status tracking for inputs losing focus (
dirty
), having changes (modified
), validation states (valid
), and having errors. - Works with any
<input>
element. Not just Vuetensil's VInput. Even works with 3rd party libraries as long as the<input>
element is used in the DOM. - Prevent navigation if the form has been changed but has not been submitted.
- Provides classes to style invalid, dirty, or error states.
- Provides method for clearing every input.
- Supports honeypots to help reduce spam.
Important: Every input should have a name
. This is important for HTML forms, in general, but also helps VForm keep track. If none is provided, it will still work though.
Styled Example
<template> <VForm class="styled" @submit.prevent="onSubmit"> <template #default="form"> <label> Name: <small>(required)</small> <input name="name" required /> </label> <label> Email: <small>(required, type="email")</small> <input name="email" type="email" required /> </label> <label> Password: <small>(required, minlength="6")</small> <input name="password" type="password" required minlength="6" /> </label> <button :type="form.valid ? 'submit' : 'button'" :aria-disabled="!form.valid"> Submit </button> </template> </VForm> </template> <script> export default { methods: { onSubmit({ target }) { const form = new FormData(target); const data = Object.fromEntries(form); console.log(data); }, }, }; </script> <style> .vts-form input { width: 100%; margin-bottom: 1rem; } .vts-form input, .vts-form button { border-radius: 4px; padding: 0.5rem; } .vts-form input { border: 1px solid darkgray; } .vts-form button { border: 0; } .vts-form:not(.vts-form--invalid) button { color: #FFF; background: seagreen; } </style>
Form Status
The form provides the following status:
valid
: Returnsfalse
if any contained input is invalid based on its HTML5 validation attributes.dirty
: Returnstrue
if any contained input has triggered itsblur
event.error
: Returnstrue
if the form is notvalid
and isdirty
.
<template> <VForm @submit.prevent> <template #default="form"> <label> Name: <input name="name" required /> </label> <button :type="form.valid ? 'submit' : 'button'" :aria-disabled="!form.valid"> Submit </button> <pre>{{ form }}</pre> </template> </VForm> </template>
Valid And Invalid Events
You probably want to run different logic in the event of an invalid form submission and a valid form submission. Most of this logic must happen within the submit
event and you need to do some conditional logic.
To avoid that, VForm provides valid
and invalid
events that you can hook into for convenience. The events both receive the form's submit
event and can therefore be treated like a standard submit. The main difference is that they only run at their appropriate times.
Note that custom events do not allow for modifiers such as .prevent
, therefore if you want to prevent the browser from reloading (native behavior), you will want to call the event.preventDefault()
method. We also use novalidate
here to prevent the native validation from running.
<template> <VForm novalidate @valid="onValid" @invalid="onInvalid"> <label> Name: <input name="name" required /> </label> <button type="submit"> Submit </button> </VForm> </template> <script> export default { methods: { onValid(event) { event.preventDefault() console.log(event) alert('Everything looks good') }, onInvalid(event) { event.preventDefault() console.log(event) alert('Please fix the errors and try again') event.target.querySelector(':invalid:not(fieldset)').focus() }, } } </script>
Important: If you provide the invalid
listener, the form will also append the novalidate
attribtue which means the default HTML validation logic will not run. So you will want to handle showing the user the errors gracefully (which is kind of the intention of this event anyway).
Preventing Navigation
Sometimes you may want to prevent user navigation when a form has been modified, but not yet submitted. This can be a user experience improvement, especially on long forms, to avoid losing work.
To do so you can use the prevent-navigation
prop
<template> <VForm @submit.prevent prevent-navigation> <template #default="form"> <label> Name: <input name="name" /> </label> <button type="submit"> Submit </button> <p v-show="!form.modified">The form has no unsaved changes.</p> <p v-show="form.modified">The form will warn you before navigation unless you submit.</p> </template> </VForm> </template>
Clearing Inputs
Sometimes we need to clear our forms. HTML provides us with a reset input
and a reset button, but those only clear the form if the form was originally cleared to begin with.
Fortunately, VForm provides a clear()
method.
<template> <VForm @submit.prevent> <template #default="form"> <label> Text Input: <input name="text" value="inital value" /> </label> <br /> <label> Textarea: <textarea>inital value</textarea> </label> <br /> <button type="reset"> Reset </button> <button type="button" @click="form.clear"> Clear </button> <button :type="form.valid ? 'submit' : 'button'" :aria-disabled="!form.valid"> Submit </button> </template> </VForm> </template>
Classes
vts-form
: The root form class.vts-form--invalid
: Added to the form whenever the statusvalid
isfalse
.vts-form--dirty
: Added to the form whenever the statusdirty
istrue
.vts-form--error
: Added to the form whenever the statuserror
istrue
.
Security
By default, the form will set the method
attribute to POST
. This is SLIGHTLY more secure to avoid adding sensitive data to a query string.
Note that this security benefit is only applicable for server-side rendering AND if JavaScript fails.
To overright the method, simply provide method="GET"
to the VForm component.