diff options
Diffstat (limited to 'includes/js/dojox/validate')
-rw-r--r-- | includes/js/dojox/validate/README | 37 | ||||
-rw-r--r-- | includes/js/dojox/validate/_base.js | 183 | ||||
-rw-r--r-- | includes/js/dojox/validate/ca.js | 44 | ||||
-rw-r--r-- | includes/js/dojox/validate/check.js | 261 | ||||
-rw-r--r-- | includes/js/dojox/validate/creditCard.js | 92 | ||||
-rw-r--r-- | includes/js/dojox/validate/isbn.js | 37 | ||||
-rw-r--r-- | includes/js/dojox/validate/regexp.js | 331 | ||||
-rw-r--r-- | includes/js/dojox/validate/tests/creditcard.js | 119 | ||||
-rw-r--r-- | includes/js/dojox/validate/tests/module.js | 14 | ||||
-rw-r--r-- | includes/js/dojox/validate/tests/runTests.html | 9 | ||||
-rw-r--r-- | includes/js/dojox/validate/tests/validate.js | 316 | ||||
-rw-r--r-- | includes/js/dojox/validate/us.js | 67 | ||||
-rw-r--r-- | includes/js/dojox/validate/web.js | 89 |
13 files changed, 1599 insertions, 0 deletions
diff --git a/includes/js/dojox/validate/README b/includes/js/dojox/validate/README new file mode 100644 index 0000000..795afcb --- /dev/null +++ b/includes/js/dojox/validate/README @@ -0,0 +1,37 @@ +------------------------------------------------------------------------------- +dojox.validate +------------------------------------------------------------------------------- +Version 0.01 +Release date: 07/12/2007 +------------------------------------------------------------------------------- +Project state: experimental / beta +------------------------------------------------------------------------------- +Credits + port: Peter Higgins (dante) + contributions: Kun Xi (bookstack@gmail.com) +------------------------------------------------------------------------------- +Project description + + Provides a set of validation functions to match + values against known constants for use in form + validation, such as email address, TLD, ipAddress, + country-specific phone numbers and SSN, among + others.. + +------------------------------------------------------------------------------- +Dependencies: + + Requires dojo base and dojo.regexp. + +------------------------------------------------------------------------------- +Installation instructions + +Grab the following from the Dojo SVN Repository: + +http://svn.dojotoolkit.org/dojo/dojox/trunk/validate.js +http://svn.dojotoolkit.org/dojo/dojox/trunk/validate + +Install into the following directory structure: +/dojox/validate/ + +...which should be at the same level as your Dojo checkout. diff --git a/includes/js/dojox/validate/_base.js b/includes/js/dojox/validate/_base.js new file mode 100644 index 0000000..9dbba59 --- /dev/null +++ b/includes/js/dojox/validate/_base.js @@ -0,0 +1,183 @@ +if(!dojo._hasResource["dojox.validate._base"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate._base"] = true; +dojo.provide("dojox.validate._base"); + +dojo.require("dojo.regexp"); // dojo core expressions +dojo.require("dojo.number"); // dojo number expressions +dojo.require("dojox.validate.regexp"); // additional expressions + +dojox.validate.isText = function(/*String*/value, /*Object?*/flags){ + // summary: + // Checks if a string has non whitespace characters. + // Parameters allow you to constrain the length. + // + // value: A string + // flags: {length: Number, minlength: Number, maxlength: Number} + // flags.length If set, checks if there are exactly flags.length number of characters. + // flags.minlength If set, checks if there are at least flags.minlength number of characters. + // flags.maxlength If set, checks if there are at most flags.maxlength number of characters. + + flags = (typeof flags == "object") ? flags : {}; + + // test for text + if(/^\s*$/.test(value)){ return false; } // Boolean + + // length tests + if(typeof flags.length == "number" && flags.length != value.length){ return false; } // Boolean + if(typeof flags.minlength == "number" && flags.minlength > value.length){ return false; } // Boolean + if(typeof flags.maxlength == "number" && flags.maxlength < value.length){ return false; } // Boolean + + return true; // Boolean + +} + +dojox.validate._isInRangeCache = {}; +dojox.validate.isInRange = function(/*String*/value, /*Object?*/flags){ + // summary: + // Validates whether a string denoting an integer, + // real number, or monetary value is between a max and min. + // + // value: A string + // flags: {max:Number, min:Number, decimal:String} + // flags.max A number, which the value must be less than or equal to for the validation to be true. + // flags.min A number, which the value must be greater than or equal to for the validation to be true. + // flags.decimal The character used for the decimal point. Default is ".". + + // fixes ticket #2908 + value = dojo.number.parse(value, flags); + if(isNaN(value)){ + return false; // Boolean + } + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + var max = (typeof flags.max == "number") ? flags.max : Infinity; + var min = (typeof flags.min == "number") ? flags.min : -Infinity; + var dec = (typeof flags.decimal == "string") ? flags.decimal : "."; + + var cache = dojox.validate._isInRangeCache; + var cacheIdx = value+"max"+max+"min"+min+"dec"+dec; + if(typeof cache[cacheIdx] != "undefined"){ + return cache[cacheIdx]; + } + + if ( value < min || value > max ) { cache[cacheIdx] = false; return false; } // Boolean + + cache[cacheIdx] = true; return true; // Boolean +} + +dojox.validate.isNumberFormat = function(/*String*/value, /*Object?*/flags){ + // summary: + // Validates any sort of number based format + // + // description: + // Use it for phone numbers, social security numbers, zip-codes, etc. + // The value can be validated against one format or one of multiple formats. + // + // Format + // # Stands for a digit, 0-9. + // ? Stands for an optional digit, 0-9 or nothing. + // All other characters must appear literally in the expression. + // + // Example + // "(###) ###-####" -> (510) 542-9742 + // "(###) ###-#### x#???" -> (510) 542-9742 x153 + // "###-##-####" -> 506-82-1089 i.e. social security number + // "#####-####" -> 98225-1649 i.e. zip code + // + // value: A string + // flags: {format:String} + // flags.format A string or an Array of strings for multiple formats. + + var re = new RegExp("^" + dojox.regexp.numberFormat(flags) + "$", "i"); + return re.test(value); // Boolean +} + +dojox.validate.isValidLuhn = function(/*String*/value){ + //summary: Compares value against the Luhn algorithm to verify its integrity + var sum, parity, curDigit; + if(typeof value!='string'){ + value = String(value); + } + value = value.replace(/[- ]/g,''); //ignore dashes and whitespaces + parity = value.length%2; + sum=0; + for(var i=0;i<value.length;i++){ + curDigit = parseInt(value.charAt(i)); + if(i%2==parity){ + curDigit*=2; + } + if(curDigit>9){ + curDigit-=9; + } + sum+=curDigit; + } + return !(sum%10); //Boolean +} + +/** + Procedural API Description + + The main aim is to make input validation expressible in a simple format. + You define profiles which declare the required and optional fields and any constraints they might have. + The results are provided as an object that makes it easy to handle missing and invalid input. + + Usage + + var results = dojo.validate.check(form, profile); + + Profile Object + + var profile = { + // filters change the field value and are applied before validation. + trim: ["tx1", "tx2"], + uppercase: ["tx9"], + lowercase: ["tx5", "tx6", "tx7"], + ucfirst: ["tx10"], + digit: ["tx11"], + + // required input fields that are blank will be reported missing. + // required radio button groups and drop-down lists with no selection will be reported missing. + // checkbox groups and selectboxes can be required to have more than one value selected. + // List required fields by name and use this notation to require more than one value: {checkboxgroup: 2}, {selectboxname: 3}. + required: ["tx7", "tx8", "pw1", "ta1", "rb1", "rb2", "cb3", "s1", {"doubledip":2}, {"tripledip":3}], + + // dependant/conditional fields are required if the target field is present and not blank. + // At present only textbox, password, and textarea fields are supported. + dependencies: { + cc_exp: "cc_no", + cc_type: "cc_no", + }, + + // Fields can be validated using any boolean valued function. + // Use arrays to specify parameters in addition to the field value. + constraints: { + field_name1: myValidationFunction, + field_name2: dojo.validate.isInteger, + field_name3: [myValidationFunction, additional parameters], + field_name4: [dojo.validate.isValidDate, "YYYY.MM.DD"], + field_name5: [dojo.validate.isEmailAddress, false, true], + }, + + // Confirm is a sort of conditional validation. + // It associates each field in its property list with another field whose value should be equal. + // If the values are not equal, the field in the property list is reported as Invalid. Unless the target field is blank. + confirm: { + email_confirm: "email", + pw2: "pw1", + } + }; + + Results Object + + isSuccessful(): Returns true if there were no invalid or missing fields, else it returns false. + hasMissing(): Returns true if the results contain any missing fields. + getMissing(): Returns a list of required fields that have values missing. + isMissing(field): Returns true if the field is required and the value is missing. + hasInvalid(): Returns true if the results contain fields with invalid data. + getInvalid(): Returns a list of fields that have invalid values. + isInvalid(field): Returns true if the field has an invalid value. + +*/ + +} diff --git a/includes/js/dojox/validate/ca.js b/includes/js/dojox/validate/ca.js new file mode 100644 index 0000000..dbd3aa5 --- /dev/null +++ b/includes/js/dojox/validate/ca.js @@ -0,0 +1,44 @@ +if(!dojo._hasResource["dojox.validate.ca"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.ca"] = true; +dojo.provide("dojox.validate.ca"); + +dojo.require("dojox.validate._base"); + +dojox.validate.ca.isPhoneNumber = function(/* String */value) { + // summary: Validates 10 Canadian digit phone number for several common formats + // returns: Boolean + return dojox.validate.us.isPhoneNumber(value); // same as US +}; + +dojox.validate.ca.isProvince = function(/* String[2] */value) { + // summary: Validates Canadian province abbreviations (2 chars) + // returns: Boolean + var re = new RegExp("^" + dojox.regexp.ca.province() + "$", "i"); + return re.test(value); +}; + +dojox.validate.ca.isSocialInsuranceNumber = function(/* String */value) { + // summary: Validates Canadian 9 digit social insurance number for several common formats + // This routine only pattern matches and does not use the Luhn Algorithm to validate number. + // returns: Boolean + var flags = { + format: [ + "###-###-###", + "### ### ###", + "#########" + ] + }; + return dojox.validate.isNumberFormat(value, flags); +}; + +dojox.validate.ca.isPostalCode = function(value) { + // summary: Validates Canadian 6 digit postal code: + // Canadian postal codes are in the format ANA NAN, + // where A is a letter and N is a digit, with a space + // separating the third and fourth characters. + // returns: Boolean + var re = new RegExp("^" + dojox.regexp.ca.postalCode() + "$", "i"); + return re.test(value); +}; + +} diff --git a/includes/js/dojox/validate/check.js b/includes/js/dojox/validate/check.js new file mode 100644 index 0000000..015da79 --- /dev/null +++ b/includes/js/dojox/validate/check.js @@ -0,0 +1,261 @@ +if(!dojo._hasResource["dojox.validate.check"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.check"] = true; +dojo.provide("dojox.validate.check"); + +dojo.require("dojox.validate._base"); + +dojox.validate.check = function(/*HTMLFormElement*/form, /*Object*/profile){ + // summary: validates user input of an HTML form based on input profile + // + // description: + // returns an object that contains several methods summarizing the results of the validation + // + // form: form to be validated + // profile: specifies how the form fields are to be validated + // {trim:Array, uppercase:Array, lowercase:Array, ucfirst:Array, digit:Array, + // required:Array, dependencies:Object, constraints:Object, confirm:Object} + + // Essentially private properties of results object + var missing = []; + var invalid = []; + + // results object summarizes the validation + var results = { + isSuccessful: function() {return ( !this.hasInvalid() && !this.hasMissing() );}, + hasMissing: function() {return ( missing.length > 0 );}, + getMissing: function() {return missing;}, + isMissing: function(elemname) { + for(var i = 0; i < missing.length; i++){ + if(elemname == missing[i]){ return true; } + } + return false; + }, + hasInvalid: function() {return ( invalid.length > 0 );}, + getInvalid: function() {return invalid;}, + isInvalid: function(elemname){ + for(var i = 0; i < invalid.length; i++){ + if(elemname == invalid[i]){ return true; } + } + return false; + } + }; + + var _undef = function(name,object){ + return (typeof object[name] == "undefined"); + }; + + // Filters are applied before fields are validated. + // Trim removes white space at the front and end of the fields. + if(profile.trim instanceof Array){ + for(var i = 0; i < profile.trim.length; i++){ + var elem = form[profile.trim[i]]; + if(_undef("type", elem) || elem.type != "text" && elem.type != "textarea" && elem.type != "password"){ continue; } + elem.value = elem.value.replace(/(^\s*|\s*$)/g, ""); + } + } + // Convert to uppercase + if(profile.uppercase instanceof Array){ + for(var i = 0; i < profile.uppercase.length; i++){ + var elem = form[profile.uppercase[i]]; + if(_undef("type", elem) || elem.type != "text" && elem.type != "textarea" && elem.type != "password"){ continue; } + elem.value = elem.value.toUpperCase(); + } + } + // Convert to lowercase + if(profile.lowercase instanceof Array){ + for (var i = 0; i < profile.lowercase.length; i++){ + var elem = form[profile.lowercase[i]]; + if(_undef("type", elem) || elem.type != "text" && elem.type != "textarea" && elem.type != "password"){ continue; } + elem.value = elem.value.toLowerCase(); + } + } + // Uppercase first letter + if(profile.ucfirst instanceof Array){ + for(var i = 0; i < profile.ucfirst.length; i++){ + var elem = form[profile.ucfirst[i]]; + if(_undef("type", elem) || elem.type != "text" && elem.type != "textarea" && elem.type != "password"){ continue; } + elem.value = elem.value.replace(/\b\w+\b/g, function(word) { return word.substring(0,1).toUpperCase() + word.substring(1).toLowerCase(); }); + } + } + // Remove non digits characters from the input. + if(profile.digit instanceof Array){ + for(var i = 0; i < profile.digit.length; i++){ + var elem = form[profile.digit[i]]; + if(_undef("type", elem) || elem.type != "text" && elem.type != "textarea" && elem.type != "password"){ continue; } + elem.value = elem.value.replace(/\D/g, ""); + } + } + + // See if required input fields have values missing. + if(profile.required instanceof Array){ + for(var i = 0; i < profile.required.length; i++){ + if(!dojo.isString(profile.required[i])){ continue; } + var elem = form[profile.required[i]]; + // Are textbox, textarea, or password fields blank. + if(!_undef("type", elem) + && (elem.type == "text" || elem.type == "textarea" || elem.type == "password" || elem.type == "file") + && /^\s*$/.test(elem.value)){ + missing[missing.length] = elem.name; + } + // Does drop-down box have option selected. + else if(!_undef("type", elem) && (elem.type == "select-one" || elem.type == "select-multiple") + && (elem.selectedIndex == -1 + || /^\s*$/.test(elem.options[elem.selectedIndex].value))){ + missing[missing.length] = elem.name; + } + // Does radio button group (or check box group) have option checked. + else if(elem instanceof Array){ + var checked = false; + for(var j = 0; j < elem.length; j++){ + if (elem[j].checked) { checked = true; } + } + if(!checked){ + missing[missing.length] = elem[0].name; + } + } + } + } + + // See if checkbox groups and select boxes have x number of required values. + if(profile.required instanceof Array){ + for (var i = 0; i < profile.required.length; i++){ + if(!dojo.isObject(profile.required[i])){ continue; } + var elem, numRequired; + for(var name in profile.required[i]){ + elem = form[name]; + numRequired = profile.required[i][name]; + } + // case 1: elem is a check box group + if(elem instanceof Array){ + var checked = 0; + for(var j = 0; j < elem.length; j++){ + if(elem[j].checked){ checked++; } + } + if(checked < numRequired){ + missing[missing.length] = elem[0].name; + } + } + // case 2: elem is a select box + else if(!_undef("type", elem) && elem.type == "select-multiple" ){ + var selected = 0; + for(var j = 0; j < elem.options.length; j++){ + if (elem.options[j].selected && !/^\s*$/.test(elem.options[j].value)) { selected++; } + } + if(selected < numRequired){ + missing[missing.length] = elem.name; + } + } + } + } + + // Dependent fields are required when the target field is present (not blank). + // Todo: Support dependent and target fields that are radio button groups, or select drop-down lists. + // Todo: Make the dependency based on a specific value of the target field. + // Todo: allow dependent fields to have several required values, like {checkboxgroup: 3}. + if(dojo.isObject(profile.dependencies)){ + // properties of dependencies object are the names of dependent fields to be checked + for(name in profile.dependencies){ + var elem = form[name]; // the dependent element + if(_undef("type", elem)){continue;} + if(elem.type != "text" && elem.type != "textarea" && elem.type != "password"){ continue; } // limited support + if(/\S+/.test(elem.value)){ continue; } // has a value already + if(results.isMissing(elem.name)){ continue; } // already listed as missing + var target = form[profile.dependencies[name]]; + if(target.type != "text" && target.type != "textarea" && target.type != "password"){ continue; } // limited support + if(/^\s*$/.test(target.value)){ continue; } // skip if blank + missing[missing.length] = elem.name; // ok the dependent field is missing + } + } + + // Find invalid input fields. + if(dojo.isObject(profile.constraints)){ + // constraint properties are the names of fields to bevalidated + for(name in profile.constraints){ + var elem = form[name]; + if(!elem) {continue;} + + // skip if blank - its optional unless required, in which case it + // is already listed as missing. + if(!_undef("tagName",elem) + && (elem.tagName.toLowerCase().indexOf("input") >= 0 + || elem.tagName.toLowerCase().indexOf("textarea") >= 0) + && /^\s*$/.test(elem.value)){ + continue; + } + + var isValid = true; + // case 1: constraint value is validation function + if(dojo.isFunction(profile.constraints[name])){ + isValid = profile.constraints[name](elem.value); + }else if(dojo.isArray(profile.constraints[name])){ + + // handle nested arrays for multiple constraints + if(dojo.isArray(profile.constraints[name][0])){ + for (var i=0; i<profile.constraints[name].length; i++){ + isValid = dojox.validate.evaluateConstraint(profile, profile.constraints[name][i], name, elem); + if(!isValid){ break; } + } + }else{ + // case 2: constraint value is array, first elem is function, + // tail is parameters + isValid = dojox.validate.evaluateConstraint(profile, profile.constraints[name], name, elem); + } + } + + if(!isValid){ + invalid[invalid.length] = elem.name; + } + } + } + + // Find unequal confirm fields and report them as Invalid. + if(dojo.isObject(profile.confirm)){ + for(name in profile.confirm){ + var elem = form[name]; // the confirm element + var target = form[profile.confirm[name]]; + if (_undef("type", elem) || _undef("type", target) || (elem.type != "text" && elem.type != "textarea" && elem.type != "password") + ||(target.type != elem.type) + ||(target.value == elem.value) // it's valid + ||(results.isInvalid(elem.name))// already listed as invalid + ||(/^\s*$/.test(target.value))) // skip if blank - only confirm if target has a value + { + continue; + } + invalid[invalid.length] = elem.name; + } + } + return results; // Object +}; + +//TODO: evaluateConstraint doesn't use profile or fieldName args? +dojox.validate.evaluateConstraint=function(profile, /*Array*/constraint, fieldName, elem){ + // summary: + // Evaluates dojo.validate.check() constraints that are specified as array + // arguments + // + // description: The arrays are expected to be in the format of: + // constraints:{ + // fieldName: [functionToCall, param1, param2, etc.], + // fieldName: [[functionToCallFirst, param1],[functionToCallSecond,param2]] + // } + // + // This function evaluates a single array function in the format of: + // [functionName, argument1, argument2, etc] + // + // The function will be parsed out and evaluated against the incoming parameters. + // + // profile: The dojo.validate.check() profile that this evaluation is against. + // constraint: The single [] array of function and arguments for the function. + // fieldName: The form dom name of the field being validated. + // elem: The form element field. + + var isValidSomething = constraint[0]; + var params = constraint.slice(1); + params.unshift(elem.value); + if(typeof isValidSomething != "undefined"){ + return isValidSomething.apply(null, params); + } + return false; // Boolean +} + +} diff --git a/includes/js/dojox/validate/creditCard.js b/includes/js/dojox/validate/creditCard.js new file mode 100644 index 0000000..903aab1 --- /dev/null +++ b/includes/js/dojox/validate/creditCard.js @@ -0,0 +1,92 @@ +if(!dojo._hasResource["dojox.validate.creditCard"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.creditCard"] = true; +dojo.provide("dojox.validate.creditCard"); + +dojo.require("dojox.validate._base"); + +/* + Validates Credit Cards using account number rules in conjunction with the Luhn algorigthm + + */ + +dojox.validate.isValidCreditCard = function(/*String|Int*/value, /*String*/ccType){ + //Summary: + // checks if ccType matches the # scheme in value, and if Luhn checksum is accurate (unless its an Enroute card, the checkSum is skipped) + + //Value: Boolean + return ((ccType.toLowerCase() == 'er' || dojox.validate.isValidLuhn(value)) && + dojox.validate.isValidCreditCardNumber(value,ccType.toLowerCase())); //Boolean +} + +dojox.validate.isValidCreditCardNumber = function(/*String|Int*/value,/*String?*/ccType){ + // summary: + // checks if value matches the pattern for that card or any card types if none is specified + // + // value: Boolean + // CC #, white spaces and dashes are ignored + // + // ccType: String? + // one of the values in cardinfo -- if Omitted it it returns a | delimited string of matching card types, or false if no matches found + + value = String(value).replace(/[- ]/g,''); //ignore dashes and whitespaces + + /* FIXME: not sure on all the abbreviations for credit cards,below is what each stands for atleast to my knowledge + mc: Mastercard + ec: Eurocard + vi: Visa + ax: American Express + dc: Diners Club + bl: Carte Blanch + di: Discover + jcb: JCB + er: Enroute + */ + var cardinfo = { + 'mc':'5[1-5][0-9]{14}','ec':'5[1-5][0-9]{14}','vi':'4(?:[0-9]{12}|[0-9]{15})', + 'ax':'3[47][0-9]{13}', 'dc':'3(?:0[0-5][0-9]{11}|[68][0-9]{12})', + 'bl':'3(?:0[0-5][0-9]{11}|[68][0-9]{12})','di':'6011[0-9]{12}', + 'jcb':'(?:3[0-9]{15}|(2131|1800)[0-9]{11})','er':'2(?:014|149)[0-9]{11}' + }; + if(ccType){ + var expr = cardinfo[ccType.toLowerCase()]; + return expr ? !!(value.match(cardinfo[ccType.toLowerCase()])) : false; // boolean + } + var results=[]; + for(var p in cardinfo){ + if(value.match('^'+cardinfo[p]+'$')){ + results.push(p); + } + } + return results.length ? results.join('|') : false; // string | boolean +} + +dojox.validate.isValidCvv = function(/*String|Int*/value, /*String*/ccType) { + //Summary: + // returns true if the security code (CCV) matches the correct format for supplied ccType + + //Value: Boolean + + if(typeof value!='string'){ + value=String(value); + } + var format; + switch (ccType.toLowerCase()){ + case 'mc': + case 'ec': + case 'vi': + case 'di': + format = '###'; + break; + case 'ax': + format = '####'; + break; + default: + return false; //Boolean + } + var flags = {format:format}; + //FIXME? Why does isNumberFormat take an object for flags when its only parameter is either a string or an array inside the object? + //FIXME: um... just check value.length? + return (value.length == format.length && dojox.validate.isNumberFormat(value, flags)); //Boolean +} + +} diff --git a/includes/js/dojox/validate/isbn.js b/includes/js/dojox/validate/isbn.js new file mode 100644 index 0000000..030c7cf --- /dev/null +++ b/includes/js/dojox/validate/isbn.js @@ -0,0 +1,37 @@ +if(!dojo._hasResource["dojox.validate.isbn"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.isbn"] = true; +dojo.provide("dojox.validate.isbn"); + +dojox.validate.isValidIsbn = function(/* String */value) { + // summary: Vadlidate ISBN-10 or ISBN-13 based on the length of value + // returns: Boolean + var len, sum, weight; + if(typeof value!='string'){ + value = String(value); + } + value = value.replace(/[- ]/g,''); //ignore dashes and whitespaces + len = value.length; + sum = 0; + if(len == 10){ + weight = 10; + // ISBN-10 validation algorithm + for(var i = 0; i< 9; i++){ + sum += parseInt(value.charAt(i)) * weight; + weight --; + } + var t = value.charAt(9).toUpperCase(); + sum += t == 'X' ? 10 : parseInt(t); + return sum % 11 == 0; + }else if(len == 13) { + weight = -1; + for(var i=0; i< len; i++){ + sum += parseInt(value.charAt(i)) * (2 + weight); + weight *= -1; + } + return sum % 10 == 0; + }else{ + return false; + } +} + +} diff --git a/includes/js/dojox/validate/regexp.js b/includes/js/dojox/validate/regexp.js new file mode 100644 index 0000000..0924750 --- /dev/null +++ b/includes/js/dojox/validate/regexp.js @@ -0,0 +1,331 @@ +if(!dojo._hasResource["dojox.validate.regexp"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.regexp"] = true; +dojo.provide("dojox.validate.regexp"); + +dojo.require("dojo.regexp"); + +// *** Regular Expression Generator does not entirely live here *** +// FIXME: is this useful enough to be in /dojox/regexp/_base.js, or +// should it respect namespace and be dojox.validate.regexp? +// some say a generic regexp to match zipcodes and urls would be useful +// others would say it's a spare tire. +dojox.regexp = { ca: {}, us: {} }; + +dojox.regexp.tld = function(/*Object?*/flags){ + // summary: Builds a RE that matches a top-level domain + // + // flags: + // flags.allowCC Include 2 letter country code domains. Default is true. + // flags.allowGeneric Include the generic domains. Default is true. + // flags.allowInfra Include infrastructure domains. Default is true. + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + if(typeof flags.allowCC != "boolean"){ flags.allowCC = true; } + if(typeof flags.allowInfra != "boolean"){ flags.allowInfra = true; } + if(typeof flags.allowGeneric != "boolean"){ flags.allowGeneric = true; } + + // Infrastructure top-level domain - only one at present + var infraRE = "arpa"; + + // Generic top-level domains RE. + var genericRE = + "aero|biz|com|coop|edu|gov|info|int|mil|museum|name|net|org|pro|travel|xxx|jobs|mobi|post"; + + // Country Code top-level domains RE + var ccRE = + "ac|ad|ae|af|ag|ai|al|am|an|ao|aq|ar|as|at|au|aw|az|ba|bb|bd|be|bf|bg|bh|bi|bj|bm|bn|bo|br|" + + "bs|bt|bv|bw|by|bz|ca|cc|cd|cf|cg|ch|ci|ck|cl|cm|cn|co|cr|cu|cv|cx|cy|cz|de|dj|dk|dm|do|dz|" + + "ec|ee|eg|er|eu|es|et|fi|fj|fk|fm|fo|fr|ga|gd|ge|gf|gg|gh|gi|gl|gm|gn|gp|gq|gr|gs|gt|gu|gw|" + + + "gy|hk|hm|hn|hr|ht|hu|id|ie|il|im|in|io|ir|is|it|je|jm|jo|jp|ke|kg|kh|ki|km|kn|kr|kw|ky|kz|" + + "la|lb|lc|li|lk|lr|ls|lt|lu|lv|ly|ma|mc|md|mg|mh|mk|ml|mm|mn|mo|mp|mq|mr|ms|mt|mu|mv|mw|mx|" + + "my|mz|na|nc|ne|nf|ng|ni|nl|no|np|nr|nu|nz|om|pa|pe|pf|pg|ph|pk|pl|pm|pn|pr|ps|pt|pw|py|qa|" + + "re|ro|ru|rw|sa|sb|sc|sd|se|sg|sh|si|sk|sl|sm|sn|sr|st|su|sv|sy|sz|tc|td|tf|tg|th|tj|tk|tm|" + + "tn|to|tr|tt|tv|tw|tz|ua|ug|uk|us|uy|uz|va|vc|ve|vg|vi|vn|vu|wf|ws|ye|yt|yu|za|zm|zw"; + + // Build top-level domain RE + var a = []; + if(flags.allowInfra){ a.push(infraRE); } + if(flags.allowGeneric){ a.push(genericRE); } + if(flags.allowCC){ a.push(ccRE); } + + var tldRE = ""; + if (a.length > 0) { + tldRE = "(" + a.join("|") + ")"; + } + + return tldRE; // String +} + +dojox.regexp.ipAddress = function(/*Object?*/flags){ + // summary: Builds a RE that matches an IP Address + // + // description: + // Supports 5 formats for IPv4: dotted decimal, dotted hex, dotted octal, decimal and hexadecimal. + // Supports 2 formats for Ipv6. + // + // flags An object. All flags are boolean with default = true. + // flags.allowDottedDecimal Example, 207.142.131.235. No zero padding. + // flags.allowDottedHex Example, 0x18.0x11.0x9b.0x28. Case insensitive. Zero padding allowed. + // flags.allowDottedOctal Example, 0030.0021.0233.0050. Zero padding allowed. + // flags.allowDecimal Example, 3482223595. A decimal number between 0-4294967295. + // flags.allowHex Example, 0xCF8E83EB. Hexadecimal number between 0x0-0xFFFFFFFF. + // Case insensitive. Zero padding allowed. + // flags.allowIPv6 IPv6 address written as eight groups of four hexadecimal digits. + // FIXME: ipv6 can be written multiple ways IIRC + // flags.allowHybrid IPv6 address written as six groups of four hexadecimal digits + // followed by the usual 4 dotted decimal digit notation of IPv4. x:x:x:x:x:x:d.d.d.d + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + if(typeof flags.allowDottedDecimal != "boolean"){ flags.allowDottedDecimal = true; } + if(typeof flags.allowDottedHex != "boolean"){ flags.allowDottedHex = true; } + if(typeof flags.allowDottedOctal != "boolean"){ flags.allowDottedOctal = true; } + if(typeof flags.allowDecimal != "boolean"){ flags.allowDecimal = true; } + if(typeof flags.allowHex != "boolean"){ flags.allowHex = true; } + if(typeof flags.allowIPv6 != "boolean"){ flags.allowIPv6 = true; } + if(typeof flags.allowHybrid != "boolean"){ flags.allowHybrid = true; } + + // decimal-dotted IP address RE. + var dottedDecimalRE = + // Each number is between 0-255. Zero padding is not allowed. + "((\\d|[1-9]\\d|1\\d\\d|2[0-4]\\d|25[0-5])\\.){3}(\\d|[1-9]\\d|1\\d\\d|2[0-4]\\d|25[0-5])"; + + // dotted hex IP address RE. Each number is between 0x0-0xff. Zero padding is allowed, e.g. 0x00. + var dottedHexRE = "(0[xX]0*[\\da-fA-F]?[\\da-fA-F]\\.){3}0[xX]0*[\\da-fA-F]?[\\da-fA-F]"; + + // dotted octal IP address RE. Each number is between 0000-0377. + // Zero padding is allowed, but each number must have at least 4 characters. + var dottedOctalRE = "(0+[0-3][0-7][0-7]\\.){3}0+[0-3][0-7][0-7]"; + + // decimal IP address RE. A decimal number between 0-4294967295. + var decimalRE = "(0|[1-9]\\d{0,8}|[1-3]\\d{9}|4[01]\\d{8}|42[0-8]\\d{7}|429[0-3]\\d{6}|" + + "4294[0-8]\\d{5}|42949[0-5]\\d{4}|429496[0-6]\\d{3}|4294967[01]\\d{2}|42949672[0-8]\\d|429496729[0-5])"; + + // hexadecimal IP address RE. + // A hexadecimal number between 0x0-0xFFFFFFFF. Case insensitive. Zero padding is allowed. + var hexRE = "0[xX]0*[\\da-fA-F]{1,8}"; + + // IPv6 address RE. + // The format is written as eight groups of four hexadecimal digits, x:x:x:x:x:x:x:x, + // where x is between 0000-ffff. Zero padding is optional. Case insensitive. + var ipv6RE = "([\\da-fA-F]{1,4}\\:){7}[\\da-fA-F]{1,4}"; + + // IPv6/IPv4 Hybrid address RE. + // The format is written as six groups of four hexadecimal digits, + // followed by the 4 dotted decimal IPv4 format. x:x:x:x:x:x:d.d.d.d + var hybridRE = "([\\da-fA-F]{1,4}\\:){6}" + + "((\\d|[1-9]\\d|1\\d\\d|2[0-4]\\d|25[0-5])\\.){3}(\\d|[1-9]\\d|1\\d\\d|2[0-4]\\d|25[0-5])"; + + // Build IP Address RE + var a = []; + if(flags.allowDottedDecimal){ a.push(dottedDecimalRE); } + if(flags.allowDottedHex){ a.push(dottedHexRE); } + if(flags.allowDottedOctal){ a.push(dottedOctalRE); } + if(flags.allowDecimal){ a.push(decimalRE); } + if(flags.allowHex){ a.push(hexRE); } + if(flags.allowIPv6){ a.push(ipv6RE); } + if(flags.allowHybrid){ a.push(hybridRE); } + + var ipAddressRE = ""; + if(a.length > 0){ + ipAddressRE = "(" + a.join("|") + ")"; + } + return ipAddressRE; // String +} + +dojox.regexp.host = function(/*Object?*/flags){ + // summary: Builds a RE that matches a host + // description: A host is a named host (A-z0-9_- but not starting with -), a domain name or an IP address, possibly followed by a port number. + // flags: An object. + // flags.allowNamed Allow a named host for local networks. Default is false. + // flags.allowIP Allow an IP address for hostname. Default is true. + // flags.allowLocal Allow the host to be "localhost". Default is false. + // flags.allowPort Allow a port number to be present. Default is true. + // flags in regexp.ipAddress can be applied. + // flags in regexp.tld can be applied. + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + if(typeof flags.allowIP != "boolean"){ flags.allowIP = true; } + if(typeof flags.allowLocal != "boolean"){ flags.allowLocal = false; } + if(typeof flags.allowPort != "boolean"){ flags.allowPort = true; } + if(typeof flags.allowNamed != "boolean"){ flags.allowNamed = false; } + + // Domain names can not end with a dash. + var domainNameRE = "([0-9a-zA-Z]([-0-9a-zA-Z]{0,61}[0-9a-zA-Z])?\\.)+" + dojox.regexp.tld(flags); + + // port number RE + var portRE = flags.allowPort ? "(\\:\\d+)?" : ""; + + // build host RE + var hostNameRE = domainNameRE; + if(flags.allowIP){ hostNameRE += "|" + dojox.regexp.ipAddress(flags); } + if(flags.allowLocal){ hostNameRE += "|localhost"; } + if(flags.allowNamed){ hostNameRE += "|^[^-][a-zA-Z0-9_-]*"; } + return "(" + hostNameRE + ")" + portRE; // String +} + +dojox.regexp.url = function(/*Object?*/flags){ + // summary: Builds a regular expression that matches a URL + // + // flags: An object + // flags.scheme Can be true, false, or [true, false]. + // This means: required, not allowed, or match either one. + // flags in regexp.host can be applied. + // flags in regexp.ipAddress can be applied. + // flags in regexp.tld can be applied. + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + if(!("scheme" in flags)){ flags.scheme = [true, false]; } + + // Scheme RE + var protocolRE = dojo.regexp.buildGroupRE(flags.scheme, + function(q){ if(q){ return "(https?|ftps?)\\://"; } return ""; } + ); + + // Path and query and anchor RE + var pathRE = "(/([^?#\\s/]+/)*)?([^?#\\s/]+(\\?[^?#\\s/]*)?(#[A-Za-z][\\w.:-]*)?)?"; + + return protocolRE + dojox.regexp.host(flags) + pathRE; +} + +dojox.regexp.emailAddress = function(/*Object?*/flags){ + + // summary: Builds a regular expression that matches an email address + // + //flags: An object + // flags.allowCruft Allow address like <mailto:foo@yahoo.com>. Default is false. + // flags in regexp.host can be applied. + // flags in regexp.ipAddress can be applied. + // flags in regexp.tld can be applied. + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + if (typeof flags.allowCruft != "boolean") { flags.allowCruft = false; } + flags.allowPort = false; // invalid in email addresses + + // user name RE - apostrophes are valid if there's not 2 in a row + var usernameRE = "([\\da-zA-Z]+[-._+&'])*[\\da-zA-Z]+"; + + // build emailAddress RE + var emailAddressRE = usernameRE + "@" + dojox.regexp.host(flags); + + // Allow email addresses with cruft + if ( flags.allowCruft ) { + emailAddressRE = "<?(mailto\\:)?" + emailAddressRE + ">?"; + } + + return emailAddressRE; // String +} + +dojox.regexp.emailAddressList = function(/*Object?*/flags){ + // summary: Builds a regular expression that matches a list of email addresses. + // + // flags: An object. + // flags.listSeparator The character used to separate email addresses. Default is ";", ",", "\n" or " ". + // flags in regexp.emailAddress can be applied. + // flags in regexp.host can be applied. + // flags in regexp.ipAddress can be applied. + // flags in regexp.tld can be applied. + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + if(typeof flags.listSeparator != "string"){ flags.listSeparator = "\\s;,"; } + + // build a RE for an Email Address List + var emailAddressRE = dojox.regexp.emailAddress(flags); + var emailAddressListRE = "(" + emailAddressRE + "\\s*[" + flags.listSeparator + "]\\s*)*" + + emailAddressRE + "\\s*[" + flags.listSeparator + "]?\\s*"; + + return emailAddressListRE; // String +} + +dojox.regexp.us.state = function(/*Object?*/flags){ + // summary: A regular expression to match US state and territory abbreviations + // + // flags An object. + // flags.allowTerritories Allow Guam, Puerto Rico, etc. Default is true. + // flags.allowMilitary Allow military 'states', e.g. Armed Forces Europe (AE). Default is true. + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + if(typeof flags.allowTerritories != "boolean"){ flags.allowTerritories = true; } + if(typeof flags.allowMilitary != "boolean"){ flags.allowMilitary = true; } + + // state RE + var statesRE = + "AL|AK|AZ|AR|CA|CO|CT|DE|DC|FL|GA|HI|ID|IL|IN|IA|KS|KY|LA|ME|MD|MA|MI|MN|MS|MO|MT|" + + "NE|NV|NH|NJ|NM|NY|NC|ND|OH|OK|OR|PA|RI|SC|SD|TN|TX|UT|VT|VA|WA|WV|WI|WY"; + + // territories RE + var territoriesRE = "AS|FM|GU|MH|MP|PW|PR|VI"; + + // military states RE + var militaryRE = "AA|AE|AP"; + + // Build states and territories RE + if(flags.allowTerritories){ statesRE += "|" + territoriesRE; } + if(flags.allowMilitary){ statesRE += "|" + militaryRE; } + + return "(" + statesRE + ")"; // String +} + +dojox.regexp.ca.postalCode = function(){ + var postalRE = + "[A-Z][0-9][A-Z] [0-9][A-Z][0-9]"; + return "(" + postalRE + ")"; +} + +dojox.regexp.ca.province = function(){ + // summary: a regular expression to match Canadian Province Abbreviations + var stateRE = + "AB|BC|MB|NB|NL|NS|NT|NU|ON|PE|QC|SK|YT"; + return "(" + statesRE + ")"; +} + +dojox.regexp.numberFormat = function(/*Object?*/flags){ + // summary: Builds a regular expression to match any sort of number based format + // description: + // Use this method for phone numbers, social security numbers, zip-codes, etc. + // The RE can match one format or one of multiple formats. + // + // Format + // # Stands for a digit, 0-9. + // ? Stands for an optional digit, 0-9 or nothing. + // All other characters must appear literally in the expression. + // + // Example + // "(###) ###-####" -> (510) 542-9742 + // "(###) ###-#### x#???" -> (510) 542-9742 x153 + // "###-##-####" -> 506-82-1089 i.e. social security number + // "#####-####" -> 98225-1649 i.e. zip code + // + // flags: An object + // flags.format A string or an Array of strings for multiple formats. + + // assign default values to missing paramters + flags = (typeof flags == "object") ? flags : {}; + if(typeof flags.format == "undefined"){ flags.format = "###-###-####"; } + + // Converts a number format to RE. + var digitRE = function(format){ + // escape all special characters, except '?' + format = dojo.regexp.escapeString(format, "?"); + + // Now replace '?' with Regular Expression + format = format.replace(/\?/g, "\\d?"); + + // replace # with Regular Expression + format = format.replace(/#/g, "\\d"); + + return format; // String + }; + + // build RE for multiple number formats + return dojo.regexp.buildGroupRE(flags.format, digitRE); //String +} + +} diff --git a/includes/js/dojox/validate/tests/creditcard.js b/includes/js/dojox/validate/tests/creditcard.js new file mode 100644 index 0000000..bacb01e --- /dev/null +++ b/includes/js/dojox/validate/tests/creditcard.js @@ -0,0 +1,119 @@ +if(!dojo._hasResource["dojox.validate.tests.creditcard"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.tests.creditcard"] = true; +dojo.provide("dojox.validate.tests.creditcard"); +dojo.require("dojox.validate.creditCard"); + +tests.register("dojox.validate.tests.creditcard", + [{ + name:"isValidLuhn", + runTests: function(tests) { + tests.t(dojox.validate.isValidLuhn('5105105105105100')); //test string input + tests.t(dojox.validate.isValidLuhn('5105-1051 0510-5100')); //test string input with dashes and spaces (commonly used when entering card #'s) + tests.t(dojox.validate.isValidLuhn(38520000023237)); //test numerical input as well + tests.f(dojox.validate.isValidLuhn(3852000002323)); //testing failures + tests.t(dojox.validate.isValidLuhn(18)); //length doesnt matter + tests.f(dojox.validate.isValidLuhn(818181)); //short length failure + } + }, + { + name:"isValidCvv", + runTests: function(tests) { + tests.t(dojox.validate.isValidCvv('123','mc')); //string is ok + tests.f(dojox.validate.isValidCvv('5AA','ec')); //invalid characters are not ok + tests.t(dojox.validate.isValidCvv(723,'mc')); //numbers are ok too + tests.f(dojox.validate.isValidCvv(7234,'mc')); //too long + tests.t(dojox.validate.isValidCvv(612,'ec')); + tests.t(dojox.validate.isValidCvv(421,'vi')); + tests.t(dojox.validate.isValidCvv(543,'di')); + tests.t(dojox.validate.isValidCvv('1234','ax')); + tests.t(dojox.validate.isValidCvv(4321,'ax')); + tests.f(dojox.validate.isValidCvv(43215,'ax')); //too long + tests.f(dojox.validate.isValidCvv(215,'ax')); //too short + } + }, + { + name:"isValidCreditCard", + runTests: function(tests) { + //misc checks + tests.t(dojox.validate.isValidCreditCard('5105105105105100','mc')); //test string input + tests.t(dojox.validate.isValidCreditCard('5105-1051 0510-5100','mc')); //test string input with dashes and spaces (commonly used when entering card #'s) + tests.t(dojox.validate.isValidCreditCard(5105105105105100,'mc')); //test numerical input as well + tests.f(dojox.validate.isValidCreditCard('5105105105105100','vi')); //fails, wrong card type + //Mastercard/Eurocard checks + tests.t(dojox.validate.isValidCreditCard('5105105105105100','mc')); + tests.t(dojox.validate.isValidCreditCard('5204105105105100','ec')); + tests.t(dojox.validate.isValidCreditCard('5303105105105100','mc')); + tests.t(dojox.validate.isValidCreditCard('5402105105105100','ec')); + tests.t(dojox.validate.isValidCreditCard('5501105105105100','mc')); + //Visa card checks + tests.t(dojox.validate.isValidCreditCard('4111111111111111','vi')); + tests.t(dojox.validate.isValidCreditCard('4111111111010','vi')); + //American Express card checks + tests.t(dojox.validate.isValidCreditCard('378 2822 4631 0005','ax')); + tests.t(dojox.validate.isValidCreditCard('341-1111-1111-1111','ax')); + //Diners Club/Carte Blanch card checks + tests.t(dojox.validate.isValidCreditCard('36400000000000','dc')); + tests.t(dojox.validate.isValidCreditCard('38520000023237','bl')); + tests.t(dojox.validate.isValidCreditCard('30009009025904','dc')); + tests.t(dojox.validate.isValidCreditCard('30108009025904','bl')); + tests.t(dojox.validate.isValidCreditCard('30207009025904','dc')); + tests.t(dojox.validate.isValidCreditCard('30306009025904','bl')); + tests.t(dojox.validate.isValidCreditCard('30405009025904','dc')); + tests.t(dojox.validate.isValidCreditCard('30504009025904','bl')); + //Discover card checks + tests.t(dojox.validate.isValidCreditCard('6011111111111117','di')); + //JCB card checks + tests.t(dojox.validate.isValidCreditCard('3530111333300000','jcb')); + tests.t(dojox.validate.isValidCreditCard('213100000000001','jcb')); + tests.t(dojox.validate.isValidCreditCard('180000000000002','jcb')); + tests.f(dojox.validate.isValidCreditCard('1800000000000002','jcb')); //should fail, good checksum, good prefix, but wrong length' + //Enroute card checks + tests.t(dojox.validate.isValidCreditCard('201400000000000','er')); + tests.t(dojox.validate.isValidCreditCard('214900000000000','er')); + } + }, + { + name:"isValidCreditCardNumber", + runTests: function(tests) { + //misc checks + tests.t(dojox.validate.isValidCreditCardNumber('5105105105105100','mc')); //test string input + tests.t(dojox.validate.isValidCreditCardNumber('5105-1051 0510-5100','mc')); //test string input with dashes and spaces (commonly used when entering card #'s) + tests.t(dojox.validate.isValidCreditCardNumber(5105105105105100,'mc')); //test numerical input as well + tests.f(dojox.validate.isValidCreditCardNumber('5105105105105100','vi')); //fails, wrong card type + //Mastercard/Eurocard checks + tests.is("mc|ec", dojox.validate.isValidCreditCardNumber('5100000000000000')); //should match 'mc|ec' + tests.is("mc|ec", dojox.validate.isValidCreditCardNumber('5200000000000000')); //should match 'mc|ec' + tests.is("mc|ec", dojox.validate.isValidCreditCardNumber('5300000000000000')); //should match 'mc|ec' + tests.is("mc|ec", dojox.validate.isValidCreditCardNumber('5400000000000000')); //should match 'mc|ec' + tests.is("mc|ec", dojox.validate.isValidCreditCardNumber('5500000000000000')); //should match 'mc|ec' + tests.f(dojox.validate.isValidCreditCardNumber('55000000000000000')); //should fail, too long + //Visa card checks + tests.is("vi", dojox.validate.isValidCreditCardNumber('4111111111111111')); //should match 'vi' + tests.is("vi", dojox.validate.isValidCreditCardNumber('4111111111010')); //should match 'vi' + //American Express card checks + tests.is("ax", dojox.validate.isValidCreditCardNumber('378 2822 4631 0005')); //should match 'ax' + tests.is("ax", dojox.validate.isValidCreditCardNumber('341-1111-1111-1111')); //should match 'ax' + //Diners Club/Carte Blanch card checks + tests.is("dc|bl", dojox.validate.isValidCreditCardNumber('36400000000000')); //should match 'dc|bl' + tests.is("dc|bl", dojox.validate.isValidCreditCardNumber('38520000023237')); //should match 'dc|bl' + tests.is("dc|bl", dojox.validate.isValidCreditCardNumber('30009009025904')); //should match 'di|bl' + tests.is("dc|bl", dojox.validate.isValidCreditCardNumber('30108009025904')); //should match 'di|bl' + tests.is("dc|bl", dojox.validate.isValidCreditCardNumber('30207009025904')); //should match 'di|bl' + tests.is("dc|bl", dojox.validate.isValidCreditCardNumber('30306009025904')); //should match 'di|bl' + tests.is("dc|bl", dojox.validate.isValidCreditCardNumber('30405009025904')); //should match 'di|bl' + tests.is("dc|bl", dojox.validate.isValidCreditCardNumber('30504009025904')); //should match 'di|bl' + //Discover card checks + tests.is("di", dojox.validate.isValidCreditCardNumber('6011111111111117')); //should match 'di' + //JCB card checks + tests.is("jcb", dojox.validate.isValidCreditCardNumber('3530111333300000')); //should match 'jcb' + tests.is("jcb", dojox.validate.isValidCreditCardNumber('213100000000001')); //should match 'jcb' + tests.is("jcb", dojox.validate.isValidCreditCardNumber('180000000000002')); //should match 'jcb' + tests.f(dojox.validate.isValidCreditCardNumber('1800000000000002')); //should fail, good checksum, good prefix, but wrong length' + //Enroute card checks + tests.is("er", dojox.validate.isValidCreditCardNumber('201400000000000')); //should match 'er' + tests.is("er", dojox.validate.isValidCreditCardNumber('214900000000000')); //should match 'er' + } + } +]); + +} diff --git a/includes/js/dojox/validate/tests/module.js b/includes/js/dojox/validate/tests/module.js new file mode 100644 index 0000000..c04fe4f --- /dev/null +++ b/includes/js/dojox/validate/tests/module.js @@ -0,0 +1,14 @@ +if(!dojo._hasResource["dojox.validate.tests.module"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.tests.module"] = true; +dojo.provide("dojox.validate.tests.module"); + +try{ + dojo.require("dojox.validate.tests.creditcard"); + dojo.require("dojox.validate.tests.validate"); + +}catch(e){ + doh.debug(e); + console.debug(e); +} + +} diff --git a/includes/js/dojox/validate/tests/runTests.html b/includes/js/dojox/validate/tests/runTests.html new file mode 100644 index 0000000..390674c --- /dev/null +++ b/includes/js/dojox/validate/tests/runTests.html @@ -0,0 +1,9 @@ +<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.0 Transitional//EN"> +<html> + <head> + <title>Dijit Unit Test Runner</title> + <meta http-equiv="REFRESH" content="0;url=../../../util/doh/runner.html?testModule=dojox.validate.tests.module"></HEAD> + <BODY> + Redirecting to D.O.H runner. + </BODY> +</HTML> diff --git a/includes/js/dojox/validate/tests/validate.js b/includes/js/dojox/validate/tests/validate.js new file mode 100644 index 0000000..2fb272f --- /dev/null +++ b/includes/js/dojox/validate/tests/validate.js @@ -0,0 +1,316 @@ +if(!dojo._hasResource["dojox.validate.tests.validate"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.tests.validate"] = true; +dojo.provide("dojox.validate.tests.validate"); + +dojo.require("dojox.validate._base"); +dojo.require("dojox.validate.check"); +dojo.require("dojox.validate.us"); +dojo.require("dojox.validate.ca"); +dojo.require("dojox.validate.web"); +dojo.require("dojox.validate.isbn"); + +tests.register("dojox.validate.tests.validate", + [{ + name: "isText", + runTest: function(tests){ + tests.t(dojox.validate.isValidIsbn('0596007590')); //test string input + tests.t(dojox.validate.isValidIsbn('0-596-00759-0')); //test string input with dashes + tests.f(dojox.validate.isValidIsbn(596007590)); //test numerical input as well + tests.t(dojox.validate.isValidIsbn("960-425-059-0")); + tests.t(dojox.validate.isValidIsbn(9604250590)); //test numerical input as well + tests.t(dojox.validate.isValidIsbn('0-9752298-0-X')); // test string with X + tests.t(dojox.validate.isValidIsbn('0-9752298-0-x')); + tests.t(dojox.validate.isValidIsbn('097522980x')); + tests.t(dojox.validate.isValidIsbn('097522980X')); + tests.f(dojox.validate.isValidIsbn(596007598)); //testing failures + tests.f(dojox.validate.isValidIsbn('059-600759-X')); //testing failures + tests.f(dojox.validate.isValidIsbn('059600')); // too short + + tests.t(dojox.validate.isValidIsbn('9780596007591')); + tests.t(dojox.validate.isValidIsbn('978-0-596 00759-1')); + tests.t(dojox.validate.isValidIsbn(9780596007591)); + tests.f(dojox.validate.isValidIsbn('978059600759X')); + tests.f(dojox.validate.isValidIsbn('978-3250-596 00759-1 ')); + tests.f(dojox.validate.isValidIsbn('3250-596 00759 ')); + + tests.t(dojox.validate.isText(' x')); + tests.t(dojox.validate.isText('x ')); + tests.t(dojox.validate.isText(' x ')); + tests.f(dojox.validate.isText(' ')); + tests.f(dojox.validate.isText('')); + + // test lengths + tests.t(dojox.validate.isText('123456', {length: 6} )); + tests.f(dojox.validate.isText('1234567', {length: 6} )); + tests.t(dojox.validate.isText('1234567', {minlength: 6} )); + tests.t(dojox.validate.isText('123456', {minlength: 6} )); + tests.f(dojox.validate.isText('12345', {minlength: 6} )); + tests.f(dojox.validate.isText('1234567', {maxlength: 6} )); + tests.t(dojox.validate.isText('123456', {maxlength: 6} )); + } + }, + { + name: "isIpAddress", + runTest: function(tests){ + tests.t(dojox.validate.isIpAddress('24.17.155.40')); + tests.f(dojox.validate.isIpAddress('024.17.155.040')); + tests.t(dojox.validate.isIpAddress('255.255.255.255')); + tests.f(dojox.validate.isIpAddress('256.255.255.255')); + tests.f(dojox.validate.isIpAddress('255.256.255.255')); + tests.f(dojox.validate.isIpAddress('255.255.256.255')); + tests.f(dojox.validate.isIpAddress('255.255.255.256')); + + // test dotted hex + tests.t(dojox.validate.isIpAddress('0x18.0x11.0x9b.0x28')); + tests.f(dojox.validate.isIpAddress('0x18.0x11.0x9b.0x28', {allowDottedHex: false}) ); + tests.t(dojox.validate.isIpAddress('0x18.0x000000011.0x9b.0x28')); + tests.t(dojox.validate.isIpAddress('0xff.0xff.0xff.0xff')); + tests.f(dojox.validate.isIpAddress('0x100.0xff.0xff.0xff')); + + // test dotted octal + tests.t(dojox.validate.isIpAddress('0030.0021.0233.0050')); + tests.f(dojox.validate.isIpAddress('0030.0021.0233.0050', {allowDottedOctal: false}) ); + tests.t(dojox.validate.isIpAddress('0030.0000021.0233.00000050')); + tests.t(dojox.validate.isIpAddress('0377.0377.0377.0377')); + tests.f(dojox.validate.isIpAddress('0400.0377.0377.0377')); + tests.f(dojox.validate.isIpAddress('0377.0378.0377.0377')); + tests.f(dojox.validate.isIpAddress('0377.0377.0380.0377')); + tests.f(dojox.validate.isIpAddress('0377.0377.0377.377')); + + // test decimal + tests.t(dojox.validate.isIpAddress('3482223595')); + tests.t(dojox.validate.isIpAddress('0')); + tests.t(dojox.validate.isIpAddress('4294967295')); + tests.f(dojox.validate.isIpAddress('4294967296')); + tests.f(dojox.validate.isIpAddress('3482223595', {allowDecimal: false})); + + // test hex + tests.t(dojox.validate.isIpAddress('0xCF8E83EB')); + tests.t(dojox.validate.isIpAddress('0x0')); + tests.t(dojox.validate.isIpAddress('0x00ffffffff')); + tests.f(dojox.validate.isIpAddress('0x100000000')); + tests.f(dojox.validate.isIpAddress('0xCF8E83EB', {allowHex: false})); + + // IPv6 + tests.t(dojox.validate.isIpAddress('fedc:BA98:7654:3210:FEDC:BA98:7654:3210')); + tests.t(dojox.validate.isIpAddress('1080:0:0:0:8:800:200C:417A')); + tests.f(dojox.validate.isIpAddress('1080:0:0:0:8:800:200C:417A', {allowIPv6: false})); + + // Hybrid of IPv6 and IPv4 + tests.t(dojox.validate.isIpAddress('0:0:0:0:0:0:13.1.68.3')); + tests.t(dojox.validate.isIpAddress('0:0:0:0:0:FFFF:129.144.52.38')); + tests.f(dojox.validate.isIpAddress('0:0:0:0:0:FFFF:129.144.52.38', {allowHybrid: false})); + + } + }, + { + name: "isUrlTest", + runTest: function(tests){ + + tests.t(dojox.validate.isUrl('www.yahoo.com')); + tests.t(dojox.validate.isUrl('http://www.yahoo.com')); + tests.t(dojox.validate.isUrl('https://www.yahoo.com')); + tests.f(dojox.validate.isUrl('http://.yahoo.com')); + tests.f(dojox.validate.isUrl('http://www.-yahoo.com')); + tests.f(dojox.validate.isUrl('http://www.yahoo-.com')); + tests.t(dojox.validate.isUrl('http://y-a---h-o-o.com')); + tests.t(dojox.validate.isUrl('http://www.y.com')); + tests.t(dojox.validate.isUrl('http://www.yahoo.museum')); + tests.t(dojox.validate.isUrl('http://www.yahoo.co.uk')); + tests.f(dojox.validate.isUrl('http://www.micro$oft.com')); + + tests.t(dojox.validate.isUrl('http://www.y.museum:8080')); + tests.t(dojox.validate.isUrl('http://12.24.36.128:8080')); + tests.f(dojox.validate.isUrl('http://12.24.36.128:8080', {allowIP: false} )); + tests.t(dojox.validate.isUrl('www.y.museum:8080')); + tests.f(dojox.validate.isUrl('www.y.museum:8080', {scheme: true} )); + tests.t(dojox.validate.isUrl('localhost:8080', {allowLocal: true} )); + tests.f(dojox.validate.isUrl('localhost:8080', {} )); + tests.t(dojox.validate.isUrl('http://www.yahoo.com/index.html?a=12&b=hello%20world#anchor')); + tests.f(dojox.validate.isUrl('http://www.yahoo.xyz')); + tests.t(dojox.validate.isUrl('http://www.yahoo.com/index.html#anchor')); + tests.t(dojox.validate.isUrl('http://cocoon.apache.org/2.1/')); + } + }, + { + name: "isEmailAddress", + runTest: function(tests) { + tests.t(dojox.validate.isEmailAddress('x@yahoo.com')); + tests.t(dojox.validate.isEmailAddress('x.y.z.w@yahoo.com')); + tests.f(dojox.validate.isEmailAddress('x..y.z.w@yahoo.com')); + tests.f(dojox.validate.isEmailAddress('x.@yahoo.com')); + tests.t(dojox.validate.isEmailAddress('x@z.com')); + tests.f(dojox.validate.isEmailAddress('x@yahoo.x')); + tests.t(dojox.validate.isEmailAddress('x@yahoo.museum')); + tests.t(dojox.validate.isEmailAddress("o'mally@yahoo.com")); + tests.f(dojox.validate.isEmailAddress("'mally@yahoo.com")); + tests.t(dojox.validate.isEmailAddress("fred&barney@stonehenge.com")); + tests.f(dojox.validate.isEmailAddress("fred&&barney@stonehenge.com")); + + // local addresses + tests.t(dojox.validate.isEmailAddress("fred&barney@localhost", {allowLocal: true} )); + tests.f(dojox.validate.isEmailAddress("fred&barney@localhost")); + + // addresses with cruft + tests.t(dojox.validate.isEmailAddress("mailto:fred&barney@stonehenge.com", {allowCruft: true} )); + tests.t(dojox.validate.isEmailAddress("<fred&barney@stonehenge.com>", {allowCruft: true} )); + tests.f(dojox.validate.isEmailAddress("mailto:fred&barney@stonehenge.com")); + tests.f(dojox.validate.isEmailAddress("<fred&barney@stonehenge.com>")); + + // local addresses with cruft + tests.t(dojox.validate.isEmailAddress("<mailto:fred&barney@localhost>", {allowLocal: true, allowCruft: true} )); + tests.f(dojox.validate.isEmailAddress("<mailto:fred&barney@localhost>", {allowCruft: true} )); + tests.f(dojox.validate.isEmailAddress("<mailto:fred&barney@localhost>", {allowLocal: true} )); + } + }, + { + name: "isEmailsAddressList", + runTest: function(tests) { + tests.t(dojox.validate.isEmailAddressList( + "x@yahoo.com \n x.y.z.w@yahoo.com ; o'mally@yahoo.com , fred&barney@stonehenge.com \n" ) + ); + tests.t(dojox.validate.isEmailAddressList( + "x@yahoo.com \n x.y.z.w@localhost \n o'mally@yahoo.com \n fred&barney@localhost", + {allowLocal: true} ) + ); + tests.f(dojox.validate.isEmailAddressList( + "x@yahoo.com; x.y.z.w@localhost; o'mally@yahoo.com; fred&barney@localhost", {listSeparator: ";"} ) + ); + tests.t(dojox.validate.isEmailAddressList( + "mailto:x@yahoo.com; <x.y.z.w@yahoo.com>; <mailto:o'mally@yahoo.com>; fred&barney@stonehenge.com", + {allowCruft: true, listSeparator: ";"} ) + ); + tests.f(dojox.validate.isEmailAddressList( + "mailto:x@yahoo.com; <x.y.z.w@yahoo.com>; <mailto:o'mally@yahoo.com>; fred&barney@stonehenge.com", + {listSeparator: ";"} ) + ); + tests.t(dojox.validate.isEmailAddressList( + "mailto:x@yahoo.com; <x.y.z.w@localhost>; <mailto:o'mally@localhost>; fred&barney@localhost", + {allowLocal: true, allowCruft: true, listSeparator: ";"} ) + ); + } + }, + { + name: "getEmailAddressList", + runTest: function(tests) { + var list = "x@yahoo.com \n x.y.z.w@yahoo.com ; o'mally@yahoo.com , fred&barney@stonehenge.com"; + tests.is(4, dojox.validate.getEmailAddressList(list).length); + + var localhostList = "x@yahoo.com; x.y.z.w@localhost; o'mally@yahoo.com; fred&barney@localhost"; + tests.is(0, dojox.validate.getEmailAddressList(localhostList).length); + tests.is(4, dojox.validate.getEmailAddressList(localhostList, {allowLocal: true} ).length); + } + }, + { + name: "isInRangeInt", + runTest: function(tests) { + // test integers + tests.f(dojox.validate.isInRange( '0', {min: 1, max: 100} )); + tests.t(dojox.validate.isInRange( '1', {min: 1, max: 100} )); + tests.f(dojox.validate.isInRange( '-50', {min: 1, max: 100} )); + tests.t(dojox.validate.isInRange( '+50', {min: 1, max: 100} )); + tests.t(dojox.validate.isInRange( '100', {min: 1, max: 100} )); + tests.f(dojox.validate.isInRange( '101', {min: 1, max: 100} )); + } + }, + { + name:"isInRangeReal", + runTest: function(tests){ + + tests.f(dojox.validate.isInRange( '0.9', {min: 1.0, max: 10.0} )); + tests.t(dojox.validate.isInRange( '1.0', {min: 1.0, max: 10.0} )); + tests.f(dojox.validate.isInRange( '-5.0', {min: 1.0, max: 10.0} )); + tests.t(dojox.validate.isInRange( '+5.50', {min: 1.0, max: 10.0} )); + tests.t(dojox.validate.isInRange( '10.0', {min: 1.0, max: 10.0} )); + tests.f(dojox.validate.isInRange( '10.1', {min: 1.0, max: 10.0} )); + tests.f(dojox.validate.isInRange( '5.566e28', {min: 5.567e28, max: 6.000e28} )); + tests.t(dojox.validate.isInRange( '5.7e28', {min: 5.567e28, max: 6.000e28} )); + tests.f(dojox.validate.isInRange( '6.00000001e28', {min: 5.567e28, max: 6.000e28} )); + tests.f(dojox.validate.isInRange( '10.000.000,12345e-5', {decimal: ",", max: 10000000.1e-5} )); + tests.f(dojox.validate.isInRange( '10.000.000,12345e-5', {decimal: ",", min: 10000000.2e-5} )); + tests.t(dojox.validate.isInRange('1,500,000', {separator: ',', min: 0})); + tests.f(dojox.validate.isInRange('1,500,000', {separator: ',', min: 1000, max: 20000})); + } + }, + { + + name:"isInRangeCurrency", + runTest: function(test){ + + tests.f(dojox.validate.isInRange('\u20AC123,456,789', {max: 123456788, symbol: '\u20AC'} )); + tests.f(dojox.validate.isInRange('\u20AC123,456,789', { min: 123456790, symbol: '\u20AC'} )); + tests.f(dojox.validate.isInRange('$123,456,789.07', { max: 123456789.06} )); + tests.f(dojox.validate.isInRange('$123,456,789.07', { min: 123456789.08} )); + tests.f(dojox.validate.isInRange('123.456.789,00 \u20AC', {max: 123456788, decimal: ",", symbol: '\u20AC'} )); + tests.f(dojox.validate.isInRange('123.456.789,00 \u20AC', {min: 123456790, decimal: ",", symbol: '\u20AC'} )); + tests.f(dojox.validate.isInRange('- T123 456 789-00', {decimal: "-", min:0} )); + tests.t(dojox.validate.isInRange('\u20AC123,456,789', { max: 123456790, symbol: '\u20AC'} )); + tests.t(dojox.validate.isInRange('$123,456,789.07', { min: 123456789.06} )); + // test non number + //tests.f("test25", dojox.validate.isInRange( 'a')); + } + }, + { + name: "isUsPhoneNumber", + runTest: function(tests) { + tests.t(dojox.validate.us.isPhoneNumber('(111) 111-1111')); + tests.t(dojox.validate.us.isPhoneNumber('(111) 111 1111')); + tests.t(dojox.validate.us.isPhoneNumber('111 111 1111')); + tests.t(dojox.validate.us.isPhoneNumber('111.111.1111')); + tests.t(dojox.validate.us.isPhoneNumber('111-111-1111')); + tests.t(dojox.validate.us.isPhoneNumber('111/111-1111')); + tests.f(dojox.validate.us.isPhoneNumber('111 111-1111')); + tests.f(dojox.validate.us.isPhoneNumber('111-1111')); + tests.f(dojox.validate.us.isPhoneNumber('(111)-111-1111')); + + // test extensions + tests.t(dojox.validate.us.isPhoneNumber('111-111-1111 x1')); + tests.t(dojox.validate.us.isPhoneNumber('111-111-1111 x12')); + tests.t(dojox.validate.us.isPhoneNumber('111-111-1111 x1234')); + } + }, + { + name: "isUsSocialSecurityNumber", + runTest: function(tests) { + tests.t(dojox.validate.us.isSocialSecurityNumber('123-45-6789')); + tests.t(dojox.validate.us.isSocialSecurityNumber('123 45 6789')); + tests.t(dojox.validate.us.isSocialSecurityNumber('123456789')); + tests.f(dojox.validate.us.isSocialSecurityNumber('123-45 6789')); + tests.f(dojox.validate.us.isSocialSecurityNumber('12345 6789')); + tests.f(dojox.validate.us.isSocialSecurityNumber('123-456789')); + } + }, + { + name:"isUsZipCode", + runTest: function(tests) { + tests.t(dojox.validate.us.isZipCode('12345-6789')); + tests.t(dojox.validate.us.isZipCode('12345 6789')); + tests.t(dojox.validate.us.isZipCode('123456789')); + tests.t(dojox.validate.us.isZipCode('12345')); + } + }, + { + name:"isCaZipCode", + runTest: function(tests) { + tests.t(dojox.validate.ca.isPostalCode('A1Z 3F3')); + tests.f(dojox.validate.ca.isPostalCode('1AZ 3F3')); + tests.t(dojox.validate.ca.isPostalCode('a1z 3f3')); + tests.f(dojox.validate.ca.isPostalCode('xxxxxx')); + tests.t(dojox.validate.ca.isPostalCode('A1Z3F3')); + + } + }, + { + name:"isUsState", + runTest: function(tests) { + tests.t(dojox.validate.us.isState('CA')); + tests.t(dojox.validate.us.isState('ne')); + tests.t(dojox.validate.us.isState('PR')); + tests.f(dojox.validate.us.isState('PR', {allowTerritories: false} )); + tests.t(dojox.validate.us.isState('AA')); + tests.f(dojox.validate.us.isState('AA', {allowMilitary: false} )); + } + } +]); + +} diff --git a/includes/js/dojox/validate/us.js b/includes/js/dojox/validate/us.js new file mode 100644 index 0000000..531b96d --- /dev/null +++ b/includes/js/dojox/validate/us.js @@ -0,0 +1,67 @@ +if(!dojo._hasResource["dojox.validate.us"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.us"] = true; +dojo.provide("dojox.validate.us"); +dojo.require("dojox.validate._base"); + +dojox.validate.us.isState = function(/*String*/value, /*Object?*/flags){ + // summary: Validates US state and territory abbreviations. + // + // value: A two character string + // flags: An object + // flags.allowTerritories Allow Guam, Puerto Rico, etc. Default is true. + // flags.allowMilitary Allow military 'states', e.g. Armed Forces Europe (AE). Default is true. + + var re = new RegExp("^" + dojox.regexp.us.state(flags) + "$", "i"); + return re.test(value); // Boolean +} + +dojox.validate.us.isPhoneNumber = function(/*String*/value){ + // summary: Validates 10 US digit phone number for several common formats + // value: The telephone number string + + var flags = { + format: [ + "###-###-####", + "(###) ###-####", + "(###) ### ####", + "###.###.####", + "###/###-####", + "### ### ####", + "###-###-#### x#???", + "(###) ###-#### x#???", + "(###) ### #### x#???", + "###.###.#### x#???", + "###/###-#### x#???", + "### ### #### x#???", + "##########" + ] + }; + return dojox.validate.isNumberFormat(value, flags); // Boolean +} + +dojox.validate.us.isSocialSecurityNumber = function(/*String*/value){ + // summary: Validates social security number + var flags = { + format: [ + "###-##-####", + "### ## ####", + "#########" + ] + }; + return dojox.validate.isNumberFormat(value, flags); // Boolean +} + +dojox.validate.us.isZipCode = function(/*String*/value){ + // summary: Validates U.S. zip-code + var flags = { + format: [ + "#####-####", + "##### ####", + "#########", + "#####" + ] + }; + return dojox.validate.isNumberFormat(value, flags); // Boolean +} + +} diff --git a/includes/js/dojox/validate/web.js b/includes/js/dojox/validate/web.js new file mode 100644 index 0000000..c6d7a45 --- /dev/null +++ b/includes/js/dojox/validate/web.js @@ -0,0 +1,89 @@ +if(!dojo._hasResource["dojox.validate.web"]){ //_hasResource checks added by build. Do not use _hasResource directly in your code. +dojo._hasResource["dojox.validate.web"] = true; +dojo.provide("dojox.validate.web"); +dojo.require("dojox.validate._base"); + +dojox.validate.isIpAddress = function(/*String*/value, /*Object?*/flags) { + // summary: Validates an IP address + // + // description: + // Supports 5 formats for IPv4: dotted decimal, dotted hex, dotted octal, decimal and hexadecimal. + // Supports 2 formats for Ipv6. + // + // value A string. + // flags An object. All flags are boolean with default = true. + // flags.allowDottedDecimal Example, 207.142.131.235. No zero padding. + // flags.allowDottedHex Example, 0x18.0x11.0x9b.0x28. Case insensitive. Zero padding allowed. + // flags.allowDottedOctal Example, 0030.0021.0233.0050. Zero padding allowed. + // flags.allowDecimal Example, 3482223595. A decimal number between 0-4294967295. + // flags.allowHex Example, 0xCF8E83EB. Hexadecimal number between 0x0-0xFFFFFFFF. + // Case insensitive. Zero padding allowed. + // flags.allowIPv6 IPv6 address written as eight groups of four hexadecimal digits. + // flags.allowHybrid IPv6 address written as six groups of four hexadecimal digits + // followed by the usual 4 dotted decimal digit notation of IPv4. x:x:x:x:x:x:d.d.d.d + + var re = new RegExp("^" + dojox.regexp.ipAddress(flags) + "$", "i"); + return re.test(value); // Boolean +} + + +dojox.validate.isUrl = function(/*String*/value, /*Object?*/flags) { + // summary: Checks if a string could be a valid URL + // value: A string + // flags: An object + // flags.scheme Can be true, false, or [true, false]. + // This means: required, not allowed, or either. + // flags in regexp.host can be applied. + // flags in regexp.ipAddress can be applied. + // flags in regexp.tld can be applied. + + var re = new RegExp("^" + dojox.regexp.url(flags) + "$", "i"); + return re.test(value); // Boolean +} + +dojox.validate.isEmailAddress = function(/*String*/value, /*Object?*/flags) { + // summary: Checks if a string could be a valid email address + // + // value: A string + // flags: An object + // flags.allowCruft Allow address like <mailto:foo@yahoo.com>. Default is false. + // flags in regexp.host can be applied. + // flags in regexp.ipAddress can be applied. + // flags in regexp.tld can be applied. + + var re = new RegExp("^" + dojox.regexp.emailAddress(flags) + "$", "i"); + return re.test(value); // Boolean +} + +dojox.validate.isEmailAddressList = function(/*String*/value, /*Object?*/flags) { + // summary: Checks if a string could be a valid email address list. + // + // value A string. + // flags An object. + // flags.listSeparator The character used to separate email addresses. Default is ";", ",", "\n" or " ". + // flags in regexp.emailAddress can be applied. + // flags in regexp.host can be applied. + // flags in regexp.ipAddress can be applied. + // flags in regexp.tld can be applied. + + var re = new RegExp("^" + dojox.regexp.emailAddressList(flags) + "$", "i"); + return re.test(value); // Boolean +} + +dojox.validate.getEmailAddressList = function(/*String*/value, /*Object?*/flags) { + // summary: Check if value is an email address list. If an empty list + // is returned, the value didn't pass the test or it was empty. + // + // value: A string + // flags: An object (same as dojo.validate.isEmailAddressList) + + if(!flags) { flags = {}; } + if(!flags.listSeparator) { flags.listSeparator = "\\s;,"; } + + if ( dojox.validate.isEmailAddressList(value, flags) ) { + return value.split(new RegExp("\\s*[" + flags.listSeparator + "]\\s*")); // Array + } + return []; // Array +} + +} |