React Minimalistic Use Form
Table of contents
Minimalistic react hook for handling forms without much pain.
React-minimalistic-use-form provides minimalistic way to handle forms in react.
It takes care of form state and validation. Validation is based on html5 form validation.
It uses Constraint Validation API and
validates form based on input validity
.
This library has just peer dependencies "react": ">=16.13.1"
, "react-dom": ">=16.13.1"
and absolutely ZERO OTHER DEPENDENCIES, it's lightweight and powerful!
It provides ability to validate form fields on user input AND|OR on form submit.
It automatically adds or removes error class name to input field if field is valid or not.
Form can be explicitly controlled by manually attaching event handlers or implicitly by combining useForm
hook
and <Form />
component where Form component will bind values and event handlers automatically for you!
Installation
-
NPM
npm i react-minimalistic-use-form
-
YARN
yarn add react-minimalistic-use-form
Features
- Form state management
- Native html5 form validation
- Form validation
on input
oron submit
- Automatically toggles error and "is-touched" input class names
- Scroll to error
- Very lightweight with no dependencies
- Typescript support
- Built-in validation debounce with auto cancellation for stale validations
- Opt-in via plugins to add
custom validator
or custom scroll to error functionality
Current supported form fields
- input
- text
- number
- password
- textarea
- checkbox
- radio
- select
- tel
- date
- search
- url
- color
Supported validation rules - built in Form validation
- valid – Is
true
when the field passes validation. - valueMissing – Is
true
when the field is empty butrequired
. - typeMismatch – Is
true
when the field type isemail
orurl
but the entered value isnot
thecorrect type
. - tooShort – Is
true
when the field contains aminLength attribute
and the entered value isshorter than that length
. - tooLong – Is
true
when the field contains amaxLength attribute
and the entered value islonger than that length
. - patternMismatch – Is
true
when the field contains apattern attribute
and the entered valuedoes not match the pattern
. - badInput – Is
true
when the input type isnumber
and the entered value isnot a number
. - stepMismatch – Is
true
when the field has astep attribute
and the entered valuedoes not adhere to the step values
. - rangeOverflow – Is
true
when the field has amax attribute
and the entered number value isgreater than the max
. - rangeUnderflow – Is
true
when the field has amin attribute
and the entered number value islower than the min
.
For custom validation see Plugins - validator.
Example usage
EXPLICIT FORM CONTROL
- import useForm hook
- attach formRef to the form
- attach event handlers (needed form handling iternal state and validation)
import React from 'react';
import { useForm } from 'react-minimalistic-use-form';
export const MyForm = () => {
const {
isFormValid, values, onChange, onBlur, onSubmit, errors, formRef
} = useForm({ initialValues: { field1: 'foo', field2: 'bar'} });
const submitForm = ({ event, values, errors, isFormValid }) => {
event.preventDefault();
/* do some logic here... */
};
return (
<form onSubmit={onSubmit(submitForm)} noValidate ref={formRef}>
<div>
<label htmlFor="field1">Field1</label>
<input id="field1" name="field1" type="email" required value={values.field1} onChange={onChange} onBlur={onBlur} />
{errros.field1 && Object.values(errors.field1).map((error, index) => (
<span key={index} className="text-error">
{error}
<br />
</span>))}
</div>
<div>
<label htmlFor="field2">Field2</label>
<input id="field2" name="field2" type="text" minLength="5" required value={values.field2} onChange={onChange} onBlur={onBlur} />
{errros.field2 && Object.values(errors.field2).map((error, index) => (
<span key={index} className="text-error">
{error}
<br />
</span>))}
</div>
<div>
<button type="submit" disabled={isFormValid === false}>
<span>Start!</span>
</button>
</div>
</form>
);
}
IMPLICIT FORM CONTROL
If you find adding onChange
and onBlur
event handlers painful and repetitive job, or you simply don't want to add any extra
logic, it is possible to let useForm hook do that automatically for you by combining useForm
and <Form />
component.
- import
useForm
- import
Form
- just attach
bindUseForm
toForm
import React from 'react';
import { useForm, Form } from 'react-minimalistic-use-form';
export const MyForm = () => {
const {
isFormValid, onSubmit, errors, bindUseForm
} = useForm({ initialValues: { field1: 'foo', field2: 'bar'} });
const submitForm = ({ event, values, errors, isFormValid }) => {
event.preventDefault();
/* do some logic here... */
};
return (
<Form onSubmit={onSubmit(submitForm)} noValidate bindUseForm={bindUseForm}>
<div>
<label htmlFor="field1">Field1</label>
<input id="field1" name="field1" type="text" required />
{errors.field1 && Object.values(errors.field1).map((error, index) => (
<span key={index} className="text-error">
{error}
<br />
</span>
))}
</div>
<div>
<label htmlFor="field2">Field2</label>
<input id="field2" name="field2" type="text" required minLength={4} />
{errors.field2 && Object.values(errors.field2).map((error, index) => (
<span key={index} className="text-error">
{error}
<br />
</span>))}
</div>
<div>
<button type="submit" disabled={isFormValid === false}>
<span>Start!</span>
</button>
</div>
</Form>
);
}
Form component needs bindUseForm
prop from useForm hook otherwise an Error will be thrown.
Form will automatically bind input values
and onChange
, onBlur
eventHandlers that will handle form validation properly.
NOTE: Form input elements must have id
and name
attributes set, otherwise Form
will not consider it as an
input field, hence it won't attach needed event handlers.
Either using just useForm hook or in a combination with Form component, useForm hook will handle validation automatically
for you. Optionally validateOnInput (boolean)
or validateOnSubmit (boolean)
can be configured.
In first case, on user input, useForm will automatically validate that field, update errors
object properly and
it will toggle error className to input field. To provide the best user experience once input field is blurred,
useForm onBlur eventHandler will set className is-touched
(default value can be configured) and it will
trigger input validation. If field has validity: false
an error className has-error
(default value, can be configured)
will be attached to input field. As soon as field validity is valid, error class is removed from the field.
errors
object contains all form field errors in format:
errors: {
[fieldName]: {
[validityName]: 'Error message',
[otherValidityName]: 'Another error message'
}
}
Validity names are native html5 ValidityState interface properties. It's up to consumer how error messages will be displayed/styled/translated...
Custom controlled inputs
It's fully possible to have custom controlled inputs either by just using useForm hook or
with <Form />
component as well. Only difference is that useForm hook without Form component
requires onChange and onBlur handlers to be called in order to track form internal state and validation,
where Form component automatically binds those so custom controlled inputs don't need
to attach them additionally.
Here are both examples.
import React, { useState} from 'react';
import { useForm, Form } from 'react-minimalistic-use-form';
export const MyForm = () => {
const {
isFormValid, onChange, onBlur, onSubmit, formRef
} = useForm();
const [controlledField1, setControlledField1] = useState("field1");
const _onChange = event => {
setControlledField1(event.target.value);
onChange(event); // useForm onChange needs to be called!
};
const _onBlur = event => {
/* some logic goes here */
onBlur(event); // useForm onBlur needs to be called!
};
const submitForm = ({ event, values, errors, isFormValid }) => {
event.preventDefault();
/* do some logic here... */
};
return (
<form onSubmit={onSubmit(submitForm)} noValidate ref={formRef}>
<div>
<label htmlFor="field1">Field1</label>
<input id="field1" name="field1" type="email" value={controlledField1} onChange={_onChange} onBlur={_onBlur} />
</div>
<div>
<button type="submit" disabled={isFormValid === false}>
<span>Start!</span>
</button>
</div>
</form>
);
}
import React, { useState} from 'react';
import { useForm, Form } from 'react-minimalistic-use-form';
export const MyForm = () => {
const {
isFormValid, onSubmit, bindUseForm
} = useForm();
const [controlledField1, setControlledField1] = useState("field1");
const _onChange = event => {
setControlledField1(event.target.value);
};
const submitForm = ({ event, values, errors, isFormValid }) => {
event.preventDefault();
/* do some logic here... */
};
return (
<Form onSubmit={onSubmit(submitForm)} noValidate bindUseForm={bindUseForm}>
<div className="col-12 mt-3">
<label htmlFor="field1" className="m-0 mb-1 p-0">Field1</label>
<input id="field1" name="field1" type="email" value={controlledField1} onChange={_onChange} />
</div>
<div className="col-12">
<button type="submit" className="mt-3 mb-0" disabled={isFormValid === false}>
<span>Start!</span>
</button>
</div>
</Form>
);
}
API
useForm
useForm accepts configuration object with following options:
property | description | type | default |
---|---|---|---|
initialValues |
Form initial values | object | {} |
errorClassName |
Input field error class name | string | "has-error" |
touchedClassName |
Input field "touched" class name | string | "is-touched" |
scrollToError |
Scroll to form input field that has an error. | boolean | false |
scrollToErrorOptions |
Scroll to error options | object | undefined |
validateOnInput |
Validate form on user input. | boolean | true |
validateOnSubmit |
Validate form on submit. | boolean | false |
validateOnMount |
Validate form on mount. | boolean | false |
debounceValidation |
Debounce input validation. | boolean | false |
debounceTime |
Debounce input validation timeout in milliseconds. | boolean | 300 |
plugins |
Plugins to opt-in with custom logic. | object | {} |
Scroll to Error
-
scrollToError - If enabled useForm will scroll to element in case input has an error. It will try to find input label to scroll to or if no label found it will scroll to input field. By default useForm is using
Element.ScrollIntoView()
with providedscrollToErrorOptions
; -
scrollToErrorOptions - options to be passed in as an argument to
Element.ScrollIntoView()
. -
What options are supported and how scrollIntoView works, which browsers are supported etc. check out official documentation on MDN - Element.ScrollIntoView()
Plugins
Use Form offers possibility to opt-in via plugins
prop and change default useForm capabilities.
Currently supported plugins:
scrollToError
-
scrollToError
- scrolltoError plugin is a function that will be called on input field error. useForm calls a scrollToError plugin function with one argument, the DOM label element (if one found) or input element itself. If scrollToError plugin is provided the defaultElement.ScrollIntoView()
is overridden and hence ignored.
validator
-
validator
- validator plugin is a function that will be called upon onChange event. -
function will be called with object
{name, value, values, target}
where name is the name of the current input field,value
is the value of the current field,values
is an object with all form field values andtarget
is the input DOM element. -
validator function should return an object with errors in format:
{
fieldName: {
errorKey: 'Error message',
anotherErrorKey: 'Another Error message'
}
}
Example:
const validator = ({name, value, values }) => {
const errors = {};
if ((name === 'password_confirm' && value !== values.password) || (name === 'password' && value !== values.password_confirm)) {
errors.password_confirm = {
passwordMismatch: 'Passwords do not match!',
};
}
return errors;
};
useForm in return provides:
- resetForm -
function
- resets form to initial state. - onChange -
function
- event handler needed for handling validation and input state. - onBlur -
function
- event handler needed for handling form validation - onSubmit -
function
- Curry function that accepts callback as an argument. It returns an object with{ event, errors, values, isFormValid }
- validateForm -
function
- validates form and returnsisFormValid (boolean)
- isFormValid -
boolean
- Boolean describing form validity. - formRef - react useRef instance - *required to be attached to
form
- values -
object
- form values - errors -
object
- form errors - bindUseForm -
object
- *required to be attached to<Form />
to automatically bind formRef and event handlers - isSubmitting -
boolean
- Boolean flag describing is form being in submission state - setIsSubmitting -
function
- Function to ToggleisSubmitting
state
NOTE: isFormValid
onMount will be checked only against native html validation, input validity.
You can choose how to handle initial isFormValid state:
- by combining
validateOnInput
and manually triggeringvalidateForm
once your component is mounted - by just validating on submit
- by passing
validateOnMount
which will check native html validation + it will call custom validator if provided
Form Submission
- UseForm will set
isSubmitting
totrue
when form is submitted. - If
validateOnSubmit
istrue
it will run validation and updateisFormValid
anderrors
. - It Call your registered submission handler i.e.
onSubmit({ event, isFormValid, errors, values });
- You have to call
setIsSubmitting(false)
.
Form
Form accepts all html5 form attributes along with one required:
property | description | type |
---|---|---|
bindUseForm |
Object - contains formRef and eventHandlers | object *required |