js.jquery-validity.js Maven / Gradle / Ivy
/*
* jQuery.validity v1.3.1
* http://validity.thatscaptaintoyou.com/
* https://github.com/whatgoodisaroad/validity
*
* Dual licensed under MIT and GPL
*
* Date: 2013-02-10 (Sunday, 10 February 2013)
*/
(function($, undefined) {
// Default settings:
///////////////////////////////////////////////////////////////////////////////
var
defaults = {
// The default output mode is tooltip because it requires no
// dependencies:
outputMode:"label",
// The this property is set to true, validity will scroll the browser
// viewport so that the first error is visible when validation fails:
scrollTo:false,
// If this setting is true, modal errors will disappear when they are
// clicked on:
modalErrorsClickable:true,
// If a field name cannot be otherwise inferred, this will be used:
defaultFieldName:"This field",
// jQuery selector to filter down to validation-supported elements:
elementSupport:":text, :password, textarea, select, :radio, :checkbox, input[type='hidden'],input[type='date'], input[type='time'],input[type='tel'],input[type='number'], input[type='email']",
// Function to stringify argments for use when generating error
// messages. Primarily, it just generates pretty date strings:
argToString:function(val) {
return val.getDate ? [
val.getMonth() + 1,
val.getDate(),
val.getFullYear()
].join("/") :
val + "";
},
debugPrivates:false
},
__private;
// Static functions and properties:
///////////////////////////////////////////////////////////////////////////////
$.validity = {
// Clone the defaults. They can be overridden with the setup function:
settings:$.extend(defaults, {}),
// Built-in library of format-checking tools for use with the match
// validator as well as the nonHtml validator:
patterns:{
integer:/^\d+$/,
// Used to use Date.parse(), which was the cause of Issue 9, where the
// function would accept 09/80/2009 as parseable. The fix is to use a
// RegExp that will only accept American Middle-Endian form. See the
// Internationalization section in the documentation for how to cause
// it to support other date formats:
date:/^((0?\d)|(1[012]))[\/-]([012]?\d|30|31)[\/-]\d{1,4}$/,
email:/^((([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+(\.([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+)*)|((\x22)((((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(([\x01-\x08\x0b\x0c\x0e-\x1f\x7f]|\x21|[\x23-\x5b]|[\x5d-\x7e]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(\\([\x01-\x09\x0b\x0c\x0d-\x7f]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]))))*(((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(\x22)))@((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?$/i,
usd:/^\$?((\d{1,3}(,\d{3})*)|\d+)(\.(\d{2})?)?$/,
url:/^(https?|ftp):\/\/(((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:)*@)?(((\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5]))|((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?)(:\d*)?)(\/((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)+(\/(([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)*)*)?)?(\?((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|[\uE000-\uF8FF]|\/|\?)*)?(\#((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|\/|\?)*)?$/i,
// Number should accept floats or integers, be they positive or
// negative. It should also support scientific-notation, written as a
// lower or capital 'E' followed by the radix. Number assumes base 10.
// Unlike the native parseFloat or parseInt functions, this should not
// accept trailing Latin characters.
number:/^[+-]?(\d+(\.\d*)?|\.\d+)([Ee]-?\d+)?$/,
zip:/^\d{5}(-\d{4})?$/,
phone:/^[2-9]\d{2}-\d{3}-\d{4}$/,
guid:/^(\{?([0-9a-fA-F]){8}-(([0-9a-fA-F]){4}-){3}([0-9a-fA-F]){12}\}?)$/,
time12:/^((0?\d)|(1[012])):[0-5]\d?\s?[aApP]\.?[mM]\.?$/,
time24:/^(20|21|22|23|[01]\d|\d)(([:][0-5]\d){1,2})$/,
nonHtml:/^[^<>]*$/
},
// Built-in set of default error messages (for use when a message isn't
// specified):
messages:{
require:"#{field} is required.",
// Format validators:
match:"#{field} is in an invalid format.",
integer:"#{field} must be a positive, whole number.",
date:"#{field} must be formatted as a date. (mm/dd/yyyy)",
email:"#{field} must be formatted as an email.",
usd:"#{field} must be formatted as a US Dollar amount.",
url:"#{field} must be formatted as a URL.",
number:"#{field} must be formatted as a number.",
zip:"#{field} must be formatted as a zipcode ##### or #####-####.",
phone:"#{field} must be formatted as a phone number ###-###-####.",
guid:"#{field} must be formatted as a guid like {3F2504E0-4F89-11D3-9A0C-0305E82C3301}.",
time24:"#{field} must be formatted as a 24 hour time: 23:00.",
time12:"#{field} must be formatted as a 12 hour time: 12:00 AM/PM",
// Value range messages:
lessThan:"#{field} must be less than #{max}.",
lessThanOrEqualTo:"#{field} must be less than or equal to #{max}.",
greaterThan:"#{field} must be greater than #{min}.",
greaterThanOrEqualTo:"#{field} must be greater than or equal to #{min}.",
range:"#{field} must be between #{min} and #{max}.",
// Value length messages:
tooLong:"#{field} cannot be longer than #{max} characters.",
tooShort:"#{field} cannot be shorter than #{min} characters.",
// Composition validators:
nonHtml:"#{field} cannot contain HTML characters.",
alphabet:"#{field} contains disallowed characters.",
minCharClass:"#{field} cannot have more than #{min} #{charClass} characters.",
maxCharClass:"#{field} cannot have less than #{min} #{charClass} characters.",
// Aggregate validator messages:
equal:"Values don't match.",
distinct:"A value was repeated.",
sum:"Values don't add to #{sum}.",
sumMax:"The sum of the values must be less than #{max}.",
sumMin:"The sum of the values must be greater than #{min}.",
// Radio validator messages:
radioChecked:"The selected value is not valid.",
generic:"Invalid."
},
// Abstract output:
out:{
start:function() {
this.defer("start");
},
end:function(results) {
this.defer("end", results);
},
raise:function($obj, msg) {
this.defer("raise", $obj, msg);
},
raiseAggregate:function($obj, msg) {
this.defer("raiseAggregate", $obj, msg);
},
defer:function(name) {
var
v = $.validity,
o = v.outputs[v.settings.outputMode];
o[name].apply(
o,
Array.prototype.slice.call(arguments, 1)
);
}
},
// Character classes can be used to determine the quantity of a given type
// of character in a string:
charClasses:{
alphabetical:/\w/g,
numeric:/\d/g,
alphanumeric:/[A-Za-z0-9]/g,
symbol:/[^A-Za-z0-9]/g
},
// Object to contain the output modes. The three built-in output modes are
// installed later on in this script.
outputs:{},
// By default, private members are not exposed.
__private:undefined,
// Override the default settings with user-specified ones.
setup:function(options) {
this.settings = $.extend(this.settings, options);
if (this.settings.debugPrivates) {
this.__private = __private;
}
else {
this.__private = undefined;
}
},
// Object to store information about ongoing validation. When validation
// starts, this will be set to a report object. When validators fail, they
// will inform this object. When validation is completed, this object will
// contain the information of whether it succeeded:
report:null,
// Determine whether validity is in the middle of validation:
isValidating:function() {
return !!this.report;
},
// Function to prepare validity to start validating:
start:function() {
// The output mode should be notified that validation is starting. This
// usually means that the output mode will erase errors from the
// document in whatever way the mode needs to:
this.out.start();
// Initialize the report object:
this.report = { errors:0, valid:true };
},
// Function called when validation is over to examine the results and
// clean-up:
end:function() {
// Null coalescence: fix for Issue 5:
var results = this.report || { errors: 0, valid: true };
this.report = null;
// Notify the current output mode that validation is over:
this.out.end(results);
return results;
},
// Remove validiation errors:
clear:function() {
this.start();
this.end();
}
};
// jQuery instance methods:
///////////////////////////////////////////////////////////////////////////////
$.fn.extend({
// The validity function is how validation can be bound to forms. The user
// may pass in a validation function or, as a shortcut, pass in a string of
// a CSS selector that grabs all the inputs to require:
validity:function(arg) {
return this.each(
function() {
// Only operate on forms:
if (this.tagName.toLowerCase() == "form") {
var f = null;
// If the user entered a string to select the inputs to
// require, then make the validation logic ad hoc:
if (typeof (arg) == "string") {
f = function() {
$(arg).require();
};
}
// If the user entered a validation function then just call
// that at the appropriate time:
else if ($.isFunction(arg)) {
f = arg;
}
if (arg) {
$(this).bind(
"submit",
function() {
$.validity.start();
f();
return $.validity.end().valid;
}
);
}
}
}
);
},
// Validators:
///////////////////////////////////////////////////////////////////////////
// Common validators:
///////////////////////////////////////////////////////////////////////////
// Validate whether the field has a value.
require:function(msg) {
return validate(
this,
function(obj) {
return !!$(obj).val().length;
},
msg || $.validity.messages.require
);
},
// Validate whether the field matches a regex.
match:function(rule, msg) {
// If a default message is to be used:
if (!msg) {
// First grab the generic one:
msg = $.validity.messages.match;
// If there's a more specific one, use that.
if (typeof (rule) === "string" && $.validity.messages[rule]) {
msg = $.validity.messages[rule];
}
}
// If the rule is named, rather than specified:
if (typeof (rule) == "string") {
rule = $.validity.patterns[rule];
}
return validate(
this,
// Some of the named rules can be functions, such as 'date'. If the
// discovered rule is a function use it as such. Otherwise, assume
// it's a RegExp.
$.isFunction(rule) ?
function(obj) {
return !obj.value.length || rule(obj.value);
} :
function(obj) {
// Fix for regexes where the global flag is set. Make sure
// to test from the start of the string.
if (rule.global) {
rule.lastIndex = 0;
}
return !obj.value.length || rule.test(obj.value);
},
msg
);
},
range:function(min, max, msg) {
return validate(
this,
min.getTime && max.getTime ?
// If both arguments are dates then use them that way.
function(obj) {
var d = new Date(obj.value);
return d >= new Date(min) && d <= new Date(max);
} :
min.substring && max.substring && Big ?
// If both arguments are strings then parse them using the
// Arbitrary-Precision library.
function(obj) {
var n = new Big(obj.value);
return (
n.greaterThanOrEqualTo(new Big(min)) &&
n.lessThanOrEqualTo(new Big(max))
);
} :
// Otherwise treat them like floats.
function(obj) {
var f = parseFloat(obj.value);
return f >= min && f <= max;
},
msg || format(
$.validity.messages.range, {
min:$.validity.settings.argToString(min),
max:$.validity.settings.argToString(max)
}
)
);
},
greaterThan:function(min, msg) {
return validate(
this,
min.getTime ?
function(obj) {
return new Date(obj.value) > min;
} :
min.substring && Big ?
function(obj) {
return new Big(obj.value).greaterThan(new Big(min));
} :
function(obj) {
return parseFloat(obj.value) > min;
},
msg || format(
$.validity.messages.greaterThan, {
min:$.validity.settings.argToString(min)
}
)
);
},
greaterThanOrEqualTo:function(min, msg) {
return validate(
this,
min.getTime ?
function(obj) {
return new Date(obj.value) >= min;
} :
min.substring && Big ?
function(obj) {
return new Big(obj.value).greaterThanOrEqualTo(new Big(min));
} :
function(obj) {
return parseFloat(obj.value) >= min;
},
msg || format(
$.validity.messages.greaterThanOrEqualTo, {
min:$.validity.settings.argToString(min)
}
)
);
},
lessThan:function(max, msg) {
return validate(
this,
max.getTime ?
function(obj) {
return new Date(obj.value) < max;
} :
max.substring && Big ?
function(obj) {
return new Big(obj.value).lessThan(new Big(max));
} :
function(obj) {
return parseFloat(obj.value) < max;
},
msg || format(
$.validity.messages.lessThan, {
max:$.validity.settings.argToString(max)
}
)
);
},
lessThanOrEqualTo:function(max, msg) {
return validate(
this,
max.getTime ?
function(obj) {
return new Date(obj.value) <= max;
} :
max.substring && Big ?
function(obj) {
return new Big(obj.value).lessThanOrEqualTo(new Big(max));
} :
function(obj) {
return parseFloat(obj.value) <= max;
},
msg || format(
$.validity.messages.lessThanOrEqualTo, {
max:$.validity.settings.argToString(max)
}
)
);
},
maxLength:function(max, msg) {
return validate(
this,
function(obj) {
return obj.value.length <= max;
},
msg || format(
$.validity.messages.tooLong, {
max:max
}
)
);
},
minLength:function(min, msg) {
return validate(
this,
function(obj) {
return obj.value.length >= min;
},
msg || format(
$.validity.messages.tooShort, {
min:min
}
)
);
},
alphabet:function(alpha, msg) {
var chars = [];
return validate(
this,
function(obj) {
// For each character in the string, ensure that it's in the
// alphabet definition:
for (var idx = 0; idx < obj.value.length; ++idx) {
if (alpha.indexOf(obj.value.charAt(idx)) == -1) {
chars.push(obj.value.charAt(idx));
return false;
}
}
return true;
},
msg || format(
$.validity.messages.alphabet, {
chars:chars.join(", ")
}
)
);
},
minCharClass:function(charClass, min, msg) {
if (typeof(charClass) == "string") {
charClass = charClass.toLowerCase();
if ($.validity.charClasses[charClass]) {
charClass = $.validity.charClasses[charClass];
}
}
return validate(
this,
function(obj) {
return (obj.value.match(charClass) || []).length >= min;
},
msg || format(
$.validity.messages.minCharClass, {
min:min,
charClass:charClass
}
)
);
},
maxCharClass:function(charClass, max, msg) {
if (typeof(charClass) == "string") {
charClass = charClass.toLowerCase();
if ($.validity.charClasses[charClass]) {
charClass = $.validity.charClasses[charClass];
}
}
return validate(
this,
function(obj) {
return (obj.value.match(charClass) || []).length <= max;
},
msg || format(
$.validity.messages.maxCharClass, {
max:max,
charClass:charClass
}
)
);
},
// Validate that the input does not contain potentially dangerous strings.
nonHtml:function(msg) {
return validate(
this,
function(obj) {
return $.validity.patterns.nonHtml.test(obj.value);
},
msg || $.validity.messages.nonHtml
);
},
// Aggregate validators:
///////////////////////////////////////////////////////////////////////////
// Validate that all matched elements bear the same values. Accepts a
// function to transform the values for testing.
equal:function(arg0, arg1) {
var
// If a reduced set is attached, use it. Also, remove unsupported
// elements.
$reduction = (this.reduction || this)
.filter($.validity.settings.elementSupport),
transform = function(val) {
return val;
},
msg = $.validity.messages.equal;
if ($reduction.length) {
// Figure out what arguments were specified.
if ($.isFunction(arg0)) {
transform = arg0;
if (typeof (arg1) == "string") {
msg = arg1;
}
}
else if (typeof (arg0) == "string") {
msg = arg0;
}
var
map = $.map(
$reduction,
function(obj) {
return transform(obj.value);
}
),
// Get the first value.
first = map[0],
valid = true;
// If any value is not equal to the first value, then they aren't
// all equal, and it's not valid.
for (var i in map) {
if (map[i] != first) {
valid = false;
}
}
if (!valid) {
raiseAggregateError($reduction, msg);
// The set reduces to zero valid elements.
this.reduction = $([]);
}
}
return this;
},
// Validate that all matched elements bear distinct values. Accepts an
// optional function to transform the values for testing.
distinct:function(arg0, arg1) {
var
// If a reduced set is attached, use it.
// Also, remove unsupported elements.
$reduction = (this.reduction || this).filter($.validity.settings.elementSupport),
transform = function(val) {
return val;
},
msg = $.validity.messages.distinct,
// An empty array to store already-examined values
subMap = [],
// An array with repeated values
repeatedVal = [],
valid = true;
if ($reduction.length) {
// Figure out what arguments were specified.
if ($.isFunction(arg0)) {
transform = arg0;
if (typeof (arg1) == "string") {
msg = arg1;
}
}
else if (typeof (arg0) == "string") {
msg = arg0;
}
// Map all the matched values into an array.
var map = $.map(
$reduction,
function(obj) {
return transform(obj.value);
}
);
// For each transformed value:
for (var i1 = 0; i1 < map.length; ++i1) {
// Unless it's an empty string:
if (map[i1].length) {
// For each value we've already looked at:
for (var i2 = 0; i2 < subMap.length; ++i2) {
// If we've already seen the transformed value:
if (subMap[i2] == map[i1]) {
valid = false;
// Collect repeated values
repeatedVal.push(map[i1]);
}
}
// We looked at the value.
subMap.push(map[i1]);
}
}
if (!valid) {
// Remove duplicates of duplicates
repeatedVal = $.unique(repeatedVal);
// For all repeated values found...
for (
var
i = 0,
repeatedLength = repeatedVal.length;
i < repeatedLength;
++i) {
// raise the error - aggregate will use last repeated value
raiseAggregateError(
$reduction.filter(
"[value=\'" + repeatedVal[i] + "\']"
),
msg
);
}
// The set reduces to zero valid elements.
this.reduction = $([]);
}
}
return this;
},
// Validate that the numeric sum of all values is equal to a given value.
sum:function(sum, msg) {
// If a reduced set is attached, use it. Also, remove unsupported
// elements.
var $reduction = (this.reduction || this).filter($.validity.settings.elementSupport);
if ($reduction.length && sum != numericSum($reduction)) {
raiseAggregateError(
$reduction,
msg || format(
$.validity.messages.sum,
{ sum:sum }
)
);
// The set reduces to zero valid elements.
this.reduction = $([]);
}
return this;
},
// Validates an inclusive upper-bound on the numeric sum of the values of
// all matched elements.
sumMax:function(max, msg) {
// If a reduced set is attached, use it. Also, remove unsupported
// elements.
var $reduction = (this.reduction || this).filter($.validity.settings.elementSupport);
if ($reduction.length && max < numericSum($reduction)) {
raiseAggregateError(
$reduction,
msg || format(
$.validity.messages.sumMax,
{ max:max }
)
);
// The set reduces to zero valid elements.
this.reduction = $([]);
}
return this;
},
// Validates an inclusive lower-bound on the numeric sum of the values of
// all matched elements.
sumMin:function(min, msg) {
// If a reduced set is attached, use it. Also, remove unsupported
// elements.
var $reduction = (this.reduction || this).filter($.validity.settings.elementSupport);
if ($reduction.length && min > numericSum($reduction)) {
raiseAggregateError(
$reduction,
msg || format(
$.validity.messages.sumMin,
{ min:min }
)
);
// The set reduces to zero valid elements.
this.reduction = $([]);
}
return this;
},
// Radio group validators:
///////////////////////////////////////////////////////////////////////////
radioChecked:function(val, msg) {
// If a reduced set is attached, use it. Also, remove unsupported
// elements.
var $reduction = (this.reduction || this).filter($.validity.settings.elementSupport);
if ($reduction.is(":radio") && $reduction.find(":checked").val() != val) {
raiseAggregateError(
$reduction,
msg || $.validity.messages.radioChecked
);
}
},
radioNotChecked:function(val, msg) {
// If a reduced set is attached, use it. Also, remove unsupported
// elements.
var $reduction = (this.reduction || this).filter($.validity.settings.elementSupport);
if ($reduction.is(":radio") && $reduction.filter(":checked").val() == val) {
raiseAggregateError(
$reduction,
msg || $.validity.messages.radioChecked
);
}
},
// Checkbox validators:
///////////////////////////////////////////////////////////////////////////
checkboxChecked:function(msg) {
// If a reduced set is attached, use it. Also, remove unsupported
// elements.
var $reduction = (this.reduction || this).filter($.validity.settings.elementSupport);
if ($reduction.is(":checkbox") && !$reduction.is(":checked")) {
raiseAggregateError(
$reduction,
msg || $.validity.messages.radioChecked
);
}
},
// Specialized validators:
///////////////////////////////////////////////////////////////////////////
// If expression is a function, it will be called on each matched element.
// Otherwise, it is treated as a boolean, and the determines the validity
// of elements in an aggregate fashion.
assert:function(expression, msg) {
// If a reduced set is attached, use it. Do not reduce to supported
// elements.
var $reduction = this.reduction || this;
if ($reduction.length) {
// In the case that 'expression' is a function, use it as a
// regimen on each matched element individually:
// add $.validity.messages[msg] duantihua 2011-07-16 support localized message
if ($.isFunction(expression)) {
return validate(
this,
expression,
$.validity.messages[msg] || msg || $.validity.messages.generic
);
}
// Otherwise map it to a boolean and consider this as an aggregate
// validation:
else if (!expression) {
raiseAggregateError(
$reduction,
msg || $.validity.messages.generic
);
// The set reduces to zero valid elements.
this.reduction = $([]);
}
}
return this;
},
fail:function(msg) {
return this.assert(false, msg);
}
});
// Private utilities:
///////////////////////////////////////////////////////////////////////////////
// Do non-aggregate validation. Subject each element in $obj to the regimen.
// Raise the specified error on failures. This function is the heart of
// validity:
function validate($obj, regimen, message) {
var
// If a reduced set is attached, use it Also, remove any unsupported
// elements.
$reduction = ($obj.reduction || $obj).filter($.validity.settings.elementSupport),
// Array to store only elements that pass the regimen.
elements = [];
// For each in the reduction.
$reduction.each(
function() {
// If the element passes the regimen, include it in the reduction.
if (regimen(this)) {
elements.push(this);
}
// Else give the element an error message.
else {
raiseError(
this,
format(message, {
field:infer(this)
})
);
}
}
);
// Attach a (potentially) reduced set of only elements that passed.
$obj.reduction = $(elements);
// Return the full set with attached reduction.
return $obj;
}
// Inform the report object that there was a failure.
function addToReport() {
if ($.validity.isValidating()) {
$.validity.report.errors++;
$.validity.report.valid = false;
}
}
// Inform the report of a failure and display an error according to the idiom
// of the current output mode.
function raiseError(obj, msg) {
addToReport();
$.validity.out.raise($(obj), msg);
}
// Inform the report of a failure and display an aggregate error according to
// the idiom of the current output mode.
function raiseAggregateError($obj, msg) {
addToReport();
$.validity.out.raiseAggregate($obj, msg);
}
// Yield the sum of the values of all fields matched in obj that can be parsed.
function numericSum(obj) {
var accumulator = 0;
obj.each(
function() {
var n = parseFloat(this.value);
accumulator += isNaN(n) ? 0 : n;
}
);
return accumulator;
}
// Using the associative array supplied as the 'obj' argument, replace tokens
// of the format #{} in the 'str' argument with that key's value.
function format(str, obj) {
for (var p in obj) {
if (obj.hasOwnProperty(p)) {
str = str.replace(
new RegExp("#\\{" + p + "\\}", "g"),
obj[p]
);
}
}
return capitalize(str);
}
// Infer the field name of the passed DOM element. If a title is specified, its
// value is returned. Otherwise, attempt to parse a field name out of the id
// attribute. If that doesn't work, return the default field name in the
// configuration.
function infer(field) {
var
$f = $(field),
id = $f.prop("id"),
ret = $.validity.settings.defaultFieldName;
// Check for title.
if ($f.prop("title").length) {
ret = $f.prop("title");
}
// Check for UpperCamelCase.
else if (/^([A-Z0-9][a-z]*)+$/.test(id)) {
ret = id.replace(/([A-Z0-9])[a-z]*/g, " $&");
}
// Check for lowercase_separated_by_underscores
else if (/^[a-z0-9]+(_[a-z0-9]+)*$/.test(id)) {
var arr = id.split("_");
for (var i = 0; i < arr.length; ++i) {
arr[i] = capitalize(arr[i]);
}
ret = arr.join(" ");
}
return $.trim(ret);
}
// Capitalize the first character of the string argument.
function capitalize(sz) {
return sz.substring ?
sz.substring(0, 1).toUpperCase() + sz.substring(1, sz.length) :
sz;
}
__private = {
validate:validate,
addToReport:addToReport,
raiseError:raiseError,
raiseAggregateError:raiseAggregateError,
numericSum:numericSum,
format:format,
infer:infer,
capitalize:capitalize
};
})(jQuery);
// Output modes:
////////////////////////////////////////////////////////////////
// Each output mode gets its own closure,
// distinct from the validity closure.
// Install the tooltip output.
(function($) {
$.validity.outputs.tooltip = {
tooltipClass:"validity-tooltip",
start:function() {
$("." + $.validity.outputs.tooltip.tooltipClass)
.remove();
},
end:function(results) {
// If not valid and scrollTo is enabled, scroll the page to the first error.
if (!results.valid && $.validity.settings.scrollTo) {
$(document).scrollTop(
$("." + $.validity.outputs.tooltip.tooltipClass)
.offset()
.top
);
}
},
raise:function($obj, msg) {
var pos = $obj.offset();
pos.left += $obj.width() + 18;
pos.top += 8;
$(
"" +
msg +
"" +
"" +
"" +
""
)
.click(function() {
$obj.focus();
$(this).fadeOut();
})
.css(pos)
.hide()
.appendTo("body")
.fadeIn();
},
raiseAggregate:function($obj, msg) {
// Just raise the error on the last input.
if ($obj.length) {
this.raise($obj.filter(":last"), msg);
}
}
};
})(jQuery);
// Install the label output.
(function($) {
function getIdentifier($obj) {
return $obj.attr('id').length ?
$obj.attr('id') :
$obj.attr('name');
}
$.validity.outputs.label = {
cssClass:"error",
start:function() {
// Remove all the existing error labels.
$("." + $.validity.outputs.label.cssClass)
.remove();
},
end:function(results) {
// If not valid and scrollTo is enabled, scroll the page to the first error.
if (!results.valid && $.validity.settings.scrollTo) {
location.hash = $("." + $.validity.outputs.label.cssClass + ":eq(0)").attr('for');
}
},
raise:function($obj, msg) {
var
labelSelector = "." + $.validity.outputs.label.cssClass + "[for='" + getIdentifier($obj) + "']";
// If an error label already exists for the bad input just update its text:
if ($(labelSelector).length) {
$(labelSelector).text(msg);
}
// Otherwize create a new one and stick it after the input:
else {
$("")
.attr("for", getIdentifier($obj))
.addClass($.validity.outputs.label.cssClass)
.text(msg)
// In the case that the element does not have an id
// then the for attribute in the label will not cause
// clicking the label to focus the element. This line
// will make that happen.
.click(function() {
if ($obj.length) {
$obj[0].select();
}
})
.insertAfter($obj);
}
},
raiseAggregate:function($obj, msg) {
// Just raise the error on the last input.
if ($obj.length) {
this.raise($($obj.get($obj.length - 1)), msg);
}
}
};
})(jQuery);
// Install the modal output.
(function($) {
var
// Class to apply to modal errors.
errorClass = "validity-modal-msg",
// The selector for the element where modal errors will me injected semantically.
container = "body";
$.validity.outputs.modal = {
start:function() {
// Remove all the existing errors.
$("." + errorClass).remove();
},
end:function(results) {
// If not valid and scrollTo is enabled, scroll the page to the first error.
if (!results.valid && $.validity.settings.scrollTo) {
location.hash = $("." + errorClass + ":eq(0)").attr('id');
}
},
raise:function($obj, msg) {
if ($obj.length) {
var
off = $obj.offset(),
obj = $obj.get(0),
// Design a style object based off of the input's location.
errorStyle = {
left:parseInt(off.left + $obj.width() + 4, 10) + "px",
top:parseInt(off.top - 10, 10) + "px"
};
// Create one and position it next to the input.
$("")
.addClass(errorClass)
.css(errorStyle)
.text(msg)
.click($.validity.settings.modalErrorsClickable ?
function() { $(this).remove(); } : null
)
.appendTo(container);
}
},
raiseAggregate:function($obj, msg) {
// Just raise the error on the last input.
if ($obj.length) {
this.raise($($obj.get($obj.length - 1)), msg);
}
}
};
})(jQuery);
// Install the summary output
(function($) {
var
// Container contains the summary. This is the element that is shown or hidden.
container = ".validity-summary-container",
// Erroneous refers to an input with an invalid value,
// not the error message itself.
erroneous = "validity-erroneous",
// Selector for erroneous inputs.
errors = "." + erroneous,
// The wrapper for entries in the summary.
wrapper = "",
// Buffer to contain all the error messages that build up during validation.
// When validation ends, it'll be flushed into the summary.
// This way, the summary doesn't flicker empty then fill up.
buffer = [];
$.validity.outputs.summary = {
start:function() {
$(errors).removeClass(erroneous);
buffer = [];
},
end:function(results) {
// Hide the container and empty its summary.
$(container)
.hide()
.find("ul")
.html('');
// If there are any errors at all:
// (Otherwise the container shouldn't be shown):
if (buffer.length) {
// Use integer based iteration for solution to Issue 7.
for (var i = 0; i < buffer.length; ++i) {
$(wrapper)
.text(buffer[i])
.appendTo(container + " ul");
}
$(container).show();
// If scrollTo is enabled, scroll the page to the first error.
if ($.validity.settings.scrollTo) {
location.hash = $(errors + ":eq(0)").attr("id");
}
}
},
raise:function($obj, msg) {
buffer.push(msg);
$obj.addClass(erroneous);
},
raiseAggregate:function($obj, msg) {
this.raise($obj, msg);
},
container:function() {
document.write(
"" +
"The form didn't submit for the following reason(s):" +
"
" +
""
);
}
};
})(jQuery);
// duantihua 2012-09-02
jQuery.validity.patterns.notBlank=/[^\s]+/;
jQuery.validity.patterns.chinaIdcard=/^\d{6}(18|19|20)?\d{2}(0[1-9]|1[012])(0[1-9]|[12]\d|3[01])\d{3}(\d|X)?$/i;
jQuery.validity.patterns.yearMonth=/^\d{1,4}-((0?\d)|(1[012]))$/;
jQuery.validity.messages= {
require:"请填写#{field}",
requireSelect:"请选择#{field}",
// Format validators:
match:"#{field}格式无效",
integer:"#{field}应为整数",
date:"#{field}应为日期格式",
yearMonth:"#{field}格式无效.例:'2011-1','2011-10'.",
year:"#{field}应为'yyyy'.",
idCardPRC:"#{field}格式错误.",
email:"#{field}应为邮件格式",
usd:"#{field}应为美元格式",
url:"#{field}应为URL格式",
number:"#{field}应为数字格式",
zip:"#{field}应为邮政编码格式 如######",
phone:"#{field}应为电话号码格式如 ###-########",
guid:"#{field}应为 guid格式,如{3F2504E0-4F89-11D3-9A0C-0305E82C3301}",
time24:"#{field} 应为24小时格式,如 23:00",
time12:"#{field} 应为12小时格式,如 12:00 AM/PM",
// Value range messages:
lessThan:"#{field}须小于#{max}.",
lessThanOrEqualTo:"#{field} 须小于或等于 #{max}",
greaterThan:"#{field}须大于 #{min}",
greaterThanOrEqualTo:"#{field} 须大于或等于 #{min}",
range:"#{field} 应在#{min}~#{max}之间",
// Value length messages:
tooLong:"#{field}不应超过#{max}个字符",
tooShort:"#{field}不能少于#{min}个字符",
// Composition validators:
nonHtml:"#{field}不能包含html代码",
alphabet:"#{field}包含了不允许的字符",
minCharClass:"#{field} 不能超过 #{min} #{charClass} 字符",
maxCharClass:"#{field} 不能少于 #{min} #{charClass} 字符",
// Aggregate validator messages:
equal:"数据不相等",
distinct:"存在数据重复",
sum:"数字累加之和不等与#{sum}",
sumMax:"数字累加之和必须小于#{max}",
sumMin:"数字累加之和必须大于#{min}",
// Radio validator messages:
radioChecked:"选择项无效",
generic:"无效"
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy