Form

Form wrapper that provides better security practices and automatic input validation based on HTML5 form validation APIopen in new window.

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 VInputopen in new window. 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 nameopen in new window. 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: Returns false if any contained input is invalid based on its HTML5 validation attributesopen in new window.
  • dirty: Returns true if any contained input has triggered its blur event.
  • error: Returns true if the form is not valid and is dirty.
<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": false,
  "dirty": false,
  "modified": false,
  "error": false,
  "inputs": {
    "name": {
      "_inputEl": "[object HTMLInputElement]",
      "valid": false,
      "dirty": false,
      "invalid": {
        "type": false,
        "required": true,
        "minlength": false,
        "maxlength": false,
        "min": false,
        "max": false,
        "pattern": false
      },
      "value": "",
      "error": false,
      "errors": []
    }
  }
}

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 submitopen in new window 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 modifiersopen in new window such as .prevent, therefore if you want to prevent the browser from reloading (native behavior), you will want to call the event.preventDefault()open in new window 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>

The form has no unsaved changes.

The form will warn you before navigation unless you submit.

Clearing Inputs

Sometimes we need to clear our forms. HTML provides us with a reset inputopen in new window 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 status valid is false.
  • vts-form--dirty: Added to the form whenever the status dirty is true.
  • vts-form--error: Added to the form whenever the status error is true.

Security

By default, the form will set the method attributeopen in new window 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.