/*
* Really easy field validation with Prototype
* http://tetlaw.id.au/view/javascript/really-easy-field-validation
* Andrew Tetlaw
* Version 1.5.4.1 (2007-01-05)
* 
* Copyright (c) 2007 Andrew Tetlaw
* 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.
* 
*/
var Validator = Class.create();

Validator.prototype = {
  initialize : function(className, error, test, options) {
    if (typeof test == 'function') {
      this.options = $H(options);
      this._test = test;
    } else {
      this.options = $H(test);
      this._test = function() {
        return true
      };
    }
    this.error = error || 'Validation failed.';
    this.className = className;
  },
  test : function(v, elm) {
    return (this._test(v, elm) && this.options.all(function(p) {
      return Validator.methods[p.key] ? Validator.methods[p.key](v, elm, p.value) : true;
    }));
  }
}
Validator.methods = {
  pattern : function(v, elm, opt) {
    return Validation.get('IsEmpty').test(v) || opt.test(v)
  },
  minLength : function(v, elm, opt) {
    return v.length >= opt
  },
  maxLength : function(v, elm, opt) {
    return v.length <= opt
  },
  min : function(v, elm, opt) {
    return v >= parseFloat(opt)
  },
  max : function(v, elm, opt) {
    return v <= parseFloat(opt)
  },
  notOneOf : function(v, elm, opt) {
    return $A(opt).all(function(value) {
      return v != value;
    })
  },
  oneOf : function(v, elm, opt) {
    return $A(opt).any(function(value) {
      return v == value;
    })
  },
  is : function(v, elm, opt) {
    return v == opt
  },
  isNot : function(v, elm, opt) {
    return v != opt
  },
  equalToField : function(v, elm, opt) {
    return v == $F(opt)
  },
  notEqualToField : function(v, elm, opt) {
    return v != $F(opt)
  },
  include : function(v, elm, opt) {
    return $A(opt).all(function(value) {
      return Validation.get(value).test(v, elm);
    })
  }
}

var Validation = Class.create();

Validation.prototype = {
  initialize : function(form, options) {
    this.options = Object.extend({
      onSubmit : true,
      stopOnFirst : false,
      immediate : false,
      focusOnError : true,
      useTitles : false,
      onFormValidate : function(result, form) {
      },
      onElementValidate : function(result, elm) {
      }
    }, options || {});
    this.form = $(form);
    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;
    });
  },
  test : function(name, elm, useTitle) {
    var v = Validation.get(name);
    var prop = '__advice' + name.camelize();
    try {
      if (Validation.isVisible(elm) && !v.test($F(elm), elm)) {
        if (!elm[prop]) {
          var advice = Validation.getAdvice(name, elm);
          if (advice == null) {
            var errorMsg = useTitle ? ((elm && elm.title) ? elm.title : v.error) : 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 = Validation.getAdvice(name, elm);
          }//shkate - add posibility to setup standart valiadors message into user element
          else if (advice.innerHTML.length == 0) {
            advice.innerHTML = useTitle ? ((elm && elm.title) ? elm.title : v.error) : v.error;
            advice = Validation.getAdvice(name, elm);
          }
          if (typeof Effect == 'undefined') {
            advice.style.display = 'block';
          } else {
            new Effect.Appear(advice, {duration : 1 });
          }
        }
        elm[prop] = true;
        elm.removeClassName('validation-passed');
        elm.addClassName('validation-failed');
        return false;
      } else {
        var advice = Validation.getAdvice(name, elm);
        if (advice != null) advice.hide();
        elm[prop] = '';
        elm.removeClassName('validation-failed');
        elm.addClassName('validation-passed');
        return true;
      }
    } catch(e) {
      throw(e)
    }
  },
  isVisible : function(elm) {
    while (elm.tagName != 'BODY') {
      if (!$(elm).visible()) return false;
      elm = elm.parentNode;
    }
    return true;
  },
  getAdvice : function(name, elm) {
    return $('advice-' + name + '-' + Validation.getElmID(elm)) || $('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) {
    return  Validation.methods[name] ? Validation.methods[name] : Validation.methods['_LikeNoIDIEverSaw_'];
  },
  methods : {
    '_LikeNoIDIEverSaw_' : new Validator('_LikeNoIDIEverSaw_', '', {})
  }
});

String.prototype.trim = function() {
  a = this.replace(/^\s+/, '');
  return a.replace(/\s+$/, '');
};
Validation.add('IsEmpty', '', function(v) {
  return  ((v == null) || (v.trim().length == 0)); // || /^\s+$/.test(v));
});

Validation.addAllThese([
    ['required', 'This is a required field.', function(v) {
      return !Validation.get('IsEmpty').test(v);
    }],
    ['validate-number', 'Please enter a valid number in this field.', function(v) {
      return Validation.get('IsEmpty').test(v) || (!isNaN(v) && !/^\s+$/.test(v));
    }],
    ['validate-digits', 'Please use numbers only in this field. please avoid spaces or other characters such as dots or commas.', function(v) {
      return Validation.get('IsEmpty').test(v) || !/[^\d]/.test(v);
    }],
    ['validate-alpha', 'Please use letters only (a-z) in this field.', function (v) {
      return Validation.get('IsEmpty').test(v) || /^[a-zA-Z]+$/.test(v)
    }],
    ['validate-alphanum', 'Please use only letters (a-z) or numbers (0-9) only in this field. No spaces or other characters are allowed.', function(v) {
      return Validation.get('IsEmpty').test(v) || !/\W/.test(v)
    }],
    ['validate-date', 'Please enter a valid date.', function(v) {
      var test = new Date(v);
      return Validation.get('IsEmpty').test(v) || !isNaN(test);
    }],
    ['validate-email', 'Please enter a valid email address. For example fred@domain.com .', function (v) {
      return Validation.get('IsEmpty').test(v) || /\w{1,}[@][\w\-]{1,}([.]([\w\-]{1,})){1,3}$/.test(v)
    }],
    ['validate-url', 'Please enter a valid 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', 'Please use this date format: dd/mm/yyyy. For example 17/03/2006 for the 17th of March, 2006.', 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', 'Please enter a valid $ amount. For example $100.00 .', 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-selection', 'Please make a selection', function(v, elm) {
      return elm.options ? elm.selectedIndex > 0 : !Validation.get('IsEmpty').test(v);
    }],
    ['validate-one-required', 'Please select one of the above options.', function (v, elm) {
      var p = elm.parentNode;
      var options = p.getElementsByTagName('INPUT');
      return $A(options).any(function(elm) {
        return $F(elm);
      });
    }]
    ]);