Skip to content

Latest commit

 

History

History
722 lines (494 loc) · 21.9 KB

README.md

File metadata and controls

722 lines (494 loc) · 21.9 KB

Ustream jQuery form validator plugin (uvalidator)

  • Separate validation from displaying and handling errors.
  • Asynchronous validation support (ajax calls).
  • Pluggable validation methods.
  • Skin support.
  • Validating group of fields as one.

Usage

Basic usage after you included the following files:

  1. jquery.uvalidator.js
  2. jquery.uvalidator.messages.js
  3. jquery.uvalidator.rules.js
  4. jquery.uvalidator.skin.js
  5. jquery.uvalidator.skin.base.js
<form action="" method="" id="Form">
    <div class="control-group">
        <label for="RequireField">A required field</label>
        <input type="text" name="" value="" id="RequireField" required />
    </div>
</form>
var form, events, validator;

form = $("#Form");
events = $.uvalidator.events;

validator = $.uvalidatorApplySkin("base", form, {
    // validation events when should validate
    validationEvents: {
        change: true, // change event on fields
        submit: true // submit event of the form
    }
});

// catch the FORM_VALID event, which is fired after all validation ran on the
// form and all returned that the fields are valid.
validator.on(events.FORM_VALID, function () {
    form[0].submit();
});

Here you are applying the "base" skin on form which has the id 'Form'. When it comes out that the form is valid, the form is going to be submitted. If any field is invalid a a label will be appended after the input field with the error message.

If you don't want to use any skin, just call the uvalidator method on the form's jquery object.

var form, events;

form = $("#Form");
events = $.uvalidator.events;

form.uvalidator({
    validationEvents: {
        change: true,
        submit: true
    }
});

form.on(events.FORM_VALID, function () {
    form[0].submit();
});

It does the same like the one above, but it won't show any error message, because here we didn't use any skin.

Available options

  • validationEvents Object
    • focusout Boolean, default: false
    • focusin Boolean, default: false
    • change Boolean, default: true
    • keyup Boolean, default: true
    • click Boolean, default: false
    • submit Boolean, default: true

How it works

We can split the validator into 4 modules:

  • core
  • validator methods
  • skin
  • messages

The core module will listen to the events of the form and if the event is enabled for validation it will run the validation methods. The core also uses events to communicate with other code. All events are listed under the $.uvalidator.events namespace.

Validator events

The events listed under the $.uvalidator.events namespace:

  • START_FIELD_VALIDATION
  • FINISH_FIELD_VALIDATION
  • START_FORM_VALIDATION
  • FINISH_FORM_VALIDATION
  • FIELD_VALID
  • FIELD_INVALID
  • FORM_VALID
  • FORM_INVALID

You can access to them like this: $.uvalidator.START_FIELD_VALIDATION

START_FIELD_VALIDATION

callback arguments:

  • field jQueryObject Field which going to be validated

It runs right before the field validation starts. Useful for example when you use a AJAX validation to show a loading indicator on the field.

validator.on($.uvalidator.events.START_FIELD_VALIDATION, function (event, field) {
    $(field).addClass('loading');
});

FINISH_FIELD_VALIDATION

callback arguments: = field jQueryObject Field which has been validated

It runs right after all of the validation methods ran on the field (ajax validations too). For example you can remove the loading indicator from the field

validator.on($.uvalidator.events.FINISH_FIELD_VALIDATION, function (event, field) {
    $(field).removeClass('loading');
});

START_FORM_VALIDATION

callback arguments:

  • event 'jQueryEventObject` Standard jquery event object
  • form HTMLFormElement The form, which going to be validated

Runs right before a full form validation starts.

validator.on($.uvalidator.events.START_FORM_VALIDATION, function (event, form) {
    $(form).addClass('loading');
});

FINISH_FORM_VALIDATION

callback arguments:

  • event 'jQueryEventObject` Standard jquery event object
  • form HTMLFormElement The form, which has been validated

Called right after all field has been validated in the form (including ajax calls).

validator.on($.uvalidator.events.FINISH_FORM_VALIDATION, function (event, form) {
    $(form).removeClass('loading');
});

FIELD_VALID

callback arguments:

  • event 'jQueryEventObject` Standard jquery event object
  • result Object
    • field jQueryObject The field, which has been validated
    • isGroup Boolean True, if the field belongs to a group, so only groupValidation methods were running
    • isValid Boolean True, if the field is valid
    • validator String Name of the validator which has been called last time. (Mostly useful in invalid callback, but for the consistency it passed here too).

Called when all validation ran for a field and all validation passed. For example using that event you can mark that the field is valid.

validator.on($.uvalidator.events.FINISH_FORM_VALIDATION, function (event, result) {
    $(form).removeClass('loading');
});

When all validation ran for the field and all found out that the field is valid, the FIELD_VALID event will be triggered. You can use this event for example to mark if a field is valid adding a class to it.

validator.on($.uvalidator.events.FIELD_VALID, function (event, result) {
    result.field.addClass('valid');
});

FIELD_INVALID

callback arguments:

  • event 'jQueryEventObject` Standard jquery event object
  • result Object
    • field jQueryObject The field, which has been validated
    • isGroup Boolean True, if the field belongs to a group, so only groupValidation methods were running
    • isValid Boolean True, if the field is valid
    • validator String Name of the validator which has been called last time. Pretty useful if field is invalid, because you use this name to display a proper error message.

When one of the validators of the field founds that something is wrong with the value of the field, the FIELD_INVALID event will be triggered.

validator.on($.uvalidator.events.FIELD_INVALID, function (event, result) {
    result.field.addClass('invalid');
});

FORM_VALID

callback arguments:

  • event 'jQueryEventObject` Standard jquery event object
  • validationResults
    • results Array Result object array, see FIELD_VALID event for result object structure.

The event triggered when all validator of all fields ran and all field is valid.

FORM_INVALID

callback arguments:

  • event 'jQueryEventObject` Standard jquery event object
  • result Object
    • field jQueryObject The field, which has been validated
    • isGroup Boolean True, if the field belongs to a group, so only groupValidation methods were running
    • isValid Boolean True, if the field is valid
    • validator String Name of the validator which has been called last time. Pretty useful if field is invalid, because you use this name to display a proper error message.

The event triggered when all validator of all fields ran and at lease one field was invalid.

Validator methods

Group validator methods

Group validator methods are special methods. You need group validator when you need to validate two more field together, having only one FIELD_VALID or FIELD_INVALID event for all elements. An example could be a credit card expiration date field, where you have a month and a year field. A month field alone can not be valid, without the year you don't know if the credit card is expired.

To define a validator group you must use the data-validator-group attribute. All field, which belongs to one group must have the same value in that attribute

<input type="text" name="groupItem1" value="" data-validator-group="group1" />
<input type="text" name="groupItem2" value="" data-validator-group="group1" />
<input type="text" name="groupItem3" value="" data-validator-group="group2" />
<input type="text" name="groupItem4" value="" data-validator-group="group2" />

Here we defined two groups (group1 and group2). groupItem1 and groupItem2 belongs to group1, groupItem3 and groupItem4 belongs to group2.

Applying validator methods on a field

A validator method will run on a field if the jQuerySelector, which defined in the validatator method matches with the input. The plugin will search for the fields under the form's children. For example, if you want to make sure that the field will be filled, you must use the required validator. The required validator is checking with the following selector: [required],.required. It means that either the field must have a require attribute or a required class name.

Example using a class:

<input type="text" name="aName" value="" class="required" />

Example using attribute:

<input type="text" name="aName" value="" required />

Predefined validator methods

required

Checks if a field's value isn't empty. If the field is a checkbox, it will check if it's checked or not. For radio fields see required group method

selector

[required],.required

examples
<input type="text" name="aName" value="" required />
<input type="checkbox" name="aName" value="" required />
<input type="text" name="aName" value="" class="required" />
<input type="checkbox" name="aName" value="" class="required" />

number

Checks if a field's value is a number or not. If the required validator rule not set, it can be empty.

selector

.number,[type="number"]

examples
<input type="number" name="aName" value="" />
<input type="text" name="aName" value="" class="number"/>

url

Checks if a field's value's format is a valid url. It uses regular expression to check the format. If the required validator rule not set, it can be empty.

selector

.input-url,[type="url"]

examples
<input type="url" name="aName" value="" />
<input type="text" name="aName" value="" class="url"/>

email

Checks if a field's value's format is a valid email. It uses regular expression to check the format. If the required validator rule not set, it can be empty.

selector

.input-email,[type="email"]

examples
<input type="email" name="aName" value="" />
<input type="text" name="aName" value="" class="email"/>

userpassword

Tests if the field's value is matching with a predefined pattern. The predefined pattern requires that the password must be at least 5 characters length, must contain at least a lowercase alpha, a uppercase alpha and a numeric character.

selector

.userpassword

example
<input type="password" name="aName" value="" class="userpassword" />

passwordverify

You can use this on registration forms for example, where the user needs to type his password twice to verify that he didn't mistyped something in it. Using this validator you need to set a data-refelem attribute to the element, which must be jQuerySelector what can identify the other element, which value must be the same with this element's value.

data attributes
  • data-refelem jQuerySelector Selector to the other element, which value must be compared to this element's value.
selector

.password-verify,.input-password-verify

example
<input type="password" name="userpassword" value="" id="UserPassword" class="userpassword"/>
<input type="password" name="userpasswordverify" value="" data-refelem="#UserPassword" />

min

not compatible with the HTML5 specification, needs some work, recommended to use with the class selector to not brake the HTML validator

Checks if the value of the input is a number and the number is greater or equals to the defined minimum value. If the required validator rule not set, it can be empty.

To define a minimum value there are two ways:

  • If you use the class selector (input-min), then you must use a data attribute: data-validator-min="5"
  • If you use the min attribute, then it's value must be the minimum number: min="5"
data attributes
  • data-validator-min Number Minimum value must be in the input.
selector

.input-min,[min]

examples
<input type="text" name="aName" value="" class="input-min" data-validator-min="5" />
<input type="text" name="aName" value="" min="5" />

max

not compatible with the HTML5 specification, needs some work, recommended to use with the class selector to not brake the HTML validator

Checks if the value of the input is a number and the number is less or equals to the defined maximum value. If the required validator rule not set, it can be empty.

To define a maximum value there are two ways:

  • If you use the class selector (input-max), then you must use a data attribute: data-validator-max="5"
  • If you use the max attribute, then it's value must be the maximum number: max="5"
data attributes
  • data-validator-max Number Maximum value must be in the input.
selector

.input-max,[max]

examples
<input type="text" name="aName" value="" class="input-max" data-validator-max="5" />
<input type="text" name="aName" value="" max="5" />

creditcard

Validates using the Luhn algorithm (http://en.wikipedia.org/wiki/Luhn_algorithm) if the value is a valid credit card.

selector

.creditcard,.input-creditcard

examples
<input type="text" name="aName" value="" class="creditcard" />
<input type="text" name="aName" value="" class="input-creditcard" />

pattern

Checks if the value is matching with the regular expression pattern which is defined in a input's attribute.

The attribute name must be a data-validator-pattern attribute if you use the class name selector. If you use the pattern attribute, it's value must be the pattern.

selector

.pattern,.input-pattern,[pattern]

examples
<input type="text" name="aName" value="" class="pattern" data-validator-pattern="^[a-z]+$" />
<input type="text" name="aName" value="" class="input-pattern" data-validator-pattern="^[a-z]+$" />
<input type="text" name="aName" value="" pattern="^[a-z]+$" />

Predefined group validators

required

That validator is pretty similar to the one above, but we use it when we in a radio group at least on of the radios must be selected.

selector

:radio[required],:radio.required

examples
<input type="radio" name="bar" value="1" data-validator-group="group1" id="radio-1" required />
<input type="radio" name="bar" value="2" data-validator-group="group1" id="radio-2" required />
<input type="radio" name="bar" value="1" data-validator-group="group1" id="radio-1" class="required" />
<input type="radio" name="bar" value="2" data-validator-group="group1" id="radio-2" class="required" />

required-date

Needs clarification, a more general name

Checks if every field has value of the group. The group must have the data-validator-type="date" attribute.

data-attributes
  • data-validator-type "date"
selector

:input[data-validator-type="date"][required],:input[data-validator-type="date"].required

examples
<input type="text" name="bar" value="1" data-validator-group="group1" data-validator-type="date" id="radio-1" class="required" />
<input type="text" name="bar" value="2" data-validator-group="group1" data-validator-type="date" id="radio-2" class="required" />
<input type="text" name="bar" value="1" data-validator-group="group1" data-validator-type="date" id="radio-1" required />
<input type="text" name="bar" value="2" data-validator-group="group1" data-validator-type="date" id="radio-2" required />

expiration-date

Useful for credit card expiration date validation. The date must be greater or equals to the current date (year and month). Not more and not less than two inputs must belong to a group. Each input must have a data-validator-ccexp attribute. For the year field the value must be year, for month field the value must be month.

data attributes
  • data-validator-ccexp "year"|"month"
selector

:input[data-validator-type="date"].cc-expiration

examples
<input type="text" name="month" value="" class="cc-expiration" data-validator-ccexp="month" data-validator-group="cc-expiration-date" />
<input type="text" name="year" value="" class="cc-expiration" data-validator-ccexp="year" data-validator-group="cc-expiration-date" />

Defining validator methods

You can see examples in the jquery.uvalidator.rules.js file, but here is a simple one, which checks if the entered value is a alphanumeric or not:

$.uvalidator.addMethod(
	// defining the selector, to know which elements must be validated with this method
	'.alphanumeric',
	// defining the name of the validator
	'alphanumeric',
	// defining the validator method
	function (value, element, callback) {
		// If the field is not required, empty value is allowed;
		// $.uvalidator.isOptional is only available if the `jquery.uvalidator.rules.js` is included
		var valid = $.uvalidator.isOptional(element, value);

		// if filled, check for if matches with the regexp
		if (!valid) {
			valid = /[a-z0-9]/i.test(value);
		}

		// call the callback with the validation result
		callback(valid);
	}
);

The HTML code to use this validator method:

<input type="text" name="name" value="" class="alphanumeric" />

Defining group validator method

When you want to validate two or more fields as one, you will need to define validator method. A good example is the credit card expiration date.

// use the addGroupMethod to define group validator
$.uvalidator.addGroupMethod(
	// defining the selector, to know which elements must be validated with this method
	':input[data-validator-type="date"].cc-expiration',
	// name of the validator method
	'expiration-date',
	// the validator function
	function (value, items, callback) {
		// getting the year field
		var year = items.filter('[data-validator-ccexp="year"]'),
		// getting the month field
			month = items.filter('[data-validator-ccexp="month"]'),
			yearVal = year.val(),
			// zeroPad is defined earlier, converts 5 to 05, 6 to 06 and so on
			monthVal = zeroPad(month.val(), 2),
			now = new Date(),
			thisYear,
			thisMonth,
			isValid;

		thisYear = String(now.getFullYear());
		thisMonth = zeroPad(now.getMonth() + 1, 2);

		if (yearVal.length === 2) {
			thisYear = thisYear.slice(-2);
		}

		// the callback must be called with one boolean argument, which should
		// be true if the fields are valid.
		callback(+(thisYear + thisMonth) <= +(yearVal + monthVal));
	}
);

The HTML code to use this validator method:

<input type="text" name="month" class="cc-expiration" data-validator-group="expiration-date" data-validator-ccexp="month" value="" />
<input type="text" name="year" class="cc-expiration" data-validator-group="expiration-date" data-validator-ccexp="year" value="" />

Skins

When you validate a form, you probably want to show the validation results somehow. The way how you would display error messages can be different on every site. So the plugin itself doesn't show any error messages, instead of it makes it easy to be notified about errors and you can decide it how to display the error.

We have built a helper which makes it easier to create your own skin.

To create a new skin, call the $.uvalidatorSkin method.

$.uvalidatorSkin('skin-name', {
		// here you must define your skin behavior
});

You can apply the newly created skin on a form by calling the $.uvalidatorApplySkin('skin-name', $('#Form')).

There are a bunch of predefined methods what you can use to define the behavior of your skin:

showFieldError Must append and display the error message for a field.

setFieldInvalid Must add styles to mark somehow that the field is invalid.

setFieldValid Must add styles to mark somehow that the field is valid.

hideFieldError Removes/hides error message of a field.

onFieldValidationStart Marks a field that validation is in progress on it.

onFieldValidationFinish Removes the mark of the field which shows that validation is in progress.

onFormValidationStart Marks that the form validation is in progress.

onFormValidationFinish Removes the mark of form which shows that the form validation is in progress.

onFormValid Callback, which called when form valid event triggereed.

onFormInvalid Callback, which called when form invalid event triggereed.

Skin helpers

$.uvalidatorSkin.format

Method to create a string where placeholders could be defined which then could be replaced to other values. The first argument must be the string with placeholders. The placeholder should be a number which should be wrapped into {} pair. It could be useful when you want to add custom error messages.

$.uvalidatorSkin.format('foo {0} baz {1}', 'bar', 'qux'); // returns "foo bar baz qux"

To build your own skin handler, you only need to subscribe to the events of the validator. Check the jquery.uvalidator.skin.js for ideas.

Running tests

npm test

Distribution files

  • jquery.uvalidator.min.js Only the validator file itself, minified by uglify js
    • jquery.uvalidator.js
  • jquery.uvalidator.skin.ustream.min.js:
    • jquery.uvalidator.js
    • jquery.uvalidator.rules.js
    • jquery.uvalidator.skin.js
    • jquery.uvalidator.skin.ustream.js
  • jquery.uvalidator.skin.ustream.messages.min.js
    • jquery.uvalidator.js
    • jquery.uvalidator.rules.js
    • jquery.uvalidator.skin.js
    • jquery.uvalidator.skin.ustream.js
    • jquery.uvalidator.messages.js

Messages not included into most of the packages, because the implementation may differ on different environment (localization).

TODO

These inconsitencies or strange behaviours found so far:

  • Fix min and max validators
  • More general required group validator