Cell validator
Cell validators run when a user finishes editing a cell. Use them to enforce data rules such as required fields, numeric ranges, or pattern matching.
Overview
When you create a validator, assign it an alias so you can reference it by name in column configuration. Handsontable defines 5 aliases by default:
autocompleteforHandsontable.validators.AutocompleteValidatordateforHandsontable.validators.DateValidatordropdownforHandsontable.validators.DropdownValidatornumericforHandsontable.validators.NumericValidatortimeforHandsontable.validators.TimeValidator
Aliases give you a convenient way to specify which validator runs when table validation triggers. You don’t need to reference the validator function directly, and you can swap the function behind an alias without changing your column configuration.
Register custom cell validator
To register your own alias use Handsontable.validators.registerValidator() function. It takes two arguments:
validatorName- a string representing a validator functionvalidator- a validator function that will be represented byvalidatorName
If you’d like to register creditCardValidator under alias credit-card you have to call:
Handsontable.validators.registerValidator('credit-card', creditCardValidator);Choose aliases wisely. If you register your validator under name that is already registered, the target function will be overwritten:
Handsontable.validators.registerValidator('date', creditCardValidator);Now ‘date’ alias points to creditCardValidator function, not Handsontable.validators.DateValidator.
So, unless you intentionally want to overwrite an existing alias, try to choose a unique name. A good practice is prefixing your aliases with some custom name (for example your GitHub username) to minimize the possibility of name collisions. This is especially important if you want to publish your validator, because you never know aliases has been registered by the user who uses your validator.
Handsontable.validators.registerValidator('credit-card', creditCardValidator);Someone might already registered such alias.
Handsontable.validators.registerValidator('my.credit-card', creditCardValidator);That’s better.
Using an alias
The final touch is to use the registered aliases, so that you can easily refer to them without knowing the actual validator function.
To sum up, a well prepared validator function should look like this:
(Handsontable => { function customValidator(query, callback) { // ...your custom logic of the validator
callback(/* Pass `true` or `false` based on your logic */); }
// Register an alias Handsontable.validators.registerValidator('my.custom', customValidator);
})(Handsontable);From now on, you can use customValidator like so:
const container = document.querySelector('#container')const hot = new Handsontable(container, { columns: [{ validator: 'my.custom' }]});Full featured example
Use the validator method to easily validate synchronous or asynchronous changes to a cell. If you need more control, beforeValidate and afterValidate hooks are available. In the below example, email_validator_fn is an async validator that resolves after 1000 ms.
Use the allowInvalid option to define if the grid should accept input that does not validate. If you need to modify the input (e.g., censor bad words, uppercase first letter), use the plugin hook beforeChange.
By default, all invalid cells are marked by htInvalid CSS class. If you want to change class to another you can basically add the invalidCellClassName option to Handsontable settings. For example:
For the entire table
invalidCellClassName: 'myInvalidClass'For specific columns
columns: [ { data: 'firstName', invalidCellClassName: 'myInvalidClass' }, { data: 'lastName', invalidCellClassName: 'myInvalidSecondClass' }, { data: 'address' }]Callback console log:
import Handsontable from 'handsontable/base';import { registerAllModules } from 'handsontable/registry';
// Register all Handsontable's modules.registerAllModules();
const container = document.querySelector('#example1');const output = document.querySelector('#output');const ipValidatorRegexp = /^(?:\b(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\b|null)$/;
const emailValidator = (value, callback) => { setTimeout(() => { if (/.+@.+/.test(value)) { callback(true); } else { callback(false); } }, 1000);};
new Handsontable(container, { data: [66 collapsed lines
{ id: 1, name: { first: 'Joe', last: 'Fabiano' }, ip: '0.0.0.1', email: 'Joe.Fabiano@ex.com', }, { id: 2, name: { first: 'Fred', last: 'Wecler' }, ip: '0.0.0.1', email: 'Fred.Wecler@ex.com', }, { id: 3, name: { first: 'Steve', last: 'Wilson' }, ip: '0.0.0.1', email: 'Steve.Wilson@ex.com', }, { id: 4, name: { first: 'Maria', last: 'Fernandez' }, ip: '0.0.0.1', email: 'M.Fernandez@ex.com', }, { id: 5, name: { first: 'Pierre', last: 'Barbault' }, ip: '0.0.0.1', email: 'Pierre.Barbault@ex.com', }, { id: 6, name: { first: 'Nancy', last: 'Moore' }, ip: '0.0.0.1', email: 'Nancy.Moore@ex.com', }, { id: 7, name: { first: 'Barbara', last: 'MacDonald' }, ip: '0.0.0.1', email: 'B.MacDonald@ex.com', }, { id: 8, name: { first: 'Wilma', last: 'Williams' }, ip: '0.0.0.1', email: 'Wilma.Williams@ex.com', }, { id: 9, name: { first: 'Sasha', last: 'Silver' }, ip: '0.0.0.1', email: 'Sasha.Silver@ex.com', }, { id: 10, name: { first: 'Don', last: 'Pérignon' }, ip: '0.0.0.1', email: 'Don.Pérignon@ex.com', }, { id: 11, name: { first: 'Aaron', last: 'Kinley' }, ip: '0.0.0.1', email: 'Aaron.Kinley@ex.com', }, ], beforeChange(changes) { for (let i = changes.length - 1; i >= 0; i--) { const currChange = changes[i];
if (!currChange) { return false; }
// gently don't accept the word "foo" (remove the change at index i) if (currChange[3] === 'foo') { changes.splice(i, 1); } // if any of pasted cells contains the word "nuke", reject the whole paste else if (currChange[3] === 'nuke') { return false; } // capitalise first letter in column 1 and 2 else if (currChange[1] === 'name.first' || currChange[1] === 'name.last') { if (currChange[3] !== null) { changes[i][3] = currChange[3].charAt(0).toUpperCase() + currChange[3].slice(1); } } }
return true; }, afterChange(changes, source) { if (source !== 'loadData') { output.innerText = JSON.stringify(changes); } }, colHeaders: ['ID', 'First name', 'Last name', 'IP', 'E-mail'], height: 'auto', licenseKey: 'non-commercial-and-evaluation', columns: [ { data: 'id', type: 'numeric' }, { data: 'name.first' }, { data: 'name.last' }, { data: 'ip', validator: ipValidatorRegexp, allowInvalid: true }, { data: 'email', validator: emailValidator }, ], autoWrapRow: true, autoWrapCol: true,});import Handsontable from 'handsontable/base';import { registerAllModules } from 'handsontable/registry';
// Register all Handsontable's modules.registerAllModules();
const container = document.querySelector('#example1')!;const output = document.querySelector('#output') as HTMLElement;
const ipValidatorRegexp = /^(?:\b(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\b|null)$/;
const emailValidator = (value: string, callback: (value: boolean) => void) => { setTimeout(() => { if (/.+@.+/.test(value)) { callback(true); } else { callback(false); } }, 1000);};
new Handsontable(container, { data: [66 collapsed lines
{ id: 1, name: { first: 'Joe', last: 'Fabiano' }, ip: '0.0.0.1', email: 'Joe.Fabiano@ex.com', }, { id: 2, name: { first: 'Fred', last: 'Wecler' }, ip: '0.0.0.1', email: 'Fred.Wecler@ex.com', }, { id: 3, name: { first: 'Steve', last: 'Wilson' }, ip: '0.0.0.1', email: 'Steve.Wilson@ex.com', }, { id: 4, name: { first: 'Maria', last: 'Fernandez' }, ip: '0.0.0.1', email: 'M.Fernandez@ex.com', }, { id: 5, name: { first: 'Pierre', last: 'Barbault' }, ip: '0.0.0.1', email: 'Pierre.Barbault@ex.com', }, { id: 6, name: { first: 'Nancy', last: 'Moore' }, ip: '0.0.0.1', email: 'Nancy.Moore@ex.com', }, { id: 7, name: { first: 'Barbara', last: 'MacDonald' }, ip: '0.0.0.1', email: 'B.MacDonald@ex.com', }, { id: 8, name: { first: 'Wilma', last: 'Williams' }, ip: '0.0.0.1', email: 'Wilma.Williams@ex.com', }, { id: 9, name: { first: 'Sasha', last: 'Silver' }, ip: '0.0.0.1', email: 'Sasha.Silver@ex.com', }, { id: 10, name: { first: 'Don', last: 'Pérignon' }, ip: '0.0.0.1', email: 'Don.Pérignon@ex.com', }, { id: 11, name: { first: 'Aaron', last: 'Kinley' }, ip: '0.0.0.1', email: 'Aaron.Kinley@ex.com', }, ], beforeChange(changes) { for (let i = changes.length - 1; i >= 0; i--) { const currChange = changes[i];
if (!currChange) { return false; }
// gently don't accept the word "foo" (remove the change at index i) if (currChange[3] === 'foo') { changes.splice(i, 1); } // if any of pasted cells contains the word "nuke", reject the whole paste else if (currChange[3] === 'nuke') { return false; } // capitalise first letter in column 1 and 2 else if (currChange[1] === 'name.first' || currChange[1] === 'name.last') { if (currChange[3] !== null) { changes[i]![3] = currChange[3].charAt(0).toUpperCase() + currChange[3].slice(1); } } }
return true; }, afterChange(changes, source) { if (source !== 'loadData') { output.innerText = JSON.stringify(changes); } }, colHeaders: ['ID', 'First name', 'Last name', 'IP', 'E-mail'], height: 'auto', licenseKey: 'non-commercial-and-evaluation', columns: [ { data: 'id', type: 'numeric' }, { data: 'name.first' }, { data: 'name.last' }, { data: 'ip', validator: ipValidatorRegexp, allowInvalid: true }, { data: 'email', validator: emailValidator }, ], autoWrapRow: true, autoWrapCol: true,});<output class="console" id="output">Here you will see the log</output><div id="example1"></div>Edit the above grid to see the changes argument from the callback.
Mind that changes in table are applied after running all validators (both synchronous and and asynchronous) from every changed cell.
Related API reference
APIs
Configuration options
Core methods
Hooks
Result
You now have a cell validator that enforces data rules when a user finishes editing. Register it under an alias to reference it by name across your column configuration, and use allowInvalid: false to keep the editor open until the user enters a valid value.