Validplus is a validation enhancement library targeted at extending basic HTML DOM validation with smart features.
ValidPlus is designed to conform to the standard HTML5 validation spec for DOM Elements. This means, in cases where Javascript is disabled, validation will continue to function based on the browsers' validation implementation.
Importing (Module)
import { Validator, Fieldset, Field } from 'validplus'
or
import * as VP from 'validplus'
Importing (Browser)
<script src="node_modules/validplus/dist/validplus.browser.js">
Exports VP
to window
This usage pattern is designed to get you up and running quick without extra boilerplate.
All options are set using the handy vp-
DOM attribute bindings.
<form id="sample_form" vp-find>
<div class="VPFieldset" vp-find>
<div class="VPField">
<label for="sample_input">Sample Input</label>
<input id="sample_input" type="text" placeholder="Sample" required />
</div>
</div>
</form>
const Validator = new VP.Validator(document.getElementsById('sample_form'))
This usage pattern provides the most functionality but has additional boilerplate requirements.
<form id="sample_form">
<div id="sample_fieldset" class="VPFieldset">
<div id="sample_field" class="VPField">
<label for="sample_input">Sample Input</label>
<input id="sample_input" type="text" placeholder="Sample" required />
</div>
</div>
</form>
// Validator
const Validator = new VP.Validator(
document.getElementById('sample_form'), <options>)
// Fieldset
const Fieldset = new VP.Fieldset(
document.getElementsById('sample_fieldset'), <options>)
// Field
const Field = new VP.Field(
document.getElementsById('sample_field'), <options>)
// Associate
Fieldset.addField(Field);
Validator.addFieldset(Fieldset);
or
// Validator
const Validator = new VP.Validator(
document.getElementById('sample_form'), <options>)
// Fieldset
Validator.createFieldset(
document.getElementsById('sample_fieldset'), <options>, [
// Field
new VP.Field(
document.getElementsById('sample_field'), <options>)
]);
or
// Validator
const Validator = new VP.Validator(
document.getElementById('sample_form'), <options>)
// Fieldset
const Fieldset = Validator.createFieldset(
document.getElementsById('sample_fieldset'), <options>)
// Field
Fieldset.createField(
document.getElementsById('sample_field'), <options>)
Form submit validation check
// Submit Event
(e) => {
if (!Validator.isValid()) e.preventDefault();
}
or
// Submit Event
(e) => {
e.preventDefault()
Validator.isValid().then((isValid) => {
if (isValid) e.submit()
});
}
NOTE: When using async rules, you must interrupt form submission to await results.
- Lifecycle <ValidationLifecycle>
vp-valid
||vp-invalid
- Messages only- Lifecycle bindings to perform actions/messaging for Valid or Invalid outcomes
- ErrorClassName <string> ("-isError")
vp-error-class
- Classname appended when validation is erroneous
- ValidClassName <string> ("-isValid")
vp-valid-class
- Classname appended when validation is successful
- MessageClassName <string> ("VPMessage")
vp-msg-class
- Classname message nodes will bear
- MessageContainerClassName <string> ("VPMessages")
vp-msgs-class
- Classname message node container will bear
- MessageAnchor <HTMLElement> (self)
- An alternate anchor point for message nodes
- MessagePOS <"TOP"|"BOTTOM"> ("BOTTOM")
vp-msg-top
||vp-msg-bottom
- Message positioning; Determines how the messaging container is mounted to the anchor point
- ScrollAnchor <HTMLElement> (self)
- An anchor point to scroll to if scrolling to an instance
self.scrollTo()
- An anchor point to scroll to if scrolling to an instance
- ScrollOptions <ScrollIntoViewOptions|boolean> ({ behavior: 'smooth' })
- FindFieldsets <boolean> (false)
vp-find?
- Automatically parse children for Fieldsets, relying on DOM bindings for options
- FieldsetClass <string> ("VPFieldset")
vp-find
- Classname signifying a fieldset element
- ValidateLazy <boolean> (true)
vp-lazy
- Indicate child fieldsets should validate lazily
- ValidateVisible <boolean> (true)
vp-visibile
- Indicate if validation should be enforced for non-visible fieldsets
- FindFields <boolean> (false)
vp-find?
- Automatically parse children for Fields, using DOM bindings for options
- FieldClass <string> ("VPField")
vp-find
- Classname signifying a field element
- ValidateVisible <boolean> (true)
vp-visibile
- Indicate if validation should be enforced for non-visible fieldsets
- ValidationStrategy <string|function> ("all")
vp-strategy
- Strategy to validate child fields against each other. See Validation Option
- ScrollTo <boolean> (true)
vp-scroll
- Indicate if erroneous fieldsets should be scrolled to on error (first error)
- Notify <boolean> (true)
vp-notify
- Notify parent of changes, allowing siblings to validate together.
- DirtyOn <ChangeActions> ({ input: true })
vp-dirty-blur
||vp-dirty-input
||vp-dirty-change
||vp-dirty-mouseleave
- Action state required to mark a field as dirty (can validate)
- FormatOn <ChangeActions> ({ input: true })
vp-format-blur
||vp-format-input
||vp-format-change
||vp-format-mouseleave
- Action state required to trigger field formatting
- ValidateOn <ChangeActions> ({ blur: true })
vp-blur
||vp-input
||vp-change
||vp-mouseleave
- Action state required to trigger validation
- ForceRules <boolean> (false)
- Force rules defined via options to supersede DOM rules
- InputRules <HTMLValidationRules>
- Standard HTML Validation rules to apply to the field input
- CustomRules <CustomValidationRule[]>
- Custom Validation rules to apply. See Custom Validation
- InputFormatter <InputFormatters>
- Formatters to apply to input value. See Input Formatter
- ValidateAsync <boolean> (false)
vp-async
- Force validation to resolve async, regardless of the presence of async custom rules
- ValidateAsyncResolve <boolean> (true)
vp-await
- Await validation results before allowing another validation cycle.
- PrimaryInput <HTMLInputElement|HTMLSelectElement|HTMLTextAreaElement> (auto)
- Indicate the input we will validate against
- PrimaryInputIndex <number> (0)
- If multiple inputs exist within a field, indicate which should be validated against
- PrimaryInputType <string>
- The type of input we should prioritize picking from if multiple eligible inputs of different types are found
- InputTypes <string> ('select'|'input'|'textarea')[]
- The node types to consider when looking for child inputs.
- ShowFieldRuleErrors <boolean> (false)
vp-field-errors
- Display error results for standard field rules. See Standard Validation
- ShowCustomRuleErrors <boolean> (true)
vp-custom-errors
- Display custom error results for custom field rules. See Custom Validation
- ValidateLazyFieldRules <boolean> (true)
vp-lazy-fields
||vp-lazy
- Validate standard field rules lazily.
- ValidateLazyCustomRules <boolean> (true)
vp-lazy-custom
||vp-lazy
- Validate custom field rules lazily.
All validatable instances can define additional lifecycle messages and hooks to fire based on the outcome of validation.
ValidationLifeycle is defined as:
ValidationLifecycle: {
Valid: {
Message: '',
CB: [
(instance) => {
// Some action, such as commit to store,
// or save to localstorage state.
}
],
},
Invalid: {
Message: 'Field is invalid!',
CB: []
}
}
Standard validation emulates the regular validation a modern web browser will do on basic HTML inputs.
This validation includes the following properties:
- required - Value must be defined
- min - Cannot go below this value
- max - Cannot exceed this value
- minlength - Must be at least this length
- maxlength - Must be at most this length
- pattern - Must match this pattern
Inputs of type email
will be enforced to match a standard email format .+@.+\..+
. If pattern is defined,
email input value will also be validated against the pattern defined. Allowing for further customization
such as required domain .+@(validplus)\.com
.
You may specify an additional message to display for this pattern failure via the title
attribute.
<input type="email" pattern=".+@(validplus)\.com" title="A validplus email account is required." />
Inputs of type date
will enforce date values using the min
and max
attributes.
<input type="date" min="1970-01-01" />
Custom validation allows for additional validation rules using custom logic. These rules can be both synchronous or asynchronous.
CustomRules: [
(attrs, el, input) =>
+attrs.value % 2 === 0
? 'Value must be divisible by 2'
: true
]
or
CustomRules: [
async (attrs, el, input) => {
const result = await API.post('/some/validation/check', attrs.value)
return !result
? 'Value is already in use, please try another value.'
: true
}
]
Custom rules may return a boolean to indicate validity, or a string to provide a failure message.
NOTE: if using async rules and not enforcing AsyncResolved, isValid may return
a boolean
or a Promise<boolean>
, depending on whether the async rule is resolved or not.
Input formatters allow for formatting a value for pretty output to users. Common use cases include phone number masks or enforcing maximum value lengths.
InputFormatter: {
pre: (value) => value.replace(/[^0-9]/g, ''),
post: (value) => {
const areaCode = value.substr(0, 3);
const local = value.substr(3, 3);
const number = value.substr(6, 4);
let mask = '(';
if (areaCode.length > 0) mask += areaCode;
if (local.length > 0) mask += ') ' + local;
if (number.length > 0) mask += '-' + number;
return mask;
}
}
or
InputFormatter: {
pre: (value) => value.replace(/[^0-9]/g, '').substr(0, 5)
}
In addition to the formatter events, Input formatters are also called on validation. It is your responsibility to prepare the value to be parsed for validation correctly (strip masks, for example).
Pre is called prior, Post is called following validation.
Fieldset validation is determined using a validation strategy for child fieldset. By default, we define four validation strategies which can be referenced by name.
all
- All child fields must be validsome
- Some of the child fields must be validone
- One of the child fields must be validnone
- None of the child fields can be valid
You may also optionally define your own validation strategy for complex situations.
(statuses, fields) => <some logic to determine validity>
This strategy must return true/false.
See Docs
ValidPlus targets IE10+.
This version targets a minimum bundle size, meaning it is the responsibility of the consuming project to include the proper polyfills via core-js and babel.
This version exports directly to the window and is intended to be used via a script tag. This version maximizes compatibility and includes all the necessary shims to run properly in IE10+.
NOTE: It is highly recommended you utilize the standard version to minimize package size
The library comes in at:
Core validation library
|----------------------|
| Raw | GZip | Brotli |
|------|------|--------|
| 52kb | 14kb | 13kb |
|------|------|--------|
Browser shimmed
|-----------------------|
| Raw | GZip | Brotli |
|-------|------|--------|
| 111kb | 32kb | 28kb |
|-------|------|--------|
Test coverage is an ongoing focus for this library. Pull requests must include test coverage in order to be considered.
All existing tests must pass in order for a PR to be merged