|
|
/*
|
|
|
* This file has been commented to support Visual Studio Intellisense.
|
|
|
* You should not use this file at runtime inside the browser--it is only
|
|
|
* intended to be used only for design-time IntelliSense. Please use the
|
|
|
* standard jQuery library for all production use.
|
|
|
*
|
|
|
* Comment version: 1.6
|
|
|
*/
|
|
|
|
|
|
/*
|
|
|
* jQuery validation plug-in 1.6
|
|
|
*
|
|
|
* http://bassistance.de/jquery-plugins/jquery-plugin-validation/
|
|
|
* http://docs.jquery.com/Plugins/Validation
|
|
|
*
|
|
|
* Copyright (c) 2006 - 2008 J??rn Zaefferer
|
|
|
*
|
|
|
* $Id: jquery.validate.js 6403 2009-06-17 14:27:16Z joern.zaefferer $
|
|
|
*
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining
|
|
|
* a copy of this software and associated documentation files (the
|
|
|
* "Software"), to deal in the Software without restriction, including
|
|
|
* without limitation the rights to use, copy, modify, merge, publish,
|
|
|
* distribute, sublicense, and/or sell copies of the Software, and to
|
|
|
* permit persons to whom the Software is furnished to do so, subject to
|
|
|
* the following conditions:
|
|
|
*
|
|
|
* The above copyright notice and this permission notice shall be
|
|
|
* included in all copies or substantial portions of the Software.
|
|
|
*
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
|
|
* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
|
|
* LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
|
|
* OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
|
|
* WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
|
|
*/
|
|
|
|
|
|
(function($) {
|
|
|
|
|
|
$.extend($.fn, {
|
|
|
// http://docs.jquery.com/Plugins/Validation/validate
|
|
|
validate: function( options ) {
|
|
|
/// <summary>
|
|
|
/// 验证所选表单。此方法将为 submit、focus、
|
|
|
/// keyup、blur 和 click 设置事件处理程序,用于触发整个表单或各个
|
|
|
/// 元素的验证操作。可以禁用每个事件,请参见 onxxx 选项(onsubmit、onfocusout、
|
|
|
/// onkeyup 和 onclick)。当提交的表单无效时,focusInvalid 将焦点设置在元素上。
|
|
|
/// </summary>
|
|
|
/// <param name="options" type="Options">
|
|
|
/// 配置验证的一组键/值对。所有选项都是可选的。
|
|
|
/// </param>
|
|
|
/// <returns type="Validator" />
|
|
|
|
|
|
// if nothing is selected, return nothing; can't chain anyway
|
|
|
if (!this.length) {
|
|
|
options && options.debug && window.console && console.warn( "nothing selected, can't validate, returning nothing" );
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
// check if a validator for this form was already created
|
|
|
var validator = $.data(this[0], 'validator');
|
|
|
if ( validator ) {
|
|
|
return validator;
|
|
|
}
|
|
|
|
|
|
validator = new $.validator( options, this[0] );
|
|
|
$.data(this[0], 'validator', validator);
|
|
|
|
|
|
if ( validator.settings.onsubmit ) {
|
|
|
|
|
|
// allow suppresing validation by adding a cancel class to the submit button
|
|
|
this.find("input, button").filter(".cancel").click(function() {
|
|
|
validator.cancelSubmit = true;
|
|
|
});
|
|
|
|
|
|
// when a submitHandler is used, capture the submitting button
|
|
|
if (validator.settings.submitHandler) {
|
|
|
this.find("input, button").filter(":submit").click(function() {
|
|
|
validator.submitButton = this;
|
|
|
});
|
|
|
}
|
|
|
|
|
|
// validate the form on submit
|
|
|
this.submit( function( event ) {
|
|
|
if ( validator.settings.debug )
|
|
|
// prevent form submit to be able to see console output
|
|
|
event.preventDefault();
|
|
|
|
|
|
function handle() {
|
|
|
if ( validator.settings.submitHandler ) {
|
|
|
if (validator.submitButton) {
|
|
|
// insert a hidden input as a replacement for the missing submit button
|
|
|
var hidden = $("<input type='hidden'/>").attr("name", validator.submitButton.name).val(validator.submitButton.value).appendTo(validator.currentForm);
|
|
|
}
|
|
|
validator.settings.submitHandler.call( validator, validator.currentForm );
|
|
|
if (validator.submitButton) {
|
|
|
// and clean up afterwards; thanks to no-block-scope, hidden can be referenced
|
|
|
hidden.remove();
|
|
|
}
|
|
|
return false;
|
|
|
}
|
|
|
return true;
|
|
|
}
|
|
|
|
|
|
// prevent submit for invalid forms or custom submit handlers
|
|
|
if ( validator.cancelSubmit ) {
|
|
|
validator.cancelSubmit = false;
|
|
|
return handle();
|
|
|
}
|
|
|
if ( validator.form() ) {
|
|
|
if ( validator.pendingRequest ) {
|
|
|
validator.formSubmitted = true;
|
|
|
return false;
|
|
|
}
|
|
|
return handle();
|
|
|
} else {
|
|
|
validator.focusInvalid();
|
|
|
return false;
|
|
|
}
|
|
|
});
|
|
|
}
|
|
|
|
|
|
return validator;
|
|
|
},
|
|
|
// http://docs.jquery.com/Plugins/Validation/valid
|
|
|
valid: function() {
|
|
|
/// <summary>
|
|
|
/// 检查选定表单是否有效,或检查所选全部元素是否都有效。
|
|
|
/// 使用此方法检查表单之前,需要对表单调用 validate()。
|
|
|
/// </summary>
|
|
|
/// <returns type="Boolean" />
|
|
|
|
|
|
if ( $(this[0]).is('form')) {
|
|
|
return this.validate().form();
|
|
|
} else {
|
|
|
var valid = true;
|
|
|
var validator = $(this[0].form).validate();
|
|
|
this.each(function() {
|
|
|
valid &= validator.element(this);
|
|
|
});
|
|
|
return valid;
|
|
|
}
|
|
|
},
|
|
|
// attributes: space seperated list of attributes to retrieve and remove
|
|
|
removeAttrs: function(attributes) {
|
|
|
/// <summary>
|
|
|
/// 从首个匹配的元素中移除指定的特性并返回这些特性。
|
|
|
/// </summary>
|
|
|
/// <param name="attributes" type="String">
|
|
|
/// 要移除的特性名称的空格分隔列表。
|
|
|
/// </param>
|
|
|
/// <returns type="" />
|
|
|
|
|
|
var result = {},
|
|
|
$element = this;
|
|
|
$.each(attributes.split(/\s/), function(index, value) {
|
|
|
result[value] = $element.attr(value);
|
|
|
$element.removeAttr(value);
|
|
|
});
|
|
|
return result;
|
|
|
},
|
|
|
// http://docs.jquery.com/Plugins/Validation/rules
|
|
|
rules: function(command, argument) {
|
|
|
/// <summary>
|
|
|
/// 为首个选定的元素返回验证规则。
|
|
|
/// </summary>
|
|
|
/// <param name="command" type="String">
|
|
|
/// 可以是“add”或“remove”。
|
|
|
/// </param>
|
|
|
/// <param name="argument" type="">
|
|
|
/// 要添加或移除的规则的列表。
|
|
|
/// </param>
|
|
|
/// <returns type="" />
|
|
|
|
|
|
var element = this[0];
|
|
|
|
|
|
if (command) {
|
|
|
var settings = $.data(element.form, 'validator').settings;
|
|
|
var staticRules = settings.rules;
|
|
|
var existingRules = $.validator.staticRules(element);
|
|
|
switch(command) {
|
|
|
case "add":
|
|
|
$.extend(existingRules, $.validator.normalizeRule(argument));
|
|
|
staticRules[element.name] = existingRules;
|
|
|
if (argument.messages)
|
|
|
settings.messages[element.name] = $.extend( settings.messages[element.name], argument.messages );
|
|
|
break;
|
|
|
case "remove":
|
|
|
if (!argument) {
|
|
|
delete staticRules[element.name];
|
|
|
return existingRules;
|
|
|
}
|
|
|
var filtered = {};
|
|
|
$.each(argument.split(/\s/), function(index, method) {
|
|
|
filtered[method] = existingRules[method];
|
|
|
delete existingRules[method];
|
|
|
});
|
|
|
return filtered;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
var data = $.validator.normalizeRules(
|
|
|
$.extend(
|
|
|
{},
|
|
|
$.validator.metadataRules(element),
|
|
|
$.validator.classRules(element),
|
|
|
$.validator.attributeRules(element),
|
|
|
$.validator.staticRules(element)
|
|
|
), element);
|
|
|
|
|
|
// make sure required is at front
|
|
|
if (data.required) {
|
|
|
var param = data.required;
|
|
|
delete data.required;
|
|
|
data = $.extend({required: param}, data);
|
|
|
}
|
|
|
|
|
|
return data;
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// Custom selectors
|
|
|
$.extend($.expr[":"], {
|
|
|
// http://docs.jquery.com/Plugins/Validation/blank
|
|
|
blank: function(a) {return !$.trim("" + a.value);},
|
|
|
// http://docs.jquery.com/Plugins/Validation/filled
|
|
|
filled: function(a) {return !!$.trim("" + a.value);},
|
|
|
// http://docs.jquery.com/Plugins/Validation/unchecked
|
|
|
unchecked: function(a) {return !a.checked;}
|
|
|
});
|
|
|
|
|
|
// constructor for validator
|
|
|
$.validator = function( options, form ) {
|
|
|
this.settings = $.extend( {}, $.validator.defaults, options );
|
|
|
this.currentForm = form;
|
|
|
this.init();
|
|
|
};
|
|
|
|
|
|
$.validator.format = function(source, params) {
|
|
|
/// <summary>
|
|
|
/// 用参数替换 {n} 占位符。
|
|
|
/// 除了字符串模板自身以外,还可以传递一个或多个参数,以插入到
|
|
|
/// 字符串中。
|
|
|
/// </summary>
|
|
|
/// <param name="source" type="String">
|
|
|
/// 要设置格式的字符串。
|
|
|
/// </param>
|
|
|
/// <param name="params" type="String">
|
|
|
/// 要插入的首个参数,或要插入的字符串数组
|
|
|
/// </param>
|
|
|
/// <returns type="String" />
|
|
|
|
|
|
if ( arguments.length == 1 )
|
|
|
return function() {
|
|
|
var args = $.makeArray(arguments);
|
|
|
args.unshift(source);
|
|
|
return $.validator.format.apply( this, args );
|
|
|
};
|
|
|
if ( arguments.length > 2 && params.constructor != Array ) {
|
|
|
params = $.makeArray(arguments).slice(1);
|
|
|
}
|
|
|
if ( params.constructor != Array ) {
|
|
|
params = [ params ];
|
|
|
}
|
|
|
$.each(params, function(i, n) {
|
|
|
source = source.replace(new RegExp("\\{" + i + "\\}", "g"), n);
|
|
|
});
|
|
|
return source;
|
|
|
};
|
|
|
|
|
|
$.extend($.validator, {
|
|
|
|
|
|
defaults: {
|
|
|
messages: {},
|
|
|
groups: {},
|
|
|
rules: {},
|
|
|
errorClass: "error",
|
|
|
validClass: "valid",
|
|
|
errorElement: "label",
|
|
|
focusInvalid: true,
|
|
|
errorContainer: $( [] ),
|
|
|
errorLabelContainer: $( [] ),
|
|
|
onsubmit: true,
|
|
|
ignore: [],
|
|
|
ignoreTitle: false,
|
|
|
onfocusin: function(element) {
|
|
|
this.lastActive = element;
|
|
|
|
|
|
// hide error label and remove error class on focus if enabled
|
|
|
if ( this.settings.focusCleanup && !this.blockFocusCleanup ) {
|
|
|
this.settings.unhighlight && this.settings.unhighlight.call( this, element, this.settings.errorClass, this.settings.validClass );
|
|
|
this.errorsFor(element).hide();
|
|
|
}
|
|
|
},
|
|
|
onfocusout: function(element) {
|
|
|
if ( !this.checkable(element) && (element.name in this.submitted || !this.optional(element)) ) {
|
|
|
this.element(element);
|
|
|
}
|
|
|
},
|
|
|
onkeyup: function(element) {
|
|
|
if ( element.name in this.submitted || element == this.lastElement ) {
|
|
|
this.element(element);
|
|
|
}
|
|
|
},
|
|
|
onclick: function(element) {
|
|
|
// click on selects, radiobuttons and checkboxes
|
|
|
if ( element.name in this.submitted )
|
|
|
this.element(element);
|
|
|
// or option elements, check parent select in that case
|
|
|
else if (element.parentNode.name in this.submitted)
|
|
|
this.element(element.parentNode)
|
|
|
},
|
|
|
highlight: function( element, errorClass, validClass ) {
|
|
|
$(element).addClass(errorClass).removeClass(validClass);
|
|
|
},
|
|
|
unhighlight: function( element, errorClass, validClass ) {
|
|
|
$(element).removeClass(errorClass).addClass(validClass);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Validator/setDefaults
|
|
|
setDefaults: function(settings) {
|
|
|
/// <summary>
|
|
|
/// 修改验证操作的默认设置。
|
|
|
/// 接受 Plugins/Validation/validate 接受的所有内容。
|
|
|
/// </summary>
|
|
|
/// <param name="settings" type="Options">
|
|
|
/// 要设置为默认值的选项。
|
|
|
/// </param>
|
|
|
/// <returns type="undefined" />
|
|
|
|
|
|
$.extend( $.validator.defaults, settings );
|
|
|
},
|
|
|
|
|
|
messages: {
|
|
|
required: "This field is required.",
|
|
|
remote: "Please fix this field.",
|
|
|
email: "Please enter a valid email address.",
|
|
|
url: "请输入有效的 URL。",
|
|
|
date: "Please enter a valid date.",
|
|
|
dateISO: "请输入有效的日期(ISO)。",
|
|
|
number: "Please enter a valid number.",
|
|
|
digits: "请只输入数字。",
|
|
|
creditcard: "Please enter a valid credit card number.",
|
|
|
equalTo: "Please enter the same value again.",
|
|
|
accept: "Please enter a value with a valid extension.",
|
|
|
maxlength: $.validator.format("Please enter no more than {0} characters."),
|
|
|
minlength: $.validator.format("Please enter at least {0} characters."),
|
|
|
rangelength: $.validator.format("Please enter a value between {0} and {1} characters long."),
|
|
|
range: $.validator.format("Please enter a value between {0} and {1}."),
|
|
|
max: $.validator.format("Please enter a value less than or equal to {0}."),
|
|
|
min: $.validator.format("Please enter a value greater than or equal to {0}.")
|
|
|
},
|
|
|
|
|
|
autoCreateRanges: false,
|
|
|
|
|
|
prototype: {
|
|
|
|
|
|
init: function() {
|
|
|
this.labelContainer = $(this.settings.errorLabelContainer);
|
|
|
this.errorContext = this.labelContainer.length && this.labelContainer || $(this.currentForm);
|
|
|
this.containers = $(this.settings.errorContainer).add( this.settings.errorLabelContainer );
|
|
|
this.submitted = {};
|
|
|
this.valueCache = {};
|
|
|
this.pendingRequest = 0;
|
|
|
this.pending = {};
|
|
|
this.invalid = {};
|
|
|
this.reset();
|
|
|
|
|
|
var groups = (this.groups = {});
|
|
|
$.each(this.settings.groups, function(key, value) {
|
|
|
$.each(value.split(/\s/), function(index, name) {
|
|
|
groups[name] = key;
|
|
|
});
|
|
|
});
|
|
|
var rules = this.settings.rules;
|
|
|
$.each(rules, function(key, value) {
|
|
|
rules[key] = $.validator.normalizeRule(value);
|
|
|
});
|
|
|
|
|
|
function delegate(event) {
|
|
|
var validator = $.data(this[0].form, "validator");
|
|
|
validator.settings["on" + event.type] && validator.settings["on" + event.type].call(validator, this[0] );
|
|
|
}
|
|
|
$(this.currentForm)
|
|
|
.delegate("focusin focusout keyup", ":text, :password, :file, select, textarea", delegate)
|
|
|
.delegate("click", ":radio, :checkbox, select, option", delegate);
|
|
|
|
|
|
if (this.settings.invalidHandler)
|
|
|
$(this.currentForm).bind("invalid-form.validate", this.settings.invalidHandler);
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Validator/form
|
|
|
form: function() {
|
|
|
/// <summary>
|
|
|
/// 验证表单。如果表单有效,则返回 true;否则返回 false。
|
|
|
/// 此行为与普通 submit 事件类似,但它返回的是结果。
|
|
|
/// </summary>
|
|
|
/// <returns type="Boolean" />
|
|
|
|
|
|
this.checkForm();
|
|
|
$.extend(this.submitted, this.errorMap);
|
|
|
this.invalid = $.extend({}, this.errorMap);
|
|
|
if (!this.valid())
|
|
|
$(this.currentForm).triggerHandler("invalid-form", [this]);
|
|
|
this.showErrors();
|
|
|
return this.valid();
|
|
|
},
|
|
|
|
|
|
checkForm: function() {
|
|
|
this.prepareForm();
|
|
|
for ( var i = 0, elements = (this.currentElements = this.elements()); elements[i]; i++ ) {
|
|
|
this.check( elements[i] );
|
|
|
}
|
|
|
return this.valid();
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Validator/element
|
|
|
element: function( element ) {
|
|
|
/// <summary>
|
|
|
/// 验证单个元素。如果元素有效,则返回 true;否则返回 false。
|
|
|
/// 此行为类似于对 blur 或 keyup 进行验证操作,但它返回的是结果。
|
|
|
/// </summary>
|
|
|
/// <param name="element" type="Selector">
|
|
|
/// 要验证的元素,该元素必须位于已验证的表单内。
|
|
|
/// </param>
|
|
|
/// <returns type="Boolean" />
|
|
|
|
|
|
element = this.clean( element );
|
|
|
this.lastElement = element;
|
|
|
this.prepareElement( element );
|
|
|
this.currentElements = $(element);
|
|
|
var result = this.check( element );
|
|
|
if ( result ) {
|
|
|
delete this.invalid[element.name];
|
|
|
} else {
|
|
|
this.invalid[element.name] = true;
|
|
|
}
|
|
|
if ( !this.numberOfInvalids() ) {
|
|
|
// Hide error containers on last error
|
|
|
this.toHide = this.toHide.add( this.containers );
|
|
|
}
|
|
|
this.showErrors();
|
|
|
return result;
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Validator/showErrors
|
|
|
showErrors: function(errors) {
|
|
|
/// <summary>
|
|
|
/// 显示指定的消息。
|
|
|
/// 键必须指代元素的名称,将显示那些元素所对应的值(使用已配置的错误位置)。
|
|
|
/// </summary>
|
|
|
/// <param name="errors" type="Object">
|
|
|
/// 输入名称和消息的一个或多个键/值对。
|
|
|
/// </param>
|
|
|
/// <returns type="undefined" />
|
|
|
|
|
|
if(errors) {
|
|
|
// add items to error list and map
|
|
|
$.extend( this.errorMap, errors );
|
|
|
this.errorList = [];
|
|
|
for ( var name in errors ) {
|
|
|
this.errorList.push({
|
|
|
message: errors[name],
|
|
|
element: this.findByName(name)[0]
|
|
|
});
|
|
|
}
|
|
|
// remove items from success list
|
|
|
this.successList = $.grep( this.successList, function(element) {
|
|
|
return !(element.name in errors);
|
|
|
});
|
|
|
}
|
|
|
this.settings.showErrors
|
|
|
? this.settings.showErrors.call( this, this.errorMap, this.errorList )
|
|
|
: this.defaultShowErrors();
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Validator/resetForm
|
|
|
resetForm: function() {
|
|
|
/// <summary>
|
|
|
/// 重置受控表单。
|
|
|
/// 将输入字段重置为其原始值(需要表单插件),移除指示无效元素的
|
|
|
/// 类并隐藏错误消息。
|
|
|
/// </summary>
|
|
|
/// <returns type="undefined" />
|
|
|
|
|
|
if ( $.fn.resetForm )
|
|
|
$( this.currentForm ).resetForm();
|
|
|
this.submitted = {};
|
|
|
this.prepareForm();
|
|
|
this.hideErrors();
|
|
|
this.elements().removeClass( this.settings.errorClass );
|
|
|
},
|
|
|
|
|
|
numberOfInvalids: function() {
|
|
|
/// <summary>
|
|
|
/// 返回无效字段的个数。
|
|
|
/// 这依赖于内部验证程序的状态。只有在验证整个表单
|
|
|
/// (在提交时或通过调用 $("form").valid())之后才会覆盖所有字段。在验证
|
|
|
/// 单个元素之后,只计入相应的元素。与 invalidHandler-option 组合时
|
|
|
/// invalidHandler-option.
|
|
|
/// </summary>
|
|
|
/// <returns type="Number" />
|
|
|
|
|
|
return this.objectLength(this.invalid);
|
|
|
},
|
|
|
|
|
|
objectLength: function( obj ) {
|
|
|
var count = 0;
|
|
|
for ( var i in obj )
|
|
|
count++;
|
|
|
return count;
|
|
|
},
|
|
|
|
|
|
hideErrors: function() {
|
|
|
this.addWrapper( this.toHide ).hide();
|
|
|
},
|
|
|
|
|
|
valid: function() {
|
|
|
return this.size() == 0;
|
|
|
},
|
|
|
|
|
|
size: function() {
|
|
|
return this.errorList.length;
|
|
|
},
|
|
|
|
|
|
focusInvalid: function() {
|
|
|
if( this.settings.focusInvalid ) {
|
|
|
try {
|
|
|
$(this.findLastActive() || this.errorList.length && this.errorList[0].element || []).filter(":visible").focus();
|
|
|
} catch(e) {
|
|
|
// ignore IE throwing errors when focusing hidden elements
|
|
|
}
|
|
|
}
|
|
|
},
|
|
|
|
|
|
findLastActive: function() {
|
|
|
var lastActive = this.lastActive;
|
|
|
return lastActive && $.grep(this.errorList, function(n) {
|
|
|
return n.element.name == lastActive.name;
|
|
|
}).length == 1 && lastActive;
|
|
|
},
|
|
|
|
|
|
elements: function() {
|
|
|
var validator = this,
|
|
|
rulesCache = {};
|
|
|
|
|
|
// select all valid inputs inside the form (no submit or reset buttons)
|
|
|
// workaround $Query([]).add until http://dev.jquery.com/ticket/2114 is solved
|
|
|
return $([]).add(this.currentForm.elements)
|
|
|
.filter(":input")
|
|
|
.not(":submit, :reset, :image, [disabled]")
|
|
|
.not( this.settings.ignore )
|
|
|
.filter(function() {
|
|
|
!this.name && validator.settings.debug && window.console && console.error( "没有为 %o 分配名称", this);
|
|
|
|
|
|
// select only the first element for each name, and only those with rules specified
|
|
|
if ( this.name in rulesCache || !validator.objectLength($(this).rules()) )
|
|
|
return false;
|
|
|
|
|
|
rulesCache[this.name] = true;
|
|
|
return true;
|
|
|
});
|
|
|
},
|
|
|
|
|
|
clean: function( selector ) {
|
|
|
return $( selector )[0];
|
|
|
},
|
|
|
|
|
|
errors: function() {
|
|
|
return $( this.settings.errorElement + "." + this.settings.errorClass, this.errorContext );
|
|
|
},
|
|
|
|
|
|
reset: function() {
|
|
|
this.successList = [];
|
|
|
this.errorList = [];
|
|
|
this.errorMap = {};
|
|
|
this.toShow = $([]);
|
|
|
this.toHide = $([]);
|
|
|
this.currentElements = $([]);
|
|
|
},
|
|
|
|
|
|
prepareForm: function() {
|
|
|
this.reset();
|
|
|
this.toHide = this.errors().add( this.containers );
|
|
|
},
|
|
|
|
|
|
prepareElement: function( element ) {
|
|
|
this.reset();
|
|
|
this.toHide = this.errorsFor(element);
|
|
|
},
|
|
|
|
|
|
check: function( element ) {
|
|
|
element = this.clean( element );
|
|
|
|
|
|
// if radio/checkbox, validate first element in group instead
|
|
|
if (this.checkable(element)) {
|
|
|
element = this.findByName( element.name )[0];
|
|
|
}
|
|
|
|
|
|
var rules = $(element).rules();
|
|
|
var dependencyMismatch = false;
|
|
|
for( method in rules ) {
|
|
|
var rule = { method: method, parameters: rules[method] };
|
|
|
try {
|
|
|
var result = $.validator.methods[method].call( this, element.value.replace(/\r/g, ""), element, rule.parameters );
|
|
|
|
|
|
// if a method indicates that the field is optional and therefore valid,
|
|
|
// don't mark it as valid when there are no other rules
|
|
|
if ( result == "dependency-mismatch" ) {
|
|
|
dependencyMismatch = true;
|
|
|
continue;
|
|
|
}
|
|
|
dependencyMismatch = false;
|
|
|
|
|
|
if ( result == "pending" ) {
|
|
|
this.toHide = this.toHide.not( this.errorsFor(element) );
|
|
|
return;
|
|
|
}
|
|
|
|
|
|
if( !result ) {
|
|
|
this.formatAndAdd( element, rule );
|
|
|
return false;
|
|
|
}
|
|
|
} catch(e) {
|
|
|
this.settings.debug && window.console && console.log("exception occured when checking element " + element.id
|
|
|
+ ", check the '" + rule.method + "' method", e);
|
|
|
throw e;
|
|
|
}
|
|
|
}
|
|
|
if (dependencyMismatch)
|
|
|
return;
|
|
|
if ( this.objectLength(rules) )
|
|
|
this.successList.push(element);
|
|
|
return true;
|
|
|
},
|
|
|
|
|
|
// return the custom message for the given element and validation method
|
|
|
// specified in the element's "messages" metadata
|
|
|
customMetaMessage: function(element, method) {
|
|
|
if (!$.metadata)
|
|
|
return;
|
|
|
|
|
|
var meta = this.settings.meta
|
|
|
? $(element).metadata()[this.settings.meta]
|
|
|
: $(element).metadata();
|
|
|
|
|
|
return meta && meta.messages && meta.messages[method];
|
|
|
},
|
|
|
|
|
|
// return the custom message for the given element name and validation method
|
|
|
customMessage: function( name, method ) {
|
|
|
var m = this.settings.messages[name];
|
|
|
return m && (m.constructor == String
|
|
|
? m
|
|
|
: m[method]);
|
|
|
},
|
|
|
|
|
|
// return the first defined argument, allowing empty strings
|
|
|
findDefined: function() {
|
|
|
for(var i = 0; i < arguments.length; i++) {
|
|
|
if (arguments[i] !== undefined)
|
|
|
return arguments[i];
|
|
|
}
|
|
|
return undefined;
|
|
|
},
|
|
|
|
|
|
defaultMessage: function( element, method) {
|
|
|
return this.findDefined(
|
|
|
this.customMessage( element.name, method ),
|
|
|
this.customMetaMessage( element, method ),
|
|
|
// title is never undefined, so handle empty string as undefined
|
|
|
!this.settings.ignoreTitle && element.title || undefined,
|
|
|
$.validator.messages[method],
|
|
|
"<strong>警告: 没有为它们定义任何消息" + element.name + "</strong>"
|
|
|
);
|
|
|
},
|
|
|
|
|
|
formatAndAdd: function( element, rule ) {
|
|
|
var message = this.defaultMessage( element, rule.method ),
|
|
|
theregex = /\$?\{(\d+)\}/g;
|
|
|
if ( typeof message == "function" ) {
|
|
|
message = message.call(this, rule.parameters, element);
|
|
|
} else if (theregex.test(message)) {
|
|
|
message = jQuery.format(message.replace(theregex, '{$1}'), rule.parameters);
|
|
|
}
|
|
|
this.errorList.push({
|
|
|
message: message,
|
|
|
element: element
|
|
|
});
|
|
|
|
|
|
this.errorMap[element.name] = message;
|
|
|
this.submitted[element.name] = message;
|
|
|
},
|
|
|
|
|
|
addWrapper: function(toToggle) {
|
|
|
if ( this.settings.wrapper )
|
|
|
toToggle = toToggle.add( toToggle.parent( this.settings.wrapper ) );
|
|
|
return toToggle;
|
|
|
},
|
|
|
|
|
|
defaultShowErrors: function() {
|
|
|
for ( var i = 0; this.errorList[i]; i++ ) {
|
|
|
var error = this.errorList[i];
|
|
|
this.settings.highlight && this.settings.highlight.call( this, error.element, this.settings.errorClass, this.settings.validClass );
|
|
|
this.showLabel( error.element, error.message );
|
|
|
}
|
|
|
if( this.errorList.length ) {
|
|
|
this.toShow = this.toShow.add( this.containers );
|
|
|
}
|
|
|
if (this.settings.success) {
|
|
|
for ( var i = 0; this.successList[i]; i++ ) {
|
|
|
this.showLabel( this.successList[i] );
|
|
|
}
|
|
|
}
|
|
|
if (this.settings.unhighlight) {
|
|
|
for ( var i = 0, elements = this.validElements(); elements[i]; i++ ) {
|
|
|
this.settings.unhighlight.call( this, elements[i], this.settings.errorClass, this.settings.validClass );
|
|
|
}
|
|
|
}
|
|
|
this.toHide = this.toHide.not( this.toShow );
|
|
|
this.hideErrors();
|
|
|
this.addWrapper( this.toShow ).show();
|
|
|
},
|
|
|
|
|
|
validElements: function() {
|
|
|
return this.currentElements.not(this.invalidElements());
|
|
|
},
|
|
|
|
|
|
invalidElements: function() {
|
|
|
return $(this.errorList).map(function() {
|
|
|
return this.element;
|
|
|
});
|
|
|
},
|
|
|
|
|
|
showLabel: function(element, message) {
|
|
|
var label = this.errorsFor( element );
|
|
|
if ( label.length ) {
|
|
|
// refresh error/success class
|
|
|
label.removeClass().addClass( this.settings.errorClass );
|
|
|
|
|
|
// check if we have a generated label, replace the message then
|
|
|
label.attr("generated") && label.html(message);
|
|
|
} else {
|
|
|
// create label
|
|
|
label = $("<" + this.settings.errorElement + "/>")
|
|
|
.attr({"for": this.idOrName(element), generated: true})
|
|
|
.addClass(this.settings.errorClass)
|
|
|
.html(message || "");
|
|
|
if ( this.settings.wrapper ) {
|
|
|
// make sure the element is visible, even in IE
|
|
|
// actually showing the wrapped element is handled elsewhere
|
|
|
label = label.hide().show().wrap("<" + this.settings.wrapper + "/>").parent();
|
|
|
}
|
|
|
if ( !this.labelContainer.append(label).length )
|
|
|
this.settings.errorPlacement
|
|
|
? this.settings.errorPlacement(label, $(element) )
|
|
|
: label.insertAfter(element);
|
|
|
}
|
|
|
if ( !message && this.settings.success ) {
|
|
|
label.text("");
|
|
|
typeof this.settings.success == "string"
|
|
|
? label.addClass( this.settings.success )
|
|
|
: this.settings.success( label );
|
|
|
}
|
|
|
this.toShow = this.toShow.add(label);
|
|
|
},
|
|
|
|
|
|
errorsFor: function(element) {
|
|
|
var name = this.idOrName(element);
|
|
|
return this.errors().filter(function() {
|
|
|
return $(this).attr('for') == name
|
|
|
});
|
|
|
},
|
|
|
|
|
|
idOrName: function(element) {
|
|
|
return this.groups[element.name] || (this.checkable(element) ? element.name : element.id || element.name);
|
|
|
},
|
|
|
|
|
|
checkable: function( element ) {
|
|
|
return /radio|checkbox/i.test(element.type);
|
|
|
},
|
|
|
|
|
|
findByName: function( name ) {
|
|
|
// select by name and filter by form for performance over form.find("[name=...]")
|
|
|
var form = this.currentForm;
|
|
|
return $(document.getElementsByName(name)).map(function(index, element) {
|
|
|
return element.form == form && element.name == name && element || null;
|
|
|
});
|
|
|
},
|
|
|
|
|
|
getLength: function(value, element) {
|
|
|
switch( element.nodeName.toLowerCase() ) {
|
|
|
case 'select':
|
|
|
return $("option:selected", element).length;
|
|
|
case 'input':
|
|
|
if( this.checkable( element) )
|
|
|
return this.findByName(element.name).filter(':checked').length;
|
|
|
}
|
|
|
return value.length;
|
|
|
},
|
|
|
|
|
|
depend: function(param, element) {
|
|
|
return this.dependTypes[typeof param]
|
|
|
? this.dependTypes[typeof param](param, element)
|
|
|
: true;
|
|
|
},
|
|
|
|
|
|
dependTypes: {
|
|
|
"boolean": function(param, element) {
|
|
|
return param;
|
|
|
},
|
|
|
"string": function(param, element) {
|
|
|
return !!$(param, element.form).length;
|
|
|
},
|
|
|
"function": function(param, element) {
|
|
|
return param(element);
|
|
|
}
|
|
|
},
|
|
|
|
|
|
optional: function(element) {
|
|
|
return !$.validator.methods.required.call(this, $.trim(element.value), element) && "dependency-mismatch";
|
|
|
},
|
|
|
|
|
|
startRequest: function(element) {
|
|
|
if (!this.pending[element.name]) {
|
|
|
this.pendingRequest++;
|
|
|
this.pending[element.name] = true;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
stopRequest: function(element, valid) {
|
|
|
this.pendingRequest--;
|
|
|
// sometimes synchronization fails, make sure pendingRequest is never < 0
|
|
|
if (this.pendingRequest < 0)
|
|
|
this.pendingRequest = 0;
|
|
|
delete this.pending[element.name];
|
|
|
if ( valid && this.pendingRequest == 0 && this.formSubmitted && this.form() ) {
|
|
|
$(this.currentForm).submit();
|
|
|
this.formSubmitted = false;
|
|
|
} else if (!valid && this.pendingRequest == 0 && this.formSubmitted) {
|
|
|
$(this.currentForm).triggerHandler("invalid-form", [this]);
|
|
|
this.formSubmitted = false;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
previousValue: function(element) {
|
|
|
return $.data(element, "previousValue") || $.data(element, "previousValue", {
|
|
|
old: null,
|
|
|
valid: true,
|
|
|
message: this.defaultMessage( element, "remote" )
|
|
|
});
|
|
|
}
|
|
|
|
|
|
},
|
|
|
|
|
|
classRuleSettings: {
|
|
|
required: {required: true},
|
|
|
email: {email: true},
|
|
|
url: {url: true},
|
|
|
date: {date: true},
|
|
|
dateISO: {dateISO: true},
|
|
|
dateDE: {dateDE: true},
|
|
|
number: {number: true},
|
|
|
numberDE: {numberDE: true},
|
|
|
digits: {digits: true},
|
|
|
creditcard: {creditcard: true}
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Validator/addClassRules#namerules
|
|
|
addClassRules: function(className, rules) {
|
|
|
/// <summary>
|
|
|
/// 添加一个复合类方法,这对于将常见的规则组合重构为单个
|
|
|
/// 类很有用。
|
|
|
/// </summary>
|
|
|
/// <param name="name" type="String">
|
|
|
/// 要添加的类规则的名称
|
|
|
/// </param>
|
|
|
/// <param name="rules" type="Options">
|
|
|
/// 复合规则
|
|
|
/// </param>
|
|
|
/// <returns type="undefined" />
|
|
|
|
|
|
className.constructor == String ?
|
|
|
this.classRuleSettings[className] = rules :
|
|
|
$.extend(this.classRuleSettings, className);
|
|
|
},
|
|
|
|
|
|
classRules: function(element) {
|
|
|
var rules = {};
|
|
|
var classes = $(element).attr('class');
|
|
|
classes && $.each(classes.split(' '), function() {
|
|
|
if (this in $.validator.classRuleSettings) {
|
|
|
$.extend(rules, $.validator.classRuleSettings[this]);
|
|
|
}
|
|
|
});
|
|
|
return rules;
|
|
|
},
|
|
|
|
|
|
attributeRules: function(element) {
|
|
|
var rules = {};
|
|
|
var $element = $(element);
|
|
|
|
|
|
for (method in $.validator.methods) {
|
|
|
var value = $element.attr(method);
|
|
|
if (value) {
|
|
|
rules[method] = value;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// maxlength may be returned as -1, 2147483647 (IE) and 524288 (safari) for text inputs
|
|
|
if (rules.maxlength && /-1|2147483647|524288/.test(rules.maxlength)) {
|
|
|
delete rules.maxlength;
|
|
|
}
|
|
|
|
|
|
return rules;
|
|
|
},
|
|
|
|
|
|
metadataRules: function(element) {
|
|
|
if (!$.metadata) return {};
|
|
|
|
|
|
var meta = $.data(element.form, 'validator').settings.meta;
|
|
|
return meta ?
|
|
|
$(element).metadata()[meta] :
|
|
|
$(element).metadata();
|
|
|
},
|
|
|
|
|
|
staticRules: function(element) {
|
|
|
var rules = {};
|
|
|
var validator = $.data(element.form, 'validator');
|
|
|
if (validator.settings.rules) {
|
|
|
rules = $.validator.normalizeRule(validator.settings.rules[element.name]) || {};
|
|
|
}
|
|
|
return rules;
|
|
|
},
|
|
|
|
|
|
normalizeRules: function(rules, element) {
|
|
|
// handle dependency check
|
|
|
$.each(rules, function(prop, val) {
|
|
|
// ignore rule when param is explicitly false, eg. required:false
|
|
|
if (val === false) {
|
|
|
delete rules[prop];
|
|
|
return;
|
|
|
}
|
|
|
if (val.param || val.depends) {
|
|
|
var keepRule = true;
|
|
|
switch (typeof val.depends) {
|
|
|
case "string":
|
|
|
keepRule = !!$(val.depends, element.form).length;
|
|
|
break;
|
|
|
case "function":
|
|
|
keepRule = val.depends.call(element, element);
|
|
|
break;
|
|
|
}
|
|
|
if (keepRule) {
|
|
|
rules[prop] = val.param !== undefined ? val.param : true;
|
|
|
} else {
|
|
|
delete rules[prop];
|
|
|
}
|
|
|
}
|
|
|
});
|
|
|
|
|
|
// evaluate parameters
|
|
|
$.each(rules, function(rule, parameter) {
|
|
|
rules[rule] = $.isFunction(parameter) ? parameter(element) : parameter;
|
|
|
});
|
|
|
|
|
|
// clean number parameters
|
|
|
$.each(['minlength', 'maxlength', 'min', 'max'], function() {
|
|
|
if (rules[this]) {
|
|
|
rules[this] = Number(rules[this]);
|
|
|
}
|
|
|
});
|
|
|
$.each(['rangelength', 'range'], function() {
|
|
|
if (rules[this]) {
|
|
|
rules[this] = [Number(rules[this][0]), Number(rules[this][1])];
|
|
|
}
|
|
|
});
|
|
|
|
|
|
if ($.validator.autoCreateRanges) {
|
|
|
// auto-create ranges
|
|
|
if (rules.min && rules.max) {
|
|
|
rules.range = [rules.min, rules.max];
|
|
|
delete rules.min;
|
|
|
delete rules.max;
|
|
|
}
|
|
|
if (rules.minlength && rules.maxlength) {
|
|
|
rules.rangelength = [rules.minlength, rules.maxlength];
|
|
|
delete rules.minlength;
|
|
|
delete rules.maxlength;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
// To support custom messages in metadata ignore rule methods titled "messages"
|
|
|
if (rules.messages) {
|
|
|
delete rules.messages
|
|
|
}
|
|
|
|
|
|
return rules;
|
|
|
},
|
|
|
|
|
|
// Converts a simple string to a {string: true} rule, e.g., "required" to {required:true}
|
|
|
normalizeRule: function(data) {
|
|
|
if( typeof data == "string" ) {
|
|
|
var transformed = {};
|
|
|
$.each(data.split(/\s/), function() {
|
|
|
transformed[this] = true;
|
|
|
});
|
|
|
data = transformed;
|
|
|
}
|
|
|
return data;
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Validator/addMethod
|
|
|
addMethod: function(name, method, message) {
|
|
|
/// <summary>
|
|
|
/// 添加一个自定义验证方法。此方法必须包含一个名称(必须是合法的 javascript
|
|
|
/// 标识符)、一个基于 javascript 的函数和一个默认字符串消息。
|
|
|
/// </summary>
|
|
|
/// <param name="name" type="String">
|
|
|
/// 用于标识和引用方法的名称必须是有效的 javascript
|
|
|
/// 标识符
|
|
|
/// </param>
|
|
|
/// <param name="method" type="Function">
|
|
|
/// 实际的方法实现,在元素有效时返回 true
|
|
|
/// </param>
|
|
|
/// <param name="message" type="String" optional="true">
|
|
|
/// (可选)要为此方法显示的默认消息。可以是由
|
|
|
/// jQuery.validator.format(value) 创建的函数。若未定义,则使用已经存在的消息
|
|
|
/// (方便本地化),否则必须定义特定于字段的消息。
|
|
|
/// </param>
|
|
|
/// <returns type="undefined" />
|
|
|
|
|
|
$.validator.methods[name] = method;
|
|
|
$.validator.messages[name] = message != undefined ? message : $.validator.messages[name];
|
|
|
if (method.length < 3) {
|
|
|
$.validator.addClassRules(name, $.validator.normalizeRule(name));
|
|
|
}
|
|
|
},
|
|
|
|
|
|
methods: {
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/required
|
|
|
required: function(value, element, param) {
|
|
|
// check if dependency is met
|
|
|
if ( !this.depend(param, element) )
|
|
|
return "dependency-mismatch";
|
|
|
switch( element.nodeName.toLowerCase() ) {
|
|
|
case 'select':
|
|
|
// could be an array for select-multiple or a string, both are fine this way
|
|
|
var val = $(element).val();
|
|
|
return val && val.length > 0;
|
|
|
case 'input':
|
|
|
if ( this.checkable(element) )
|
|
|
return this.getLength(value, element) > 0;
|
|
|
default:
|
|
|
return $.trim(value).length > 0;
|
|
|
}
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/remote
|
|
|
remote: function(value, element, param) {
|
|
|
if ( this.optional(element) )
|
|
|
return "dependency-mismatch";
|
|
|
|
|
|
var previous = this.previousValue(element);
|
|
|
if (!this.settings.messages[element.name] )
|
|
|
this.settings.messages[element.name] = {};
|
|
|
previous.originalMessage = this.settings.messages[element.name].remote;
|
|
|
this.settings.messages[element.name].remote = previous.message;
|
|
|
|
|
|
param = typeof param == "string" && {url:param} || param;
|
|
|
|
|
|
if ( previous.old !== value ) {
|
|
|
previous.old = value;
|
|
|
var validator = this;
|
|
|
this.startRequest(element);
|
|
|
var data = {};
|
|
|
data[element.name] = value;
|
|
|
$.ajax($.extend(true, {
|
|
|
url: param,
|
|
|
mode: "abort",
|
|
|
port: "validate" + element.name,
|
|
|
dataType: "json",
|
|
|
data: data,
|
|
|
success: function(response) {
|
|
|
validator.settings.messages[element.name].remote = previous.originalMessage;
|
|
|
var valid = response === true;
|
|
|
if ( valid ) {
|
|
|
var submitted = validator.formSubmitted;
|
|
|
validator.prepareElement(element);
|
|
|
validator.formSubmitted = submitted;
|
|
|
validator.successList.push(element);
|
|
|
validator.showErrors();
|
|
|
} else {
|
|
|
var errors = {};
|
|
|
var message = (previous.message = response || validator.defaultMessage( element, "remote" ));
|
|
|
errors[element.name] = $.isFunction(message) ? message(value) : message;
|
|
|
validator.showErrors(errors);
|
|
|
}
|
|
|
previous.valid = valid;
|
|
|
validator.stopRequest(element, valid);
|
|
|
}
|
|
|
}, param));
|
|
|
return "pending";
|
|
|
} else if( this.pending[element.name] ) {
|
|
|
return "pending";
|
|
|
}
|
|
|
return previous.valid;
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/minlength
|
|
|
minlength: function(value, element, param) {
|
|
|
return this.optional(element) || this.getLength($.trim(value), element) >= param;
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/maxlength
|
|
|
maxlength: function(value, element, param) {
|
|
|
return this.optional(element) || this.getLength($.trim(value), element) <= param;
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/rangelength
|
|
|
rangelength: function(value, element, param) {
|
|
|
var length = this.getLength($.trim(value), element);
|
|
|
return this.optional(element) || ( length >= param[0] && length <= param[1] );
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/min
|
|
|
min: function( value, element, param ) {
|
|
|
return this.optional(element) || value >= param;
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/max
|
|
|
max: function( value, element, param ) {
|
|
|
return this.optional(element) || value <= param;
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/range
|
|
|
range: function( value, element, param ) {
|
|
|
return this.optional(element) || ( value >= param[0] && value <= param[1] );
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/email
|
|
|
email: function(value, element) {
|
|
|
// contributed by Scott Gonzalez: http://projects.scottsplayground.com/email_address_validation/
|
|
|
return this.optional(element) || /^((([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.test(value);
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/url
|
|
|
url: function(value, element) {
|
|
|
// contributed by Scott Gonzalez: http://projects.scottsplayground.com/iri/
|
|
|
return this.optional(element) || /^(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.test(value);
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/date
|
|
|
date: function(value, element) {
|
|
|
return this.optional(element) || !/Invalid|NaN/.test(new Date(value));
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/dateISO
|
|
|
dateISO: function(value, element) {
|
|
|
return this.optional(element) || /^\d{4}[\/-]\d{1,2}[\/-]\d{1,2}$/.test(value);
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/number
|
|
|
number: function(value, element) {
|
|
|
return this.optional(element) || /^-?(?:\d+|\d{1,3}(?:,\d{3})+)(?:\.\d+)?$/.test(value);
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/digits
|
|
|
digits: function(value, element) {
|
|
|
return this.optional(element) || /^\d+$/.test(value);
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/creditcard
|
|
|
// based on http://en.wikipedia.org/wiki/Luhn
|
|
|
creditcard: function(value, element) {
|
|
|
if ( this.optional(element) )
|
|
|
return "dependency-mismatch";
|
|
|
// accept only digits and dashes
|
|
|
if (/[^0-9-]+/.test(value))
|
|
|
return false;
|
|
|
var nCheck = 0,
|
|
|
nDigit = 0,
|
|
|
bEven = false;
|
|
|
|
|
|
value = value.replace(/\D/g, "");
|
|
|
|
|
|
for (var n = value.length - 1; n >= 0; n--) {
|
|
|
var cDigit = value.charAt(n);
|
|
|
var nDigit = parseInt(cDigit, 10);
|
|
|
if (bEven) {
|
|
|
if ((nDigit *= 2) > 9)
|
|
|
nDigit -= 9;
|
|
|
}
|
|
|
nCheck += nDigit;
|
|
|
bEven = !bEven;
|
|
|
}
|
|
|
|
|
|
return (nCheck % 10) == 0;
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/accept
|
|
|
accept: function(value, element, param) {
|
|
|
param = typeof param == "string" ? param.replace(/,/g, '|') : "png|jpe?g|gif";
|
|
|
return this.optional(element) || value.match(new RegExp(".(" + param + ")$", "i"));
|
|
|
},
|
|
|
|
|
|
// http://docs.jquery.com/Plugins/Validation/Methods/equalTo
|
|
|
equalTo: function(value, element, param) {
|
|
|
// bind to the blur event of the target in order to revalidate whenever the target field is updated
|
|
|
// TODO find a way to bind the event just once, avoiding the unbind-rebind overhead
|
|
|
var target = $(param).unbind(".validate-equalTo").bind("blur.validate-equalTo", function() {
|
|
|
$(element).valid();
|
|
|
});
|
|
|
return value == target.val();
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
// deprecated, use $.validator.format instead
|
|
|
$.format = $.validator.format;
|
|
|
|
|
|
})(jQuery);
|
|
|
|
|
|
// ajax mode: abort
|
|
|
// usage: $.ajax({ mode: "abort"[, port: "uniqueport"]});
|
|
|
// if mode:"abort" is used, the previous request on that port (port can be undefined) is aborted via XMLHttpRequest.abort()
|
|
|
;(function($) {
|
|
|
var ajax = $.ajax;
|
|
|
var pendingRequests = {};
|
|
|
$.ajax = function(settings) {
|
|
|
// create settings for compatibility with ajaxSetup
|
|
|
settings = $.extend(settings, $.extend({}, $.ajaxSettings, settings));
|
|
|
var port = settings.port;
|
|
|
if (settings.mode == "abort") {
|
|
|
if ( pendingRequests[port] ) {
|
|
|
pendingRequests[port].abort();
|
|
|
}
|
|
|
return (pendingRequests[port] = ajax.apply(this, arguments));
|
|
|
}
|
|
|
return ajax.apply(this, arguments);
|
|
|
};
|
|
|
})(jQuery);
|
|
|
|
|
|
// provides cross-browser focusin and focusout events
|
|
|
// IE has native support, in other browsers, use event caputuring (neither bubbles)
|
|
|
|
|
|
// provides delegate(type: String, delegate: Selector, handler: Callback) plugin for easier event delegation
|
|
|
// handler is only called when $(event.target).is(delegate), in the scope of the jquery-object for event.target
|
|
|
|
|
|
// provides triggerEvent(type: String, target: Element) to trigger delegated events
|
|
|
;(function($) {
|
|
|
$.each({
|
|
|
focus: 'focusin',
|
|
|
blur: 'focusout'
|
|
|
}, function( original, fix ){
|
|
|
$.event.special[fix] = {
|
|
|
setup:function() {
|
|
|
if ( $.browser.msie ) return false;
|
|
|
this.addEventListener( original, $.event.special[fix].handler, true );
|
|
|
},
|
|
|
teardown:function() {
|
|
|
if ( $.browser.msie ) return false;
|
|
|
this.removeEventListener( original,
|
|
|
$.event.special[fix].handler, true );
|
|
|
},
|
|
|
handler: function(e) {
|
|
|
arguments[0] = $.event.fix(e);
|
|
|
arguments[0].type = fix;
|
|
|
return $.event.handle.apply(this, arguments);
|
|
|
}
|
|
|
};
|
|
|
});
|
|
|
$.extend($.fn, {
|
|
|
delegate: function(type, delegate, handler) {
|
|
|
return this.bind(type, function(event) {
|
|
|
var target = $(event.target);
|
|
|
if (target.is(delegate)) {
|
|
|
return handler.apply(target, arguments);
|
|
|
}
|
|
|
});
|
|
|
},
|
|
|
triggerEvent: function(type, target) {
|
|
|
return this.triggerHandler(type, [$.event.fix({ type: type, target: target })]);
|
|
|
}
|
|
|
})
|
|
|
})(jQuery);
|
|
|
|
|
|
// SIG // Begin signature block
|
|
|
// SIG // MIIQTAYJKoZIhvcNAQcCoIIQPTCCEDkCAQExCzAJBgUr
|
|
|
// SIG // DgMCGgUAMGcGCisGAQQBgjcCAQSgWTBXMDIGCisGAQQB
|
|
|
// SIG // gjcCAR4wJAIBAQQQEODJBs441BGiowAQS9NQkAIBAAIB
|
|
|
// SIG // AAIBAAIBAAIBADAhMAkGBSsOAwIaBQAEFNY4cct9ZZYB
|
|
|
// SIG // NeMGRlEQXWP2SKiZoIIODzCCBBMwggNAoAMCAQICEGoL
|
|
|
// SIG // mU/AACKrEdsCQnwC074wCQYFKw4DAh0FADB1MSswKQYD
|
|
|
// SIG // VQQLEyJDb3B5cmlnaHQgKGMpIDE5OTkgTWljcm9zb2Z0
|
|
|
// SIG // IENvcnAuMR4wHAYDVQQLExVNaWNyb3NvZnQgQ29ycG9y
|
|
|
// SIG // YXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBUZXN0IFJv
|
|
|
// SIG // b3QgQXV0aG9yaXR5MB4XDTA2MDYyMjIyNTczMVoXDTEx
|
|
|
// SIG // MDYyMTA3MDAwMFowcTELMAkGA1UEBhMCVVMxEzARBgNV
|
|
|
// SIG // BAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1vbmQx
|
|
|
// SIG // HjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlvbjEb
|
|
|
// SIG // MBkGA1UEAxMSTWljcm9zb2Z0IFRlc3QgUENBMIIBIjAN
|
|
|
// SIG // BgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAj/Pz33qn
|
|
|
// SIG // cihhfpDzgWdPPEKAs8NyTe9/EGW4StfGTaxnm6+j/cTt
|
|
|
// SIG // fDRsVXNecQkcoKI69WVT1NzP8zOjWjMsV81IIbelJDAx
|
|
|
// SIG // UzWp2tnbdH9MLnhnzdvJ7bGPt67/eW+sIwZUDiNDN3jd
|
|
|
// SIG // Pk4KbdAq9sZ+W5J0DbMTD1yxcbQQ/LEgCAgueW5f0nI0
|
|
|
// SIG // rpI6gbAyrM5DWTCmwfyu+MzofYZrXK7r3pX6Kjl1BlxB
|
|
|
// SIG // OlHcVzVOksssnXuk3Jrp/iGcYR87pEx/UrGFOWR9kYlv
|
|
|
// SIG // nhRCs7yi2moXhyTmG9V8fY+q3ALJoV7d/YEqnybDNkHT
|
|
|
// SIG // z/xzDRx0KDjypQrF0Q+7077QkwIDAQABo4HrMIHoMIGo
|
|
|
// SIG // BgNVHQEEgaAwgZ2AEMBjRdejAX15xXp6XyjbQ9ahdzB1
|
|
|
// SIG // MSswKQYDVQQLEyJDb3B5cmlnaHQgKGMpIDE5OTkgTWlj
|
|
|
// SIG // cm9zb2Z0IENvcnAuMR4wHAYDVQQLExVNaWNyb3NvZnQg
|
|
|
// SIG // Q29ycG9yYXRpb24xJjAkBgNVBAMTHU1pY3Jvc29mdCBU
|
|
|
// SIG // ZXN0IFJvb3QgQXV0aG9yaXR5ghBf6k/S8h1DELboVD7Y
|
|
|
// SIG // lSYYMA8GA1UdEwEB/wQFMAMBAf8wHQYDVR0OBBYEFFSl
|
|
|
// SIG // IUygrm+cYE4Pzt1G1ddh1hesMAsGA1UdDwQEAwIBhjAJ
|
|
|
// SIG // BgUrDgMCHQUAA4HBACzODwWw7h9lGeKjJ7yc936jJard
|
|
|
// SIG // LMfrxQKBMZfJTb9MWDDIJ9WniM6epQ7vmTWM9Q4cLMy2
|
|
|
// SIG // kMGgdc3mffQLETF6g/v+aEzFG5tUqingK125JFP57MGc
|
|
|
// SIG // JYMlQGO3KUIcedPC8cyj+oYwi6tbSpDLRCCQ7MAFS15r
|
|
|
// SIG // 4Dnxn783pZ5nSXh1o+NrSz5mbGusDIj0ujHBCqblI96+
|
|
|
// SIG // Rk7oVQ2DI3oQkSmGQf+BrmRXoJfB3YuXXFc+F88beLHS
|
|
|
// SIG // F0S8oJhPjzCCBKgwggOQoAMCAQICCmEBi3MAAAAAABMw
|
|
|
// SIG // DQYJKoZIhvcNAQEFBQAwcTELMAkGA1UEBhMCVVMxEzAR
|
|
|
// SIG // BgNVBAgTCldhc2hpbmd0b24xEDAOBgNVBAcTB1JlZG1v
|
|
|
// SIG // bmQxHjAcBgNVBAoTFU1pY3Jvc29mdCBDb3Jwb3JhdGlv
|
|
|
// SIG // bjEbMBkGA1UEAxMSTWljcm9zb2Z0IFRlc3QgUENBMB4X
|
|
|
// SIG // DTA5MDgxNzIzMjAxN1oXDTExMDYyMTA3MDAwMFowgYEx
|
|
|
// SIG // EzARBgoJkiaJk/IsZAEZFgNjb20xGTAXBgoJkiaJk/Is
|
|
|
// SIG // ZAEZFgltaWNyb3NvZnQxFDASBgoJkiaJk/IsZAEZFgRj
|
|
|
// SIG // b3JwMRcwFQYKCZImiZPyLGQBGRYHcmVkbW9uZDEgMB4G
|
|
|
// SIG // A1UEAxMXTVNJVCBUZXN0IENvZGVTaWduIENBIDEwggEi
|
|
|
// SIG // MA0GCSqGSIb3DQEBAQUAA4IBDwAwggEKAoIBAQDKz+fW
|
|
|
// SIG // ilZnvB1mb2XQEkuK0GeO6we7n8RfXKMFTp9ifiOnD0v5
|
|
|
// SIG // FFYrPjAKGMrOxroVu8rPTOPukz6hlYdMzIkV68iyS4FU
|
|
|
// SIG // ZjQGz5wQNnLKbUN1PFlP+NsWJZjzvRuZv9WWweCKnUeE
|
|
|
// SIG // Fxur+rzMtvz50aVAechNt36xI6rIxVXRv5xvDKzkKTGv
|
|
|
// SIG // BmaP0YsqkNcUe3GJy17yWoEWX+kKGX69xNezEai06On2
|
|
|
// SIG // cpKToU0ibyRNhgs2Ygzb5U/9hISMYt7YFdEYggL0zTNp
|
|
|
// SIG // 59hmfaB5FT0yMor1iUcSFVtTGObPmB1dsD4EPcYSTZtp
|
|
|
// SIG // 5R4hzYecLp8kSV78s1ycVDt5pQY1AgMBAAGjggEvMIIB
|
|
|
// SIG // KzAQBgkrBgEEAYI3FQEEAwIBADAdBgNVHQ4EFgQUhOTQ
|
|
|
// SIG // p5jIj+9WN5bdvfFGrMW5xZ4wGQYJKwYBBAGCNxQCBAwe
|
|
|
// SIG // CgBTAHUAYgBDAEEwCwYDVR0PBAQDAgGGMA8GA1UdEwEB
|
|
|
// SIG // /wQFMAMBAf8wHwYDVR0jBBgwFoAUVKUhTKCub5xgTg/O
|
|
|
// SIG // 3UbV12HWF6wwTAYDVR0fBEUwQzBBoD+gPYY7aHR0cDov
|
|
|
// SIG // L2NybC5taWNyb3NvZnQuY29tL3BraS9jcmwvcHJvZHVj
|
|
|
// SIG // dHMvbGVnYWN5dGVzdHBjYS5jcmwwUAYIKwYBBQUHAQEE
|
|
|
// SIG // RDBCMEAGCCsGAQUFBzAChjRodHRwOi8vd3d3Lm1pY3Jv
|
|
|
// SIG // c29mdC5jb20vcGtpL2NlcnRzL0xlZ2FjeVRlc3RQQ0Eu
|
|
|
// SIG // Y3J0MA0GCSqGSIb3DQEBBQUAA4IBAQA4GSVNJtByD1os
|
|
|
// SIG // xEzGCLI18ykM+RrR02D1DyopRstCY+OoOeX5WX5BVknd
|
|
|
// SIG // j0w6P1Ea4TD450ozSN7q1yWQcgIT2K8DbwyKTnDn5enx
|
|
|
// SIG // josg2n+ljxnLputPDiFAdfNP+XHew9x/gB+JR7oSK/Ps
|
|
|
// SIG // LzXbuVITIRDkPogIJUFQMrwKI9o0bv2sLWV+fSk+fEXB
|
|
|
// SIG // OaHysKBGU+EIhjrfHx4QP38jQUi2yJZQ85klqVVuSL21
|
|
|
// SIG // dwIP5QZiYplN6zicK6ez3r+yozQLOg6mc5MBgrUPBTsV
|
|
|
// SIG // sSbHM2+BGVaorOyI7JMr0sHBl6IGFbqRIPqtWY4rimD8
|
|
|
// SIG // uNi6hHfLJFmTMDstbNJEMIIFSDCCBDCgAwIBAgIKa4DO
|
|
|
// SIG // qQAAAACmmDANBgkqhkiG9w0BAQUFADCBgTETMBEGCgmS
|
|
|
// SIG // JomT8ixkARkWA2NvbTEZMBcGCgmSJomT8ixkARkWCW1p
|
|
|
// SIG // Y3Jvc29mdDEUMBIGCgmSJomT8ixkARkWBGNvcnAxFzAV
|
|
|
// SIG // BgoJkiaJk/IsZAEZFgdyZWRtb25kMSAwHgYDVQQDExdN
|
|
|
// SIG // U0lUIFRlc3QgQ29kZVNpZ24gQ0EgMTAeFw0wOTExMDYx
|
|
|
// SIG // ODE3MDdaFw0xMDExMDYxODE3MDdaMBUxEzARBgNVBAMT
|
|
|
// SIG // ClZTIEJsZCBMYWIwgZ8wDQYJKoZIhvcNAQEBBQADgY0A
|
|
|
// SIG // MIGJAoGBAJMiPNeJy8vp5oeABJLebUDw5LUKy+N3pOFp
|
|
|
// SIG // h5QGJmE4b4JgN2LEXNVLh6lOle35xLCbQOJCVs1eDOgq
|
|
|
// SIG // puOWq5EvFYOugrxGcS4wfHNt4/Rwjigo/UQDYU755puL
|
|
|
// SIG // RBqLVtGqlcMYwLhzAWV0R7HWtmBDfhqAH19O3P3foI2X
|
|
|
// SIG // zrLrAgMBAAGjggKvMIICqzALBgNVHQ8EBAMCB4AwHQYD
|
|
|
// SIG // VR0OBBYEFAjpDmzPyPih2x+qdItA5Ul2ZAe3MD0GCSsG
|
|
|
// SIG // AQQBgjcVBwQwMC4GJisGAQQBgjcVCIPPiU2t8gKFoZ8M
|
|
|
// SIG // gvrKfYHh+3SBT4KusGqH9P0yAgFkAgEMMB8GA1UdIwQY
|
|
|
// SIG // MBaAFITk0KeYyI/vVjeW3b3xRqzFucWeMIHoBgNVHR8E
|
|
|
// SIG // geAwgd0wgdqggdeggdSGNmh0dHA6Ly9jb3JwcGtpL2Ny
|
|
|
// SIG // bC9NU0lUJTIwVGVzdCUyMENvZGVTaWduJTIwQ0ElMjAx
|
|
|
// SIG // LmNybIZNaHR0cDovL21zY3JsLm1pY3Jvc29mdC5jb20v
|
|
|
// SIG // cGtpL21zY29ycC9jcmwvTVNJVCUyMFRlc3QlMjBDb2Rl
|
|
|
// SIG // U2lnbiUyMENBJTIwMS5jcmyGS2h0dHA6Ly9jcmwubWlj
|
|
|
// SIG // cm9zb2Z0LmNvbS9wa2kvbXNjb3JwL2NybC9NU0lUJTIw
|
|
|
// SIG // VGVzdCUyMENvZGVTaWduJTIwQ0ElMjAxLmNybDCBqQYI
|
|
|
// SIG // KwYBBQUHAQEEgZwwgZkwQgYIKwYBBQUHMAKGNmh0dHA6
|
|
|
// SIG // Ly9jb3JwcGtpL2FpYS9NU0lUJTIwVGVzdCUyMENvZGVT
|
|
|
// SIG // aWduJTIwQ0ElMjAxLmNydDBTBggrBgEFBQcwAoZHaHR0
|
|
|
// SIG // cDovL3d3dy5taWNyb3NvZnQuY29tL3BraS9tc2NvcnAv
|
|
|
// SIG // TVNJVCUyMFRlc3QlMjBDb2RlU2lnbiUyMENBJTIwMS5j
|
|
|
// SIG // cnQwHwYDVR0lBBgwFgYKKwYBBAGCNwoDBgYIKwYBBQUH
|
|
|
// SIG // AwMwKQYJKwYBBAGCNxUKBBwwGjAMBgorBgEEAYI3CgMG
|
|
|
// SIG // MAoGCCsGAQUFBwMDMDoGA1UdEQQzMDGgLwYKKwYBBAGC
|
|
|
// SIG // NxQCA6AhDB9kbGFiQHJlZG1vbmQuY29ycC5taWNyb3Nv
|
|
|
// SIG // ZnQuY29tMA0GCSqGSIb3DQEBBQUAA4IBAQBqcp669vuu
|
|
|
// SIG // QzcKv0NTjeY2jhqSYRlwon/Q83ON8GCb1vf3AEFmwPNI
|
|
|
// SIG // 5hxSmGpqr4JrfuJFFa6SxO8praB4oaZeTKt7bAH/uRpb
|
|
|
// SIG // HP3U8Y6tuJAzfWaYUiNoF02lpgFEa44pw3sGJ3XA6uj0
|
|
|
// SIG // cG4jo1U5b81pkFblA4WRIuU1VHUDmARJbinQVt3JAFyU
|
|
|
// SIG // /J4SuAMUxraGUS8voUpk/Jyy8A7dhNepQQmc8BlY6lIQ
|
|
|
// SIG // fyU6WYQhOSuuQO5mfZhJaFGA53gqWzJfVBD32i7O6lAt
|
|
|
// SIG // /SXE7oV+Fwo5FHC8dOMzIn4bITvDQxgfO0M530uBmnCY
|
|
|
// SIG // qsRRYNNgYql6JvUjP/DSy6ZfMYIBqTCCAaUCAQEwgZAw
|
|
|
// SIG // gYExEzARBgoJkiaJk/IsZAEZFgNjb20xGTAXBgoJkiaJ
|
|
|
// SIG // k/IsZAEZFgltaWNyb3NvZnQxFDASBgoJkiaJk/IsZAEZ
|
|
|
// SIG // FgRjb3JwMRcwFQYKCZImiZPyLGQBGRYHcmVkbW9uZDEg
|
|
|
// SIG // MB4GA1UEAxMXTVNJVCBUZXN0IENvZGVTaWduIENBIDEC
|
|
|
// SIG // CmuAzqkAAAAAppgwCQYFKw4DAhoFAKBwMBAGCisGAQQB
|
|
|
// SIG // gjcCAQwxAjAAMBkGCSqGSIb3DQEJAzEMBgorBgEEAYI3
|
|
|
// SIG // AgEEMBwGCisGAQQBgjcCAQsxDjAMBgorBgEEAYI3AgEV
|
|
|
// SIG // MCMGCSqGSIb3DQEJBDEWBBQh9OxAzta5d4KXioUjtJT4
|
|
|
// SIG // 1rkiqjANBgkqhkiG9w0BAQEFAASBgFe2dNy4hXiECsBs
|
|
|
// SIG // CP1CfpVy1wYUdR38XO73/7bvr/chqCvJptA9f3yLVvSe
|
|
|
// SIG // Wc7jRS7kxbYkBlmRy5ggT3qW19bVBh3vQt5CX6IBlZ2S
|
|
|
// SIG // G5ElZcMy6WzrEGULSWW2oFm7s6w2nz6tNoiwu3f+IhaA
|
|
|
// SIG // lmniacDet5Cl87janHUk2d3I
|
|
|
// SIG // End signature block
|