/**
 * 
 * 新增验证:
 * 增加min-value-number验证,例: 最小值为10.1 = min-value-10.1
 * 增加max-value-number验证,例: 最大值为-100.1 = min-value--100.1
 * 增加长度范围validate-length-range-minLength-maxLength验证,例: 最小长度为1,最大长度为10 = validate-length-range-1-10
 * 增加整型数字范围validate-int-range-minValue-maxValue验证,例: 5至20 = validate-int-range-5-20
 * 增加浮点数字范围validate-float-range-minValue-maxValue验证,例: -1.1至10 = validate-float-range--1.1-10
 * 增加min-length-number验证,例: 最小长度为10 = min-length-10
 * 增加max-length-number验证,例: 最大长度为10 = max-length-10
 * 增加文件类型 validate-file-type1-type2-typeX 的验证,例: validate-file-zip-png-jpeg 将验证文件是否为zip,png,jpeg格式之一
 * 增加中文日期验证 validate-date-cn
 * 增加相等验证validate-equals-item1-item2-itemX,判断输入的值为[item1,item2,itemX]之一
 * 
 * 修改记录:
 * 增加Validation._getInputValue() 取代$F()方法以对file input进行验证
 * 修正Validation.isVisible() 中while循环中elm可能不存在为空的问题
 * 增加Validation.create() for cache
 * 修改Validation.get()方法使用indexOf()的匹配模式,以适应可以通过class传递参数
 * 修改errorMsg可由方法返回
 * 增加Validation._getErrorMsg()方法
 * 修改advice可以动态修改
 * 国际化
 * 修改_getErrorMsg()中将useTitle可以在className中指定
 * 增加Validation.autoBind()方法,可以通过在form中的class设置required-validate自动验证form
 * 增加Validation.$()方法用于查找在Validation.autoBind()的validation
 * by badqiu (badqiu@gmail.com)
 */

/*
 * Really easy field validation with Prototype
 * http://tetlaw.id.au/view/blog/really-easy-field-validation-with-prototype
 * Andrew Tetlaw
 * Version 1.5.3 (2006-07-15)
 * 
 * Copyright (c) 2006 Andrew Tetlaw
 * http://www.opensource.org/licenses/mit-license.php
 */
Validator = Class.create();

Validator.messagesSourceEn = [
	['validation-failed' , 'Validation failed.'],
	['required' , 'This is a required field.'],
	['validate-number' , 'Please enter a valid number in this field.'],
	['validate-digits' , 'Please use numbers only in this field. please avoid spaces or other characters such as dots or commas.'],
	['validate-alpha' , 'Please use letters only (a-z) in this field.'],
	['validate-alphanum' , 'Please use only letters (a-z) or numbers (0-9) only in this field. No spaces or other characters are allowed.'],
	['validate-date' , 'Please enter a valid date.'],
	['validate-email' , 'Please enter a valid email address. For example fred@domain.com .'],
	['validate-url' , 'Please enter a valid URL.'],
	['validate-date-au' , 'Please use this date format: dd/mm/yyyy. For example 17/03/2006 for the 17th of March, 2006.'],
	['validate-currency-dollar' , 'Please enter a valid $ amount. For example $100.00 .'],
	['validate-one-required' , 'Please select one of the above options.'],
	['validate-date-cn' , 'Please use this date format: yyyy-mm-dd. For example 2006-03-16.'],
	['validate-integer' , 'Please enter a valid integer in this field'],
	['min-value' , 'min value is #{minValue}.'],
	['max-value' , 'max value is #{maxValue}.'],
	['min-length' , 'min length is #{minLength},current length is #{currentLength}.'],
	['max-length' , 'max length is #{maxLength},current length is #{currentLength}.'],
	['validate-int-range' , 'Please enter integer value between #{minValue} and #{maxValue}'],
	['validate-float-range' , 'Please enter number between #{minValue} and #{maxValue}'],
	['validate-length-range' , 'Please enter value length between #{minLength} and #{maxLength},current length is #{currentLength}'],
	['validate-file' , 'Please enter file type in [#{fileTypes}]'],
	['validate-equals' , 'Please enter value in [#{values}]']
]

Validator.messagesSourceCn = [
	['validation-failed' , '验证失败.'],
	['required' , '不能为空.'],
	['validate-number' , '请输入有效的数字.'],
	['validate-digits' , '请输入一个数字. 避免输入空格],逗号,分号等字符'],
	['validate-alpha' , '请输入[a-z]的字母.'],
	['validate-alphanum' , '请输入[a-z]的字母或是[0-9]的数字,其它字符是不允许的.'],
	['validate-date' , '请输入有效的日期.'],
	['validate-email' , '请输入有效的邮件地址,如 username@example.com.'],
	['validate-url' , '请输入有效的URL地址.'],
	['validate-date-au' , 'Please use this date format: dd/mm/yyyy. For example 17/03/2006 for the 17th of March, 2006.'],
	['validate-currency-dollar' , 'Please enter a valid $ amount. For example $100.00 .'],
	['validate-one-required' , '在上面选项至少选择一个.'],
	['validate-date-cn' , '请使用这样的日期格式: yyyy-mm-dd. 例如:2006-03-17.'],
	['validate-integer' , '请输入正确的整数'],
	['min-value' , '最小值为#{minValue}'],
	['max-value' , '最大值为#{maxValue}'],
	['min-length' , '最小长度为#{minLength},当前长度为#{currentLength}.'],
	['max-length', '最大长度为#{maxLength},当前长度为#{currentLength}.'],
	['validate-int-range' , '输入值应该为 #{minValue} 至 #{maxValue} 的整数'],
	['validate-float-range' , '输入值应该为 #{minValue} 至 #{maxValue} 的数字'],
	['validate-length-range' , '输入值的长度应该在 #{minLength} 至 #{maxLength} 之间,当前长度为#{currentLength}'],
	['validate-file' , '文件类型应该为[#{fileTypes}]其中之一'],
	['validate-equals' , '输入的值应该为[#{values}]其中之一']
]

Validator.messagesSource = Validator.messagesSourceCn;
Validator.messages = {};
//init Validator.messages
Validator.messagesSource.each(function(ms){
	Validator.messages[ms[0]] = ms[1];
});

Validator.evaluate = function(msgKey,object) {
	return new Template(Validator.messages[msgKey]).evaluate(object);
}

Validator.prototype = {
	initialize : function(className, error, test, options) {
		this.options = Object.extend({}, options || {});
		this._test = test ? test : function(v,elm){ return true };
		this.error = error ? error : Validator.messages['validation-failed'];
		this.className = className;
	},
	test : function(v, elm) {
		return this._test(v,elm);
	}
}

var Validation = Class.create();

Validation.prototype = {
	initialize : function(form, options){
		this.options = Object.extend({
			onSubmit : false,//changed by jeffsang
			stopOnFirst : false,
			immediate : false,
			focusOnError : true,
			useTitles : false,
			onFormValidate : function(result, form) {},
			onElementValidate : function(result, elm) {}
		}, options || {});
		this.form = $(form);
		var id =  Validation.getElmID(form);
		Validation.validations[id] = this;
		if(this.options.onSubmit) Event.observe(this.form,'submit',this.onSubmit.bind(this),false);
		if(this.options.immediate) {
			var useTitles = this.options.useTitles;
			var callback = this.options.onElementValidate;
			Form.getElements(this.form).each(function(input) { // Thanks Mike!
				Event.observe(input, 'blur', function(ev) { Validation.validate(Event.element(ev),{useTitle : useTitles, onElementValidate : callback}); });
			});
		}
	},
	onSubmit :  function(ev){
		if(!this.validate()) Event.stop(ev);
	},
	validate : function() {
		var result = false;
		var useTitles = this.options.useTitles;
		var callback = this.options.onElementValidate;
		if(this.options.stopOnFirst) {
			result = Form.getElements(this.form).all(function(elm) { return Validation.validate(elm,{useTitle : useTitles, onElementValidate : callback}); });
		} else {
			result = Form.getElements(this.form).collect(function(elm) { return Validation.validate(elm,{useTitle : useTitles, onElementValidate : callback}); }).all();
		}
		if(!result && this.options.focusOnError) {
			Form.getElements(this.form).findAll(function(elm){return $(elm).hasClassName('validation-failed')}).first().focus()
		}
		this.options.onFormValidate(result, this.form);
		return result;
	},
	reset : function() {
		Form.getElements(this.form).each(Validation.reset);
	}
}

Object.extend(Validation, {
	validate : function(elm, options){
		options = Object.extend({
			useTitle : false,
			onElementValidate : function(result, elm) {}
		}, options || {});
		elm = $(elm);
		var cn = elm.classNames();
		return result = cn.all(function(value) {
			var test = Validation.test(value,elm,options.useTitle);
			options.onElementValidate(test, elm);
			return test;
		});
	},
	_getInputValue : function(elm) {
		var elm = $(elm);
		if(elm.type.toLowerCase() == 'file') {
			return elm.value;
		}else {
			return $F(elm);
		}
	},
	_getErrorMsg : function(useTitle,elm,error) {
		if( typeof(error) == 'function' ) {
			error = error(Validation._getInputValue(elm),elm);
		}
		if(!useTitle) {
			useTitle = elm.className.indexOf('useTitle') >= 0;
		}
		return useTitle ? ((elm && elm.title) ? elm.title : error) : error;
	},
	test : function(name, elm, useTitle) {
		var v = Validation.get(name);
		var prop = '__advice'+name.camelize();
		if(Validation.isVisible(elm) && !v.test(Validation._getInputValue(elm),elm)) {
			if(!elm[prop]) {
				var advice = Validation.getAdvice(name, elm);
				if(typeof advice == 'undefined') {
					var errorMsg = Validation._getErrorMsg(useTitle,elm,v.error);
					advice = '<div class="validation-advice" id="advice-' + name + '-' + Validation.getElmID(elm) +'" style="display:none">' + errorMsg + '</div>'
					switch (elm.type.toLowerCase()) {
						case 'checkbox':
						case 'radio':
							var p = elm.parentNode;
							if(p) {
								new Insertion.Bottom(p, advice);
							} else {
								new Insertion.After(elm, advice);
							}
							break;
						default:
							new Insertion.After(elm, advice);
				    }
					advice = $('advice-' + name + '-' + Validation.getElmID(elm));
				}
				if(typeof Effect == 'undefined') {
					advice.style.display = 'block';
				} else {
					new Effect.Appear(advice, {duration : 1 });
				}
			}
			var advice = Validation.getAdvice(name, elm);
			advice.innerHTML = Validation._getErrorMsg(useTitle,elm,v.error);
			elm[prop] = true;
			elm.removeClassName('validation-passed');
			elm.addClassName('validation-failed');
			return false;
		} else {
			var advice = Validation.getAdvice(name, elm);
			if(typeof advice != 'undefined') advice.hide();
			elm[prop] = '';
			elm.removeClassName('validation-failed');
			elm.addClassName('validation-passed');
			return true;
		}
	},
	isVisible : function(elm) {
		while(elm && elm.tagName != 'BODY') {
			if(!$(elm).visible()) return false;
			elm = elm.parentNode;
		}
		return true;
	},
	getAdvice : function(name, elm) {
		return Try.these(
			function(){ return $('advice-' + name + '-' + Validation.getElmID(elm)) },
			function(){ return $('advice-' + Validation.getElmID(elm)) }
		);
	},
	getElmID : function(elm) {
		return elm.id ? elm.id : elm.name;
	},
	reset : function(elm) {
		elm = $(elm);
		var cn = elm.classNames();
		cn.each(function(value) {
			var prop = '__advice'+value.camelize();
			if(elm[prop]) {
				var advice = Validation.getAdvice(value, elm);
				advice.hide();
				elm[prop] = '';
			}
			elm.removeClassName('validation-failed');
			elm.removeClassName('validation-passed');
		});
	},
	add : function(className, error, test, options) {
		var nv = {};
		nv[className] = new Validator(className, error, test, options);
		Object.extend(Validation.methods, nv);
	},
	addAllThese : function(validators) {
		var nv = {};
		$A(validators).each(function(value) {
				nv[value[0]] = new Validator(value[0], value[1], value[2], (value.length > 3 ? value[3] : {}));
			});
		Object.extend(Validation.methods, nv);
	},
	get : function(name) {
		var resultMethodName;
		for(var methodName in Validation.methods) {
			if(name == methodName) {
				resultMethodName = methodName;
				break;
			}
			if(name.indexOf(methodName) >= 0) {
				resultMethodName = methodName;
			}
		}
		return Validation.methods[resultMethodName] ? Validation.methods[resultMethodName] : new Validator();
		//return  Validation.methods[name] ? Validation.methods[name] : new Validator();
	},
	methods : {}
});

Validation.add('IsEmpty', '', function(v) {
				return  ((v == null) || (v.length == 0)); // || /^\s+$/.test(v));
			});

Validation.addAllThese([
	['required', Validator.messages['required'], function(v) {
				return !Validation.get('IsEmpty').test(v);
			}],
	['validate-number', Validator.messages['validate-number'], function(v) {
				return Validation.get('IsEmpty').test(v) || (!isNaN(v) && !/^\s+$/.test(v));
			}],
	['validate-digits', Validator.messages['validate-digits'], function(v) {
				return Validation.get('IsEmpty').test(v) ||  !/[^\d]/.test(v);
			}],
	['validate-alpha', Validator.messages['validate-alpha'], function (v) {
				return Validation.get('IsEmpty').test(v) ||  /^[a-zA-Z]+$/.test(v)
			}],
	['validate-alphanum', Validator.messages['validate-alphanum'], function(v) {
				return Validation.get('IsEmpty').test(v) ||  !/\W/.test(v)
			}],
	['validate-date', Validator.messages['validate-date'], function(v) {
				var test = new Date(v);
				return Validation.get('IsEmpty').test(v) || !isNaN(test);
			}],
	['validate-email', Validator.messages['validate-email'], function (v) {
				return Validation.get('IsEmpty').test(v) || /\w{1,}[@][\w\-]{1,}([.]([\w\-]{1,})){1,3}$/.test(v)
			}],
	['validate-url', Validator.messages['validate-url'], function (v) {
				return Validation.get('IsEmpty').test(v) || /^(http|https|ftp):\/\/(([A-Z0-9][A-Z0-9_-]*)(\.[A-Z0-9][A-Z0-9_-]*)+)(:(\d+))?\/?/i.test(v)
			}],
	['validate-date-au', Validator.messages['validate-date-au'], function(v) {
				if(Validation.get('IsEmpty').test(v)) return true;
				var regex = /^(\d{2})\/(\d{2})\/(\d{4})$/;
				if(!regex.test(v)) return false;
				var d = new Date(v.replace(regex, '$2/$1/$3'));
				return ( parseInt(RegExp.$2, 10) == (1+d.getMonth()) ) && 
							(parseInt(RegExp.$1, 10) == d.getDate()) && 
							(parseInt(RegExp.$3, 10) == d.getFullYear() );
			}],
	['validate-currency-dollar', Validator.messages['validate-currency-dollar'], function(v) {
				// [$]1[##][,###]+[.##]
				// [$]1###+[.##]
				// [$]0.##
				// [$].##
				return Validation.get('IsEmpty').test(v) ||  /^\$?\-?([1-9]{1}[0-9]{0,2}(\,[0-9]{3})*(\.[0-9]{0,2})?|[1-9]{1}\d*(\.[0-9]{0,2})?|0(\.[0-9]{0,2})?|(\.[0-9]{1,2})?)$/.test(v)
			}],
	['validate-one-required', Validator.messages['validate-one-required'], function (v,elm) {
				var p = elm.parentNode;
				var options = p.getElementsByTagName('INPUT');
				return $A(options).any(function(elm) {
					return $F(elm);
				});
			}]
]);


//custom validate start
Validation.addAllThese([
	['validate-date-cn', Validator.messages['validate-date-cn'], function(v) {
				if(Validation.get('IsEmpty').test(v)) return true;
				var regex = /^(\d{4})-(\d{2})-(\d{2})$/;
				if(!regex.test(v)) return false;
				var d = new Date(v.replace(regex, '$1/$2/$3'));
				return ( parseInt(RegExp.$2, 10) == (1+d.getMonth()) ) && 
							(parseInt(RegExp.$3, 10) == d.getDate()) && 
							(parseInt(RegExp.$1, 10) == d.getFullYear() );
			}],
	['validate-integer', Validator.messages['validate-integer'], function(v) {
				return Validation.get('IsEmpty').test(v) || (/-?[\d]+/.test(v) && (parseInt(v) == parseFloat(v)));
			}]
]);
/*
 * Usage: min-length-number
 * Example: min-length-10
 */
Validation.add(
	'min-length', 
	function(v,elm) {
		var results = elm.className.match(/min-length-(\d*)/);
		var minLength = parseInt(results[1]);
		return Validator.evaluate('min-length',{minLength : minLength,currentLength : v.length});
	},
	function(v,elm) {
		var results = elm.className.match(/min-length-(\d*)/);
		var minLength = parseInt(results[1]);
		return Validation.get('IsEmpty').test(v) || v.length >= minLength
	}
)
/*
 * Usage: max-length-number
 * Example: max-length-10
 */
Validation.add(
	'max-length', 
	function(v,elm) {
		var results = elm.className.match(/max-length-(\d*)/);
		var maxLength = parseInt(results[1]);
		return Validator.evaluate('max-length',{maxLength : maxLength,currentLength : v.length});
	},
	function(v,elm) {
		var results = elm.className.match(/max-length-(\d*)/);
		var maxLength = parseInt(results[1]);
		return Validation.get('IsEmpty').test(v) || v.length <= maxLength
	}
)
/*
 * Usage: validate-file-type1-type2-typeX
 * Example: validate-file-png-jpg-jpeg
 */
Validation.add(
	'validate-file', 
	function(v,elm) {
		var results = elm.className.match(/validate-file-([a-zA-Z0-9-]*)/);
		var extentionNamesStr = results[1];
		var extentionNames = extentionNamesStr.split('-');
		return Validator.evaluate('validate-file',{fileTypes : extentionNames.join(',')});
	},
	function(v,elm) {
		var results = elm.className.match(/validate-file-([a-zA-Z0-9-]*)/);
		var extentionNamesStr = results[1];
		var extentionNames = extentionNamesStr.split('-');
		return Validation.get('IsEmpty').test(v) || $A(extentionNames).any(function(extentionName) {
			var pattern = new RegExp('\\.'+extentionName+'$','i');
			return pattern.test(v);
		});
	}
)

/*
 * Usage: validate-float-range-minValue-maxValue
 * Example: -2.1 to 3 = validate-float-range--2.1-3
 */
Validation.add(
	'validate-float-range', 
	function(v,elm) {
		if(!Validation.get('validate-number').test(v)) {
			return Validation.get('validate-number').error;
		}
		var results = elm.className.match(/validate-float-range-(-?[\d\.]*)-(-?[\d\.]*)/);
		var minValue = parseFloat(results[1]);
		var maxValue = parseFloat(results[2]);
		return Validator.evaluate('validate-float-range',{minValue : minValue,maxValue : maxValue});
	},
	function(v,elm) {
		var results = elm.className.match(/validate-float-range-(-?[\d\.]*)-(-?[\d\.]*)/);
		var minValue = parseFloat(results[1]);
		var maxValue = parseFloat(results[2]);
		return Validation.get('IsEmpty').test(v) || (Validation.get('validate-number').test(v) && (parseFloat(v) >= minValue && parseFloat(v) <= maxValue))
	}
)

/*
 * Usage: validate-int-range-minValue-maxValue
 * Example: -10 to 20 = validate-int-range-10-20
 */
Validation.add(
	'validate-int-range', 
	function(v,elm) {
		if(!Validation.get('validate-integer').test(v)) {
			return Validation.get('validate-integer').error;
		}
		var results = elm.className.match(/validate-int-range-(-?\d*)-(-?\d*)/);
		var minValue = parseInt(results[1]);
		var maxValue = parseInt(results[2]);
		return Validator.evaluate('validate-int-range',{minValue : minValue,maxValue : maxValue});
	},
	function(v,elm) {
		var results = elm.className.match(/validate-int-range-(-?\d*)-(-?\d*)/);
		var minValue = parseInt(results[1]);
		var maxValue = parseInt(results[2]);
		return Validation.get('IsEmpty').test(v) || (Validation.get('validate-integer').test(v) && (parseInt(v) >= minValue && parseInt(v) <= maxValue))
	} 
)

/*
 * Usage: validate-length-range-minLength-maxLength
 * Example: 10 to 20 = validate-length-range-10-20
 */
Validation.add(
	'validate-length-range', 
	function(v,elm) {
		var results = elm.className.match(/validate-length-range-(\d*)-(\d*)/);
		var minLength = parseInt(results[1]);
		var maxLength = parseInt(results[2]);
		return Validator.evaluate('validate-length-range',{minLength : minLength,maxLength : maxLength,currentLength : v.length});
	},
	function(v,elm) {
		var results = elm.className.match(/validate-length-range-(\d*)-(\d*)/);
		var minLength = parseInt(results[1]);
		var maxLength = parseInt(results[2]);
		return Validation.get('IsEmpty').test(v) || (v.length >= minLength && v.length <= maxLength)
	}
)

/*
 * Usage: max-value-number
 * Example: max-value-10
 */
Validation.add(
	'max-value', 
	function(v,elm) {
		if(!Validation.get('validate-number').test(v)) {
			return Validation.get('validate-number').error;
		}
		var results = elm.className.match(/max-value-(-?[\d\.]*)/);
		var value = parseFloat(results[1]);
		return Validator.evaluate('max-value',{maxValue : value});
	},
	function(v,elm) {
		var results = elm.className.match(/max-value-(-?[\d\.]*)/);
		var value = parseFloat(results[1]);
		return Validation.get('IsEmpty').test(v) || (Validation.get('validate-number').test(v) && parseFloat(v) <= value);
	}
)

/*
 * Usage: min-value-number
 * Example: min-value-10
 */
Validation.add(
	'min-value', 
	function(v,elm) {
		if(!Validation.get('validate-number').test(v)) {
			return Validation.get('validate-number').error;
		}
		var results = elm.className.match(/min-value-(-?[\d\.]*)/);
		var value = parseFloat(results[1]);
		return Validator.evaluate('min-value',{minValue : value});
	},
	function(v,elm) {
		var results = elm.className.match(/min-value-(-?[\d\.]*)/);
		var value = parseFloat(results[1]);
		return Validation.get('IsEmpty').test(v) || (Validation.get('validate-number').test(v) && parseFloat(v) >= value);
	}
)

/*
 * Usage: validate-equals-item1-item2-itemX
 * Example: validate-equals-AA-BB-CC
 */
Validation.add(
	'validate-equals', 
	function(v,elm) {
		var results = elm.className.match(/validate-equals-([\S]*)/);
		var expectedValuesStr = results[1];
		var expectedValues = expectedValuesStr.split('-');
		return Validator.evaluate('validate-equals',{values : expectedValues.join(',')});
	},
	function(v,elm) {
		var results = elm.className.match(/validate-equals-([\S]*)/);
		var expectedValuesStr = results[1];
		var expectedValues = expectedValuesStr.split('-');
		return Validation.get('IsEmpty').test(v) || $A(expectedValues).any(function(expectedValue) {
			return v == expectedValue;
		});
	}
)

//Validation.create() for cache
Validation._cacheValidations = {};
Validation.create = function(form,options) {
	var inCacheValidation = Validation._cacheValidations[form];
	if(inCacheValidation)
		return  inCacheValidation;
	var validation = new Validation(form,options);
	Validation._cacheValidations[form] = validation;
	return validation;
}

Validation.validations = {};
Validation.autoBind = function() {
	 var forms = document.getElementsByClassName('required-validate');
	 $A(forms).each(function(form){
		var validation = Validation.create(form,{immediate:true});
		Event.observe(form,'reset',function() {validation.reset();},false);
	 });
};

Validation.$ = function(id) {
	return Validation.validations[id];
}

Event.observe(window,'load',Validation.autoBind,false);