/**
 * The systems singleton Core. 
 * Current system event bus.
 */
var ffCore = {
	/**
	 * The event observers.
	 * @private
	 */
	_observers : {},
	/**
	 * Binds to an event type. Use of namespaces allowed.
	 * @param eventType {string} The event type.
	 * @param handler {function} The event handler.
	 */
	bind : function(eventType, handler){
		var types = eventType.split(' '),
			obs, ev, ns, i, f;
		for (i=0, f=types.length; i < f; i++){
			ev = types[i].replace(/\..*$/, '');
			ns = types[i].replace(/^.*\./, '');
			obs = this._observers[ev];
			if ( ! obs){ obs = (this._observers[ev] = []); }
			obs.push({ fn:handler, ns:ns });
		}
		return this;
	},
	/**
	 * Unbinds from event type. Use of namespaces allowed.
	 * @param eventType {string} The event type.
	 * @param handler {function} The event handler.
	 */
	unbind : function(eventType, handler){
		var types = eventType.split(' '),
			obs, ob, temp, ev, ns, i, f, j, g;
		for (i=0, f=types.length; i < f; i++){
			ev = types[i].replace(/\..*$/, '');
			ns = types[i].replace(/^.*\./, '');
			obs = this._observers[ev];
			if (obs && obs.length){
				temp = [];
				for (j=0, g=obs.length; j < g; j++){
					if (obs[j].ns != ns && obs[j].fn != handler){
						temp.push(ob);
					}
				}
			}
			this._observers[ev] = temp;
		}
		return this;
	},
	/**
	 * Triggers an event.
	 * @param eventType {string} The event type.
	 * @param eventData {object} Extra data for the event object.
	 */
	trigger : function(eventType, eventData){
		var types = eventType.split(' '),
			obs, evObj, i, f, j, g;
		for (i=0, f=types.length; i < f; i++){
			obs = this._observers[types[i]];
			if (obs && obs.length){
				evObj = $.extend({}, eventData, {type:types[i]});
				for (j=0, g=obs.length; j < g; j++){
					obs[j].fn(evObj);
				}
			}
		}
		return this;
	},
	/**
	 * Indicates if the browser is Safari mobile.
	 */
	isMobileSafari : false,
	/**
	 * Indicates if the browser is IE 6.
	 */
	isIE6 : false,
	/**
	 * Indicates if the browser is IE7.
	 */
	isIE7 : false,
	/**
	 * Initializes the core.
	 */
	init : function(){
		this.isMobileSafari = /(iPad|iPhone|iPod)/.test(navigator.platform.toString());
		this.isIE6 = (navigator.appName == "Microsoft Internet Explorer" && parseInt(navigator.appVersion) == 4 && navigator.appVersion.indexOf("MSIE 6.0") != -1);
		this.isIE7 = (document.all && !window.opera && window.XMLHttpRequest);
	}
};
(function($){

	/**
	 * ffDialog entry point.
	 * @param options {object} The options.
	 */
	$.fn.ffDialog = function(options, name, value){
		// Handles function via $.fn.ffDialog(functionName)
		if (typeof options == "string"){
			return this.each(function(){
				var ffDialogInstance = $(this).data('ffDialog');
				if ( ! ffDialogInstance){
					return this; 
				}
				if (options == 'option'){
					if (value == undefined){
						return _getOption(ffDialogInstance, name);
					}
					else {
						_setOption(ffDialogInstance, name, value);
					}
				}
				else if (options == 'update'){
					updatePosition(ffDialogInstance, true);
	            }
				else if (options == 'show'){
					ffCore.trigger('closeDialogs', { except:ffDialogInstance });
					show(ffDialogInstance);
				}
				else if (options == 'hide'){
					hide(ffDialogInstance);
				}
				else if (options == 'destroy'){
					// TODO: Implement
				}
			});
		}

		// ffDialog real construction.
		var instance = $.extend(
			{}, 
			$.fn.ffDialog.defaults,
			{ isPositionFixed : ! ffCore.isMobileSafari },
			options
		);

		return this.each(function(){
			var self = $(this);

			// Avoid duplicate construction.
			if (self.data('ffDialog')){ return; }

			var dialog = (instance.content = self).hide(),
				children = dialog.children(),
				container = $('<div class="clearfix"></div>'),
				closeButton = (instance._closeBtn = $('<a href="#" class="close dialogClose graphic" title="Close Window">Close Window</a>'));

			dialog.addClass('ffDialog')
				.append(closeButton)
				.append(container.append(children))
				.css({
					position : instance.isPositionFixed ? 'fixed' : 'absolute',
					'z-index' : 99999
				})
				.data('ffDialog', instance);

			closeButton.bind('click', function (e) {
				e.preventDefault();
				hide(instance);
			});

			self.data('ffDialog', instance)
		});
	};

	// Binds to window events.
	$(window).bind('resize scroll', function(e){ 
		$('.ffDialog:visible').each(function(){
			var self = $(this),
				instance = self.data('ffDialog');
			if (e.type == 'resize' && instance.autoPositionOnResize){ 
				updatePosition(instance, true);
			}
			if (e.type == 'scroll' && instance.autoPositionOnScroll){ 
				updatePosition(instance);
			}
		});
	});

	// Binds to document events.
	$(document).bind('mousedown keyup', function (e) {
		var visibleDialog = $('div.ffDialog:visible'),
			i, f, isContentClick, instance;

		for (i=0, f=visibleDialog.length; i<f; i++){
			instance = visibleDialog.eq(i).data('ffDialog');

			// Close on Esc press.
			if (e.keyCode == 27) {
				if (instance && instance.enableEscapeButton){
					hide(instance);
				}
			}
			else if (e.type == 'mousedown'){
				isContentClick = $(e.target).closest('.ffDialog').length;
				// Hide on Content click
				if (isContentClick && instance.hideOnContentClick){ 
					hide(instance);
				}
				// Hide on General click.
				else if ( ! isContentClick && instance.hideOnOverlayClick) {
					hide(instance);
				}
			}
		}
    });

	ffCore.bind('closeDialogs', function(e){
		$('.ffDialog').ffDialog('hide');
	});

	/**
	 * Opens a dialog.
	 * @param ffDialog {ffDialog} The ffDialog.
	 */
	function show(ffDialog){
		ffCore.trigger('dialogOpening');
		ffDialog._closeBtn.css('display', ffDialog.showCloseButton ? 'block' : 'none');
		_showKnockout();
		ffDialog.content.fadeIn(ffDialog.duration, function(){
			ffDialog.content.trigger('dialogOpenComplete');
		});
		ffDialog.content.trigger('dialogOpen');
		updatePosition(ffDialog, true, false);
	};

	/**
	 * Hides a dialog.
	 * @param ffDialog {ffDialog} The ffDialog.
	 */
	function hide(ffDialog){
		ffDialog.content.fadeOut(ffDialog.duration, function(){
			ffDialog.content.trigger('dialogHideComplete');
		});
		_hideKnockout();
	};

	/**
	 * Updates an ffDialog position. Constraints to inside the window and
	 * only updates if the windows is bigger than the content.
	 * @param animate {boolean} [Opt] Indicates to animate to the new position.
	 * @param force {boolean} [Opt] Indicates to ignore constraints and force the position.
	 */
	function updatePosition(ffDialog, force, animate){
		animate = animate == undefined ? ffDialog.animatePositionChange : animate;
		
		var dialog = ffDialog.content,
			dialogH = dialog.height(),
			win = $(window),
			winH = win.height();

		// Change option if the window is smaller than the dialog.
		if (ffDialog.isPositionFixed && dialogH >= winH){
			_setOption(ffDialog, 'isPositionFixed', false);
		}

		if (force || ( ! ffDialog.isPositionFixed && dialogH < winH)){
			// No negative values here!
			var dialogW = dialog.width(),
				winW = win.width(),
				scrollT = ffDialog.isPositionFixed ? 0 : win.scrollTop(),
				pos = {
					top:Math.max((winH / 2) - (dialogH / 2) + scrollT, ffDialog.margin),
					left:Math.max((winW / 2) - (dialogW / 2), ffDialog.margin)
				};

			if (animate){
				dialog.animate(pos, 250).dequeue();
			}
			else {
				dialog.css(pos);
			}
		}
	};

	/**
	 * The option getter method.
	 */
	function _getOption(ffDialog, name, value){
	}

	/** 
	 * The option setter method.
	 */
	function _setOption(ffDialog, name, value){
	};

	/**
	 * Shows the dialog knockout.
	 */
	function _showKnockout(){
		if ( ! $.fn.ffDialog._knockout){
			$.fn.ffDialog._knockout = $('<div id="dialogKnockout"></div>')
				.css({
					'height': $(window).height() + 5000,
					'width': 5000,
					'left': 0,
					'top': 0
				})
				.appendTo('body');
		}
		$.fn.ffDialog._knockout.show();
	};

	/**
	 * Hides the dialog knockout.
	 */
	function _hideKnockout(){
		$.fn.ffDialog._knockout && $.fn.ffDialog._knockout.hide();
	};

	/**
	 * The ffDialog defaults.
	 */
	$.fn.ffDialog.defaults = {
		_closeBtn:null,					// The close button.
		animatePositionChange:false,	// Animate the position changes.
		autoPositionOnResize:true,		// Indicates if the dialog autopositions on windows resize
		autoPositionOnScroll:true,		// Indicates if the dialog autopositions on windows resize
		hideOnContentClick:false,		// Indicates if it hides when clicking on content.
		hideOnOverlayClick:true,		// Indicates if it clses on overlay click.
		content:null,					// The content selector or node.
		duration:'fast',				// The in/out animation duration.
		enableEscapeButton:true,		// Indicates if pressing Esc button hides ffDialog.
		isPositionFixed:true,			// Is position fixed.
		margin:10,						// Viewport margin
		onHide:null,
		onHideComplete:null,
		onOpen:null,
		onOpenComplete:null,
		showCloseButton:true			// Shows a close button
	};

})(jQuery);
/*
* jQuery validation plug-in 1.7
*
* http://bassistance.de/jquery-plugins/jquery-plugin-validation/
* http://docs.jquery.com/Plugins/Validation
*
* Copyright (c) 2006 - 2008 Jörn Zaefferer
*
* $Id: jquery.validate.js 6403 2009-06-17 14:27:16Z joern.zaefferer $
*
* Dual licensed under the MIT and GPL licenses:
*   http://www.opensource.org/licenses/mit-license.php
*   http://www.gnu.org/licenses/gpl.html
*/

(function ($) {

    $.extend($.fn, {
        // http://docs.jquery.com/Plugins/Validation/validate
        validate: function (options) {

            // if nothing is selected, return nothing; can't chain anyway
            if (!this.length) {
                options && options.debug && window.console && console.warn("nothing selected, can't validate, returning nothing");
                return;
            }

            // check if a validator for this form was already created
            var validator = $.data(this[0], 'validator');
            if (validator) {
                return validator;
            }

            validator = new $.validator(options, this[0]);
            $.data(this[0], 'validator', validator);

            if (validator.settings.onsubmit) {

                // allow suppresing validation by adding a cancel class to the submit button
                this.find("input, button").filter(".cancel").click(function () {
                    validator.cancelSubmit = true;
                });

                // when a submitHandler is used, capture the submitting button
                if (validator.settings.submitHandler) {
                    this.find("input, button").filter(":submit").click(function () {
                        validator.submitButton = this;
                    });
                }

                // validate the form on submit
                this.submit(function (event) {
                    if (validator.settings.debug)
                    // prevent form submit to be able to see console output
                        event.preventDefault();

                    function handle() {
                        if (validator.settings.submitHandler) {
                            if (validator.submitButton) {
                                // insert a hidden input as a replacement for the missing submit button
                                var hidden = $("<input type='hidden'/>").attr("name", validator.submitButton.name).val(validator.submitButton.value).appendTo(validator.currentForm);
                            }
                            validator.settings.submitHandler.call(validator, validator.currentForm);
                            if (validator.submitButton) {
                                // and clean up afterwards; thanks to no-block-scope, hidden can be referenced
                                hidden.remove();
                            }
                            return false;
                        }
                        return true;
                    }

                    // prevent submit for invalid forms or custom submit handlers
                    if (validator.cancelSubmit) {
                        validator.cancelSubmit = false;
                        return handle();
                    }
                    if (validator.form()) {
                        if (validator.pendingRequest) {
                            validator.formSubmitted = true;
                            return false;
                        }
                        return handle();
                    } else {
                        validator.focusInvalid();
                        return false;
                    }
                });
            }

            return validator;
        },
        // http://docs.jquery.com/Plugins/Validation/valid
        valid: function () {
            if ($(this[0]).is('form')) {
                return this.validate().form();
            } else {
                var valid = true;
                var validator = $(this[0].form).validate();
                this.each(function () {
                    valid &= validator.element(this);
                });
                return valid;
            }
        },
        // attributes: space seperated list of attributes to retrieve and remove
        removeAttrs: function (attributes) {
            var result = {},
			$element = this;
            $.each(attributes.split(/\s/), function (index, value) {
                result[value] = $element.attr(value);
                $element.removeAttr(value);
            });
            return result;
        },
        // http://docs.jquery.com/Plugins/Validation/rules
        rules: function (command, argument) {
            var element = this[0];

            if (command) {
                var settings = $.data(element.form, 'validator').settings;
                var staticRules = settings.rules;
                var existingRules = $.validator.staticRules(element);
                switch (command) {
                    case "add":
                        $.extend(existingRules, $.validator.normalizeRule(argument));
                        staticRules[element.name] = existingRules;
                        if (argument.messages)
                            settings.messages[element.name] = $.extend(settings.messages[element.name], argument.messages);
                        break;
                    case "remove":
                        if (!argument) {
                            delete staticRules[element.name];
                            return existingRules;
                        }
                        var filtered = {};
                        $.each(argument.split(/\s/), function (index, method) {
                            filtered[method] = existingRules[method];
                            delete existingRules[method];
                        });
                        return filtered;
                }
            }

            var data = $.validator.normalizeRules(
		$.extend(
			{},
			$.validator.metadataRules(element),
			$.validator.classRules(element),
			$.validator.attributeRules(element),
			$.validator.staticRules(element)
		), element);

            // make sure required is at front
            if (data.required) {
                var param = data.required;
                delete data.required;
                data = $.extend({ required: param }, data);
            }

            return data;
        }
    });

    // Custom selectors
    $.extend($.expr[":"], {
        // http://docs.jquery.com/Plugins/Validation/blank
        blank: function (a) { return !$.trim("" + a.value); },
        // http://docs.jquery.com/Plugins/Validation/filled
        filled: function (a) { return !!$.trim("" + a.value); },
        // http://docs.jquery.com/Plugins/Validation/unchecked
        unchecked: function (a) { return !a.checked; }
    });

    // constructor for validator
    $.validator = function (options, form) {
        this.settings = $.extend(true, {}, $.validator.defaults, options);
        this.currentForm = form;
        this.init();
    };

    $.validator.format = function (source, params) {
        if (arguments.length == 1)
            return function () {
                var args = $.makeArray(arguments);
                args.unshift(source);
                return $.validator.format.apply(this, args);
            };
        if (arguments.length > 2 && params.constructor != Array) {
            params = $.makeArray(arguments).slice(1);
        }
        if (params.constructor != Array) {
            params = [params];
        }
        $.each(params, function (i, n) {
            source = source.replace(new RegExp("\\{" + i + "\\}", "g"), n);
        });
        return source;
    };

    $.extend($.validator, {

        defaults: {
            messages: {},
            groups: {},
            rules: {},
            errorClass: "error",
            validClass: "valid",
            errorElement: "label",
            focusInvalid: true,
            errorContainer: $([]),
            errorLabelContainer: $([]),
            onsubmit: true,
            ignore: [],
            ignoreTitle: false,
            onfocusin: function (element) {
                this.lastActive = element;

                // hide error label and remove error class on focus if enabled
                if (this.settings.focusCleanup && !this.blockFocusCleanup) {
                    this.settings.unhighlight && this.settings.unhighlight.call(this, element, this.settings.errorClass, this.settings.validClass);
                    this.errorsFor(element).hide();
                }
            },
            onfocusout: function (element) {
                if (!this.checkable(element) && (element.name in this.submitted || !this.optional(element))) {
                    this.element(element);
                }
            },
            onkeyup: function (element) {
                if (element.name in this.submitted || element == this.lastElement) {
                    this.element(element);
                }
            },
            onclick: function (element) {
                // click on selects, radiobuttons and checkboxes
                if (element.name in this.submitted)
                    this.element(element);
                // or option elements, check parent select in that case
                else if (element.parentNode.name in this.submitted)
                    this.element(element.parentNode);
            },
            highlight: function (element, errorClass, validClass) {
                $(element).addClass(errorClass).removeClass(validClass);
            },
            unhighlight: function (element, errorClass, validClass) {
                $(element).removeClass(errorClass).addClass(validClass);
            }
        },

        // http://docs.jquery.com/Plugins/Validation/Validator/setDefaults
        setDefaults: function (settings) {
            $.extend($.validator.defaults, settings);
        },

        messages: {
            required: "This field is required.",
            remote: "Please fix this field.",
            email: "Please enter a valid email address.",
            url: "Please enter a valid URL.",
            date: "Please enter a valid date.",
            dateISO: "Please enter a valid date (ISO).",
            number: "Please enter a valid number.",
            digits: "Please enter only digits.",
            creditcard: "Please enter a valid credit card number.",
            equalTo: "Please enter the same value again.",
            accept: "Please enter a value with a valid extension.",
            maxlength: $.validator.format("Please enter no more than {0} characters."),
            minlength: $.validator.format("Please enter at least {0} characters."),
            rangelength: $.validator.format("Please enter a value between {0} and {1} characters long."),
            range: $.validator.format("Please enter a value between {0} and {1}."),
            max: $.validator.format("Please enter a value less than or equal to {0}."),
            min: $.validator.format("Please enter a value greater than or equal to {0}.")
        },

        autoCreateRanges: false,

        prototype: {

            init: function () {
                this.labelContainer = $(this.settings.errorLabelContainer);
                this.errorContext = this.labelContainer.length && this.labelContainer || $(this.currentForm);
                this.containers = $(this.settings.errorContainer).add(this.settings.errorLabelContainer);
                this.submitted = {};
                this.valueCache = {};
                this.pendingRequest = 0;
                this.pending = {};
                this.invalid = {};
                this.reset();

                var groups = (this.groups = {});
                $.each(this.settings.groups, function (key, value) {
                    $.each(value.split(/\s/), function (index, name) {
                        groups[name] = key;
                    });
                });
                var rules = this.settings.rules;
                $.each(rules, function (key, value) {
                    rules[key] = $.validator.normalizeRule(value);
                });

                function delegate(event) {
                    var validator = $.data(this[0].form, "validator"),
					eventType = "on" + event.type.replace(/^validate/, "");
                    validator.settings[eventType] && validator.settings[eventType].call(validator, this[0]);
                }
                $(this.currentForm)
				.validateDelegate(":text, :password, :file, select, textarea", "focusin focusout keyup", delegate)
				.validateDelegate(":radio, :checkbox, select, option", "click", delegate);

                if (this.settings.invalidHandler)
                    $(this.currentForm).bind("invalid-form.validate", this.settings.invalidHandler);
            },

            // http://docs.jquery.com/Plugins/Validation/Validator/form
            form: function () {
                this.checkForm();
                $.extend(this.submitted, this.errorMap);
                this.invalid = $.extend({}, this.errorMap);
                if (!this.valid())
                    $(this.currentForm).triggerHandler("invalid-form", [this]);
                this.showErrors();
                return this.valid();
            },

            checkForm: function () {
                this.prepareForm();
                for (var i = 0, elements = (this.currentElements = this.elements()); elements[i]; i++) {
                    this.check(elements[i]);
                }
                return this.valid();
            },

            // http://docs.jquery.com/Plugins/Validation/Validator/element
            element: function (element) {
                element = this.clean(element);
                this.lastElement = element;
                this.prepareElement(element);
                this.currentElements = $(element);
                var result = this.check(element);
                if (result) {
                    delete this.invalid[element.name];
                } else {
                    this.invalid[element.name] = true;
                }
                if (!this.numberOfInvalids()) {
                    // Hide error containers on last error
                    this.toHide = this.toHide.add(this.containers);
                }
                this.showErrors();
                return result;
            },

            // http://docs.jquery.com/Plugins/Validation/Validator/showErrors
            showErrors: function (errors) {
                if (errors) {
                    // add items to error list and map
                    $.extend(this.errorMap, errors);
                    this.errorList = [];
                    for (var name in errors) {
                        this.errorList.push({
                            message: errors[name],
                            element: this.findByName(name)[0]
                        });
                    }
                    // remove items from success list
                    this.successList = $.grep(this.successList, function (element) {
                        return !(element.name in errors);
                    });
                }
                this.settings.showErrors
				? this.settings.showErrors.call(this, this.errorMap, this.errorList)
				: this.defaultShowErrors();
            },

            // http://docs.jquery.com/Plugins/Validation/Validator/resetForm
            resetForm: function () {
                if ($.fn.resetForm)
                    $(this.currentForm).resetForm();
                this.submitted = {};
                this.prepareForm();
                this.hideErrors();
                this.elements().removeClass(this.settings.errorClass);
            },

            numberOfInvalids: function () {
                return this.objectLength(this.invalid);
            },

            objectLength: function (obj) {
                var count = 0;
                for (var i in obj)
                    count++;
                return count;
            },

            hideErrors: function () {
                this.addWrapper(this.toHide).hide();
            },

            valid: function () {
                return this.size() == 0;
            },

            size: function () {
                return this.errorList.length;
            },

            focusInvalid: function () {
                if (this.settings.focusInvalid) {
                    try {
                        $(this.findLastActive() || this.errorList.length && this.errorList[0].element || [])
					.filter(":visible")
					.focus()
                        // manually trigger focusin event; without it, focusin handler isn't called, findLastActive won't have anything to find
					.trigger("focusin");
                    } catch (e) {
                        // ignore IE throwing errors when focusing hidden elements
                    }
                }
            },

            findLastActive: function () {
                var lastActive = this.lastActive;
                return lastActive && $.grep(this.errorList, function (n) {
                    return n.element.name == lastActive.name;
                }).length == 1 && lastActive;
            },

            elements: function () {
                var validator = this,
				rulesCache = {};

                // select all valid inputs inside the form (no submit or reset buttons)
                // workaround $Query([]).add until http://dev.jquery.com/ticket/2114 is solved
                return $([]).add(this.currentForm.elements)
			.filter(":input")
			.not(":submit, :reset, :image, [disabled]")
			.not(this.settings.ignore)
			.filter(function () {
			    !this.name && validator.settings.debug && window.console && console.error("%o has no name assigned", this);

			    // select only the first element for each name, and only those with rules specified
			    if (this.name in rulesCache || !validator.objectLength($(this).rules()))
			        return false;

			    rulesCache[this.name] = true;
			    return true;
			});
            },

            clean: function (selector) {
                return $(selector)[0];
            },

            errors: function () {
                return $(this.settings.errorElement + "." + this.settings.errorClass, this.errorContext);
            },

            reset: function () {
                this.successList = [];
                this.errorList = [];
                this.errorMap = {};
                this.toShow = $([]);
                this.toHide = $([]);
                this.currentElements = $([]);
            },

            prepareForm: function () {
                this.reset();
                this.toHide = this.errors().add(this.containers);
            },

            prepareElement: function (element) {
                this.reset();
                this.toHide = this.errorsFor(element);
            },

            check: function (element) {
                element = this.clean(element);

                // if radio/checkbox, validate first element in group instead
                if (this.checkable(element)) {
                    element = this.findByName(element.name)[0];
                }

                var rules = $(element).rules();
                var dependencyMismatch = false;
                for (method in rules) {
                    var rule = { method: method, parameters: rules[method] };
                    try {
                        var result = $.validator.methods[method].call(this, element.value.replace(/\r/g, ""), element, rule.parameters);

                        // if a method indicates that the field is optional and therefore valid,
                        // don't mark it as valid when there are no other rules
                        if (result == "dependency-mismatch") {
                            dependencyMismatch = true;
                            continue;
                        }
                        dependencyMismatch = false;

                        if (result == "pending") {
                            this.toHide = this.toHide.not(this.errorsFor(element));
                            return;
                        }

                        if (!result) {
                            this.formatAndAdd(element, rule);
                            return false;
                        }
                    } catch (e) {
                        this.settings.debug && window.console && console.log("exception occured when checking element " + element.id
						 + ", check the '" + rule.method + "' method", e);
                        throw e;
                    }
                }
                if (dependencyMismatch)
                    return;
                if (this.objectLength(rules))
                    this.successList.push(element);
                return true;
            },

            // return the custom message for the given element and validation method
            // specified in the element's "messages" metadata
            customMetaMessage: function (element, method) {
                if (!$.metadata)
                    return;

                var meta = this.settings.meta
				? $(element).metadata()[this.settings.meta]
				: $(element).metadata();

                return meta && meta.messages && meta.messages[method];
            },

            // return the custom message for the given element name and validation method
            customMessage: function (name, method) {
                var m = this.settings.messages[name];
                return m && (m.constructor == String
				? m
				: m[method]);
            },

            // return the first defined argument, allowing empty strings
            findDefined: function () {
                for (var i = 0; i < arguments.length; i++) {
                    if (arguments[i] !== undefined)
                        return arguments[i];
                }
                return undefined;
            },

            defaultMessage: function (element, method) {
                return this.findDefined(
				this.customMessage(element.name, method),
				this.customMetaMessage(element, method),
                // title is never undefined, so handle empty string as undefined
				!this.settings.ignoreTitle && element.title || undefined,
				$.validator.messages[method],
				"<strong>Warning: No message defined for " + element.name + "</strong>"
			);
            },

            formatAndAdd: function (element, rule) {
                var message = this.defaultMessage(element, rule.method),
				theregex = /\$?\{(\d+)\}/g;
                if (typeof message == "function") {
                    message = message.call(this, rule.parameters, element);
                } else if (theregex.test(message)) {
                    message = jQuery.format(message.replace(theregex, '{$1}'), rule.parameters);
                }
                this.errorList.push({
                    message: message,
                    element: element
                });

                this.errorMap[element.name] = message;
                this.submitted[element.name] = message;
            },

            addWrapper: function (toToggle) {
                if (this.settings.wrapper)
                    toToggle = toToggle.add(toToggle.parent(this.settings.wrapper));
                return toToggle;
            },

            defaultShowErrors: function () {
                for (var i = 0; this.errorList[i]; i++) {
                    var error = this.errorList[i];
                    this.settings.highlight && this.settings.highlight.call(this, error.element, this.settings.errorClass, this.settings.validClass);
                    this.showLabel(error.element, error.message);
                }
                if (this.errorList.length) {
                    this.toShow = this.toShow.add(this.containers);
                }
                if (this.settings.success) {
                    for (var i = 0; this.successList[i]; i++) {
                        this.showLabel(this.successList[i]);
                    }
                }
                if (this.settings.unhighlight) {
                    for (var i = 0, elements = this.validElements(); elements[i]; i++) {
                        this.settings.unhighlight.call(this, elements[i], this.settings.errorClass, this.settings.validClass);
                    }
                }
                this.toHide = this.toHide.not(this.toShow);
                this.hideErrors();
                this.addWrapper(this.toShow).show();
            },

            validElements: function () {
                return this.currentElements.not(this.invalidElements());
            },

            invalidElements: function () {
                return $(this.errorList).map(function () {
                    return this.element;
                });
            },

            showLabel: function (element, message) {
                var label = this.errorsFor(element);
                if (label.length) {
                    // refresh error/success class
                    label.removeClass().addClass(this.settings.errorClass);

                    // check if we have a generated label, replace the message then
                    label.attr("generated") && label.html(message);
                } else {
                    // create label
                    label = $("<" + this.settings.errorElement + "/>")
					.attr({ "for": this.idOrName(element), generated: true })
					.addClass(this.settings.errorClass)
					.html(message || "");
                    if (this.settings.wrapper) {
                        // make sure the element is visible, even in IE
                        // actually showing the wrapped element is handled elsewhere
                        label = label.hide().show().wrap("<" + this.settings.wrapper + "/>").parent();
                    }
                    if (!this.labelContainer.append(label).length)
                        this.settings.errorPlacement
						? this.settings.errorPlacement(label, $(element))
						: label.insertAfter(element);
                }
                if (!message && this.settings.success) {
                    label.text("");
                    typeof this.settings.success == "string"
					? label.addClass(this.settings.success)
					: this.settings.success(label);
                }
                this.toShow = this.toShow.add(label);
            },

            errorsFor: function (element) {
                var name = this.idOrName(element);
                return this.errors().filter(function () {
                    return $(this).attr('for') == name;
                });
            },

            idOrName: function (element) {
                return this.groups[element.name] || (this.checkable(element) ? element.name : element.id || element.name);
            },

            checkable: function (element) {
                return /radio|checkbox/i.test(element.type);
            },

            findByName: function (name) {
                // select by name and filter by form for performance over form.find("[name=...]")
                var form = this.currentForm;
                return $(document.getElementsByName(name)).map(function (index, element) {
                    return element.form == form && element.name == name && element || null;
                });
            },

            getLength: function (value, element) {
                switch (element.nodeName.toLowerCase()) {
                    case 'select':
                        return $("option:selected", element).length;
                    case 'input':
                        if (this.checkable(element))
                            return this.findByName(element.name).filter(':checked').length;
                }
                return value.length;
            },

            depend: function (param, element) {
                return this.dependTypes[typeof param]
				? this.dependTypes[typeof param](param, element)
				: true;
            },

            dependTypes: {
                "boolean": function (param, element) {
                    return param;
                },
                "string": function (param, element) {
                    return !!$(param, element.form).length;
                },
                "function": function (param, element) {
                    return param(element);
                }
            },

            optional: function (element) {
                return !$.validator.methods.required.call(this, $.trim(element.value), element) && "dependency-mismatch";
            },

            startRequest: function (element) {
                if (!this.pending[element.name]) {
                    this.pendingRequest++;
                    this.pending[element.name] = true;
                }
            },

            stopRequest: function (element, valid) {
                this.pendingRequest--;
                // sometimes synchronization fails, make sure pendingRequest is never < 0
                if (this.pendingRequest < 0)
                    this.pendingRequest = 0;
                delete this.pending[element.name];
                if (valid && this.pendingRequest == 0 && this.formSubmitted && this.form()) {
                    $(this.currentForm).submit();
                    this.formSubmitted = false;
                } else if (!valid && this.pendingRequest == 0 && this.formSubmitted) {
                    $(this.currentForm).triggerHandler("invalid-form", [this]);
                    this.formSubmitted = false;
                }
            },

            previousValue: function (element) {
                return $.data(element, "previousValue") || $.data(element, "previousValue", {
                    old: null,
                    valid: true,
                    message: this.defaultMessage(element, "remote")
                });
            }

        },

        classRuleSettings: {
            required: { required: true },
            email: { email: true },
            url: { url: true },
            date: { date: true },
            dateISO: { dateISO: true },
            dateDE: { dateDE: true },
            number: { number: true },
            numberDE: { numberDE: true },
            digits: { digits: true },
            creditcard: { creditcard: true }
        },

        addClassRules: function (className, rules) {
            className.constructor == String ?
			this.classRuleSettings[className] = rules :
			$.extend(this.classRuleSettings, className);
        },

        classRules: function (element) {
            var rules = {};
            var classes = $(element).attr('class');
            classes && $.each(classes.split(' '), function () {
                if (this in $.validator.classRuleSettings) {
                    $.extend(rules, $.validator.classRuleSettings[this]);
                }
            });
            return rules;
        },

        attributeRules: function (element) {
            var rules = {};
            var $element = $(element);

            for (method in $.validator.methods) {
                var value = $element.attr(method);
                if (value) {
                    rules[method] = value;
                }
            }

            // maxlength may be returned as -1, 2147483647 (IE) and 524288 (safari) for text inputs
            if (rules.maxlength && /-1|2147483647|524288/.test(rules.maxlength)) {
                delete rules.maxlength;
            }

            return rules;
        },

        metadataRules: function (element) {
            if (!$.metadata) return {};

            var meta = $.data(element.form, 'validator').settings.meta;
            return meta ?
			$(element).metadata()[meta] :
			$(element).metadata();
        },

        staticRules: function (element) {
            var rules = {};
            var validator = $.data(element.form, 'validator');
            if (validator.settings.rules) {
                rules = $.validator.normalizeRule(validator.settings.rules[element.name]) || {};
            }
            return rules;
        },

        normalizeRules: function (rules, element) {
            // handle dependency check
            $.each(rules, function (prop, val) {
                // ignore rule when param is explicitly false, eg. required:false
                if (val === false) {
                    delete rules[prop];
                    return;
                }
                if (val.param || val.depends) {
                    var keepRule = true;
                    switch (typeof val.depends) {
                        case "string":
                            keepRule = !!$(val.depends, element.form).length;
                            break;
                        case "function":
                            keepRule = val.depends.call(element, element);
                            break;
                    }
                    if (keepRule) {
                        rules[prop] = val.param !== undefined ? val.param : true;
                    } else {
                        delete rules[prop];
                    }
                }
            });

            // evaluate parameters
            $.each(rules, function (rule, parameter) {
                rules[rule] = $.isFunction(parameter) ? parameter(element) : parameter;
            });

            // clean number parameters
            $.each(['minlength', 'maxlength', 'min', 'max'], function () {
                if (rules[this]) {
                    rules[this] = Number(rules[this]);
                }
            });
            $.each(['rangelength', 'range'], function () {
                if (rules[this]) {
                    rules[this] = [Number(rules[this][0]), Number(rules[this][1])];
                }
            });

            if ($.validator.autoCreateRanges) {
                // auto-create ranges
                if (rules.min && rules.max) {
                    rules.range = [rules.min, rules.max];
                    delete rules.min;
                    delete rules.max;
                }
                if (rules.minlength && rules.maxlength) {
                    rules.rangelength = [rules.minlength, rules.maxlength];
                    delete rules.minlength;
                    delete rules.maxlength;
                }
            }

            // To support custom messages in metadata ignore rule methods titled "messages"
            if (rules.messages) {
                delete rules.messages;
            }

            return rules;
        },

        // Converts a simple string to a {string: true} rule, e.g., "required" to {required:true}
        normalizeRule: function (data) {
            if (typeof data == "string") {
                var transformed = {};
                $.each(data.split(/\s/), function () {
                    transformed[this] = true;
                });
                data = transformed;
            }
            return data;
        },

        // http://docs.jquery.com/Plugins/Validation/Validator/addMethod
        addMethod: function (name, method, message) {
            $.validator.methods[name] = method;
            $.validator.messages[name] = message != undefined ? message : $.validator.messages[name];
            if (method.length < 3) {
                $.validator.addClassRules(name, $.validator.normalizeRule(name));
            }
        },

        methods: {

            // http://docs.jquery.com/Plugins/Validation/Methods/required
            required: function (value, element, param) {
                // check if dependency is met
                if (!this.depend(param, element))
                    return "dependency-mismatch";
                switch (element.nodeName.toLowerCase()) {
                    case 'select':
                        // could be an array for select-multiple or a string, both are fine this way
                        var val = $(element).val();
                        return val && val.length > 0;
                    case 'input':
                        if (this.checkable(element))
                            return this.getLength(value, element) > 0;
                    default:
                        return $.trim(value).length > 0;
                }
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/remote
            remote: function (value, element, param) {
                if (this.optional(element))
                    return "dependency-mismatch";

                var previous = this.previousValue(element);
                if (!this.settings.messages[element.name])
                    this.settings.messages[element.name] = {};
                previous.originalMessage = this.settings.messages[element.name].remote;
                this.settings.messages[element.name].remote = previous.message;

                param = typeof param == "string" && { url: param} || param;

                if (previous.old !== value) {
                    previous.old = value;
                    var validator = this;
                    this.startRequest(element);
                    var data = {};
                    data[element.name] = value;
                    $.ajax($.extend(true, {
                        url: param,
                        mode: "abort",
                        port: "validate" + element.name,
                        dataType: "json",
                        data: data,
                        success: function (response) {
                            validator.settings.messages[element.name].remote = previous.originalMessage;
                            var valid = response === true;
                            if (valid) {
                                var submitted = validator.formSubmitted;
                                validator.prepareElement(element);
                                validator.formSubmitted = submitted;
                                validator.successList.push(element);
                                validator.showErrors();
                            } else {
                                var errors = {};
                                var message = (previous.message = response || validator.defaultMessage(element, "remote"));
                                errors[element.name] = $.isFunction(message) ? message(value) : message;
                                validator.showErrors(errors);
                            }
                            previous.valid = valid;
                            validator.stopRequest(element, valid);
                        }
                    }, param));
                    return "pending";
                } else if (this.pending[element.name]) {
                    return "pending";
                }
                return previous.valid;
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/minlength
            minlength: function (value, element, param) {
                return this.optional(element) || this.getLength($.trim(value), element) >= param;
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/maxlength
            maxlength: function (value, element, param) {
                return this.optional(element) || this.getLength($.trim(value), element) <= param;
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/rangelength
            rangelength: function (value, element, param) {
                var length = this.getLength($.trim(value), element);
                return this.optional(element) || (length >= param[0] && length <= param[1]);
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/min
            min: function (value, element, param) {
                return this.optional(element) || value >= param;
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/max
            max: function (value, element, param) {
                return this.optional(element) || value <= param;
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/range
            range: function (value, element, param) {
                return this.optional(element) || (value >= param[0] && value <= param[1]);
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/email
            email: function (value, element) {
                // contributed by Scott Gonzalez: http://projects.scottsplayground.com/email_address_validation/
                return this.optional(element) || /^((([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+(\.([a-z]|\d|[!#\$%&'\*\+\-\/=\?\^_`{\|}~]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])+)*)|((\x22)((((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(([\x01-\x08\x0b\x0c\x0e-\x1f\x7f]|\x21|[\x23-\x5b]|[\x5d-\x7e]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(\\([\x01-\x09\x0b\x0c\x0d-\x7f]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF]))))*(((\x20|\x09)*(\x0d\x0a))?(\x20|\x09)+)?(\x22)))@((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?$/i.test(value);
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/url
            url: function (value, element) {
                // contributed by Scott Gonzalez: http://projects.scottsplayground.com/iri/
                return this.optional(element) || /^(https?|ftp):\/\/(((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:)*@)?(((\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5])\.(\d|[1-9]\d|1\d\d|2[0-4]\d|25[0-5]))|((([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|\d|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.)+(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])*([a-z]|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])))\.?)(:\d*)?)(\/((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)+(\/(([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)*)*)?)?(\?((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|[\uE000-\uF8FF]|\/|\?)*)?(\#((([a-z]|\d|-|\.|_|~|[\u00A0-\uD7FF\uF900-\uFDCF\uFDF0-\uFFEF])|(%[\da-f]{2})|[!\$&'\(\)\*\+,;=]|:|@)|\/|\?)*)?$/i.test(value);
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/date
            date: function (value, element) {
                return this.optional(element) || !/Invalid|NaN/.test(new Date(value));
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/dateISO
            dateISO: function (value, element) {
                return this.optional(element) || /^\d{4}[\/-]\d{1,2}[\/-]\d{1,2}$/.test(value);
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/number
            number: function (value, element) {
                return this.optional(element) || /^-?(?:\d+|\d{1,3}(?:,\d{3})+)(?:\.\d+)?$/.test(value);
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/digits
            digits: function (value, element) {
                return this.optional(element) || /^\d+$/.test(value);
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/creditcard
            // based on http://en.wikipedia.org/wiki/Luhn
            creditcard: function (value, element) {
                if (this.optional(element))
                    return "dependency-mismatch";
                // accept only digits and dashes
                if (/[^0-9-]+/.test(value))
                    return false;
                var nCheck = 0,
				nDigit = 0,
				bEven = false;

                value = value.replace(/\D/g, "");

                for (var n = value.length - 1; n >= 0; n--) {
                    var cDigit = value.charAt(n);
                    var nDigit = parseInt(cDigit, 10);
                    if (bEven) {
                        if ((nDigit *= 2) > 9)
                            nDigit -= 9;
                    }
                    nCheck += nDigit;
                    bEven = !bEven;
                }

                return (nCheck % 10) == 0;
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/accept
            accept: function (value, element, param) {
                param = typeof param == "string" ? param.replace(/,/g, '|') : "png|jpe?g|gif";
                return this.optional(element) || value.match(new RegExp(".(" + param + ")$", "i"));
            },

            // http://docs.jquery.com/Plugins/Validation/Methods/equalTo
            equalTo: function (value, element, param) {
                // bind to the blur event of the target in order to revalidate whenever the target field is updated
                // TODO find a way to bind the event just once, avoiding the unbind-rebind overhead
                var target = $(param).unbind(".validate-equalTo").bind("blur.validate-equalTo", function () {
                    $(element).valid();
                });
                return value == target.val();
            }

        }

    });

    // deprecated, use $.validator.format instead
    $.format = $.validator.format;

})(jQuery);

// ajax mode: abort
// usage: $.ajax({ mode: "abort"[, port: "uniqueport"]});
// if mode:"abort" is used, the previous request on that port (port can be undefined) is aborted via XMLHttpRequest.abort() 
; (function ($) {
    var ajax = $.ajax;
    var pendingRequests = {};
    $.ajax = function (settings) {
        // create settings for compatibility with ajaxSetup
        settings = $.extend(settings, $.extend({}, $.ajaxSettings, settings));
        var port = settings.port;
        if (settings.mode == "abort") {
            if (pendingRequests[port]) {
                pendingRequests[port].abort();
            }
            return (pendingRequests[port] = ajax.apply(this, arguments));
        }
        return ajax.apply(this, arguments);
    };
})(jQuery);

// provides cross-browser focusin and focusout events
// IE has native support, in other browsers, use event caputuring (neither bubbles)

// provides delegate(type: String, delegate: Selector, handler: Callback) plugin for easier event delegation
// handler is only called when $(event.target).is(delegate), in the scope of the jquery-object for event.target 
; (function ($) {
    // only implement if not provided by jQuery core (since 1.4)
    // TODO verify if jQuery 1.4's implementation is compatible with older jQuery special-event APIs
    if (!jQuery.event.special.focusin && !jQuery.event.special.focusout && document.addEventListener) {
        $.each({
            focus: 'focusin',
            blur: 'focusout'
        }, function (original, fix) {
            $.event.special[fix] = {
                setup: function () {
                    this.addEventListener(original, handler, true);
                },
                teardown: function () {
                    this.removeEventListener(original, handler, true);
                },
                handler: function (e) {
                    arguments[0] = $.event.fix(e);
                    arguments[0].type = fix;
                    return $.event.handle.apply(this, arguments);
                }
            };
            function handler(e) {
                e = $.event.fix(e);
                e.type = fix;
                return $.event.handle.call(this, e);
            }
        });
    };
    $.extend($.fn, {
        validateDelegate: function (delegate, type, handler) {
            return this.bind(type, function (event) {
                var target = $(event.target);
                if (target.is(delegate)) {
                    return handler.apply(target, arguments);
                }
            });
        }
    });
})(jQuery);
// Common initialization
var xVal = xVal || {};
xVal.Plugins = xVal.Plugins || {};
xVal.Messages = xVal.Messages || {};
xVal.AttachValidator = function(elementPrefix, rulesConfig, options, pluginName) {
    if (pluginName != null)
        this.Plugins[pluginName].AttachValidator(elementPrefix, rulesConfig, options);
    else
        for (var key in this.Plugins) {
            this.Plugins[key].AttachValidator(elementPrefix, rulesConfig, options);
            return;
        }
};

// xVal.jquery.validate.js
// An xVal plugin to enable support for jQuery Validate
// http://xval.codeplex.com/
// (c) 2009 Steven Sanderson
// License: Microsoft Public License (Ms-PL) (http://www.opensource.org/licenses/ms-pl.html)

(function($) {
	xVal.Plugins["jquery.validate"] = {
		AttachValidator: function(elementPrefix, rulesConfig, options) {
			var self = this;
			self._ensureCustomFunctionsRegistered();
			$(function() {
				self._ensureValidationSummaryContainerExistsIfRequired(options);

				for (var i = 0; i < rulesConfig.Fields.length; i++) {
					var fieldName = rulesConfig.Fields[i].FieldName;
					var fieldRules = rulesConfig.Fields[i].FieldRules;

					// Is there a matching DOM element?
					var elemId = self._makeAspNetMvcHtmlHelperID((elementPrefix ? elementPrefix + "." : "") + fieldName);
					var elem = document.getElementById(elemId);
					if (elem) {
						for (var j = 0; j < fieldRules.length; j++) {
							var rule = fieldRules[j];
							if (rule != null) {
								var ruleName = rule.RuleName;
								var ruleParams = rule.RuleParameters;
								var errorText = (typeof (rule.Message) == 'undefined' ? null : rule.Message);
								self._attachRuleToDOMElement(ruleName, ruleParams, errorText, $(elem), elementPrefix, options);
							}
						}
					}
				}
			});
		},

		_makeAspNetMvcHtmlHelperID: function(fullyQualifiedModelName) {
			// If you have changed HtmlHelper.IdAttributeDotReplacement from "_" to something else, then you must also change the following line to match
			return fullyQualifiedModelName.replace(/\./g, "_");
		},

		_attachRuleToDOMElement: function(ruleName, ruleParams, errorText, element, elementPrefix, options) {
			var parentForm = element.parents("form");
			if (parentForm.length != 1)
				alert("Error: Element " + element.attr("id") + " is not in a form");
			this._ensureFormIsMarkedForValidation($(parentForm[0]), options);
			this._associateNearbyValidationMessageSpanWithElement(element);

			var options = {};

			switch (ruleName) {
				case "Required":
					options.required = true;
					options.messages = { required: errorText || xVal.Messages.Required };
					break;

				case "Range":
					if (ruleParams.Type == "string") {
						options.xVal_stringRange = [ruleParams.Min, ruleParams.Max];
						if (errorText != null) options.messages = { xVal_stringRange: $.format(errorText) };
					}
					else if (ruleParams.Type == "datetime") {
						var minDate, maxDate;
						if (typeof (ruleParams.MinYear) != 'undefined')
							minDate = new Date(ruleParams.MinYear, ruleParams.MinMonth - 1, ruleParams.MinDay, ruleParams.MinHour, ruleParams.MinMinute, ruleParams.MinSecond);
						if (typeof (ruleParams.MaxYear) != 'undefined')
							maxDate = new Date(ruleParams.MaxYear, ruleParams.MaxMonth - 1, ruleParams.MaxDay, ruleParams.MaxHour, ruleParams.MaxMinute, ruleParams.MaxSecond);
						options.xVal_dateRange = [minDate, maxDate];
						if (errorText != null) options.messages = { xVal_dateRange: $.format(errorText) };
					}
					else if (typeof (ruleParams.Min) == 'undefined') {
						options.max = ruleParams.Max;
						errorText = errorText || xVal.Messages.Range_Numeric_Max;
						if (errorText != null) options.messages = { max: $.format(errorText) };
					}
					else if (typeof (ruleParams.Max) == 'undefined') {
						options.min = ruleParams.Min;
						errorText = errorText || xVal.Messages.Range_Numeric_Min;
						if (errorText != null) options.messages = { min: $.format(errorText) };
					}
					else {
						options.range = [ruleParams.Min, ruleParams.Max];
						errorText = errorText || xVal.Messages.Range_Numeric_MinMax;
						if (errorText != null) options.messages = { range: $.format(errorText) };
					}

					break;

				case "StringLength":
					if (typeof (ruleParams.MinLength) == 'undefined') {
						options.maxlength = ruleParams.MaxLength;
						errorText = errorText || xVal.Messages.StringLength_Max;
						if (errorText != null) options.messages = { maxlength: $.format(errorText) };
					}
					else if (typeof (ruleParams.MaxLength) == 'undefined') {
						options.minlength = ruleParams.MinLength;
						errorText = errorText || xVal.Messages.StringLength_Min;
						if (errorText != null) options.messages = { minlength: $.format(errorText) };
					}
					else {
						options.rangelength = [ruleParams.MinLength, ruleParams.MaxLength];
						errorText = errorText || xVal.Messages.StringLength_MinMax;
						if (errorText != null) options.messages = { rangelength: $.format(errorText) };
					}
					break;

				case "DataType":
					switch (ruleParams.Type) {
						case "EmailAddress":
							options.email = true;
							options.messages = { email: errorText || xVal.Messages.DataType_EmailAddress };
							break;
						case "Integer":
							options.xVal_regex = ["^\\-?\\d+$", ""];
							options.messages = { xVal_regex: errorText || xVal.Messages.DataType_Integer || "Please enter a whole number." };
							break;
						case "Decimal":
							options.number = true;
							options.messages = { number: errorText || xVal.Messages.DataType_Decimal };
							break;
						case "Date":
							options.date = true;
							options.messages = { date: errorText || xVal.Messages.DataType_Date };
							break;
						case "DateTime":
							options.xVal_regex = ["^\\d{1,2}/\\d{1,2}/(\\d{2}|\\d{4})\\s+\\d{1,2}\\:\\d{2}(\\:\\d{2})?$", ""];
							options.messages = { xVal_regex: errorText || xVal.Messages.DataType_DateTime || "Please enter a valid date and time." };
							break;
						case "Currency":
							options.xVal_regex = ["^\\D?\\s?([0-9]{1,3},([0-9]{3},)*[0-9]{3}|[0-9]+)(.[0-9][0-9])?$", ""];
							options.messages = { xVal_regex: errorText || xVal.Messages.DataType_Currency || "Please enter a currency value." };
							break;
						case "CreditCardLuhn":
							options.xVal_creditCardLuhn = true;
							if (errorText != null) options.messages = { xVal_creditCardLuhn: errorText };
							break;
					}
					break;

				case "RegEx":
					options.xVal_regex = [ruleParams.Pattern, ruleParams.Options];
					if (errorText != null) options.messages = { xVal_regex: errorText };
					break;

				case "Comparison":
					var elemToCompareId = this._makeAspNetMvcHtmlHelperID((elementPrefix ? elementPrefix + "." : "") + ruleParams.PropertyToCompare);
					var elemToCompare = document.getElementById(elemToCompareId);
					if (elemToCompare != null) {
						options.xVal_comparison = [ruleParams.PropertyToCompare, elemToCompare, ruleParams.ComparisonOperator];
						if (errorText != null) options.messages = { xVal_comparison: errorText };
					}
					break;

				case "Remote":
					var dataAccessor = {};
					parentForm.find("input[name], textarea[name], select[name]").each(function() {
						var input = this;
						dataAccessor[input.name] = function() { return $(input).val(); };
					});
					options.remote = {
						url: ruleParams.url,
						data: dataAccessor,
						type: 'post'
					};
					break;

				case "Custom":
					var ruleFunction = this._parseAsFunctionWithWarnings(ruleParams.Function);
					if (ruleFunction != null) {
						var customFunctionName = this._registerCustomValidationFunction(ruleFunction);
						var evaluatedParams = ruleParams.Parameters == "null" ? null : eval("(" + ruleParams.Parameters + ")");
						options[customFunctionName] = evaluatedParams || true;
						options.messages = [];
						options.messages[customFunctionName] = errorText;
					}
					break;
			}

			element.rules("add", options);
		},

		_parseAsFunctionWithWarnings: function(functionString) {
			var result;
			try { result = eval("(" + functionString + ")") }
			catch (ex) {
				alert("Custom rule error: Could not find or could not parse the function '" + functionString + "'");
				return null;
			}
			if (typeof (result) != 'function') {
				alert("Custom rule error: The JavaScript object '" + functionString + "' is not a function.");
				return null;
			}
			return result;
		},

		_associateNearbyValidationMessageSpanWithElement: function(element) {
			// If there's a <p class='error'> soon after, it's probably supposed to display the error message
			// jquery.validation goes looking for an attribute called "htmlfor" as follows
			var nearbyMessages = element.nextAll("span.validationError");
			if (nearbyMessages.length > 0) {
				$(nearbyMessages[0]).attr("generated", "true")
                                    .attr("htmlfor", element.attr("id"));
			}
		},

		_ensureFormIsMarkedForValidation: function(formElement, options) {
			if (!formElement.data("isMarkedForValidation")) {
				formElement.data("isMarkedForValidation", true);
				var validationOptions = {
					errorClass: "validationError",
					errorElement: "span",
					highlight: function(element) {
						$(element).addClass("validationError").parents('p:first').addClass('validationError');
					},
					unhighlight: function(element) {
						$(element).removeClass("validationError").parents('p:first').removeClass('validationError');
					}
				};
				if (options.ValidationSummary) {
					validationOptions.wrapper = "li";
					validationOptions.errorLabelContainer = "#" + options.ValidationSummary.ElementID + " ul:first";
				}
				var validator = formElement.validate(validationOptions);
				if (options.ValidationSummary)
					this._modifyJQueryValidationElementHidingBehaviourToSupportValidationSummary(validator, options);
				else
					this._modifyJQueryValidationElementHidingBehaviourToSupportGlobalValidationMessage(validator, options);
			}
		},

		_registerCustomValidationFunction: function(evalFn) {
			jQuery.validator.xValCustomFunctionCount = (jQuery.validator.xValCustomFunctionCount || 0) + 1;
			var functionName = "xVal_customFunction_" + jQuery.validator.xValCustomFunctionCount;
			jQuery.validator.addMethod(functionName, function(value, element, params) {
				if (this.optional(element))
					return true;
				return evalFn(value, element, params);
			});
			return functionName;
		},

		_ensureCustomFunctionsRegistered: function() {
			if (!jQuery.validator.xValFunctionsRegistered) {
				jQuery.validator.xValFunctionsRegistered = true;

				jQuery.validator.addMethod("xVal_stringRange", function(value, element, params) {
					if (this.optional(element)) return true;
					if (params[0] != null)
						if (value < params[0]) return false;
					if (params[1] != null)
						if (value > params[1]) return false;
					return true;
				}, function(params) {
					if ((params[0] != null) && (params[1] != null))
						return $.format(xVal.Messages.Range_String_MinMax || "Please enter a value alphabetically between '{0}' and '{1}'.", params[0], params[1]);
					else if (params[0] != null)
						return $.format(xVal.Messages.Range_String_Min || "Please enter a value not alphabetically before '{0}'.", params[0]);
					else
						return $.format(xVal.Messages.Range_String_Max || "Please enter a value not alphabetically after '{0}'.", params[1]);
				});

				jQuery.validator.addMethod("xVal_dateRange", function(value, element, params) {
					if (this.optional(element)) return true;

					var parsedValue = Date.parse(value);
					if (isNaN(parsedValue))
						return false;
					else
						parsedValue = new Date(parsedValue);
					if (params[0] != null)
						if (parsedValue < params[0]) return false;
					if (params[1] != null)
						if (parsedValue > params[1]) return false;
					return true;
				}, function(params, elem) {
					if (isNaN(Date.parse(elem.value)))
						return xVal.Messages.DataType_Date || "Please enter a valid date in yyyy/mm/dd format.";
					var formatDate = function(date) {
						var result = date.getFullYear() + "-" + (date.getMonth() + 1) + "-" + date.getDate();
						if (date.getHours() + date.getMinutes() + date.getSeconds() != 0)
							result += " " + date.getHours() + ":" + date.getMinutes() + ":" + date.getSeconds();
						return result.replace(/\b(\d)\b/g, '0$1');
					};
					if ((params[0] != null) && (params[1] != null))
						return $.format(xVal.Messages.Range_DateTime_MinMax || "Please enter a date between {0} and {1}.", formatDate(params[0]), formatDate(params[1]));
					else if (params[0] != null)
						return $.format(xVal.Messages.Range_DateTime_Min || "Please enter a date no earlier than {0}.", formatDate(params[0]));
					else
						return $.format(xVal.Messages.Range_DateTime_Max || "Please enter a date no later than {0}.", formatDate(params[1]));
				});

				jQuery.validator.addMethod("xVal_regex", function(value, element, params) {
					if (this.optional(element)) return true;
					var pattern = params[0];
					var options = params[1];
					var regex = new RegExp(pattern, options);
					return regex.test(value);
				}, function(params) {
					return xVal.Messages.Regex || "This value is invalid."; // Pity we can't be more descriptive
				});

				jQuery.validator.addMethod("xVal_creditCardLuhn", function(value, element, params) {
					if (this.optional(element)) return true;
					value = value.replace(/\D/g, "");
					if (value == "") return false;
					var sum = 0;
					for (var i = value.length - 2; i >= 0; i -= 2)
						sum += Array(0, 2, 4, 6, 8, 1, 3, 5, 7, 9)[parseInt(value.charAt(i), 10)];
					for (var i = value.length - 1; i >= 0; i -= 2)
						sum += parseInt(value.charAt(i), 10);
					return (sum % 10) == 0;
				}, function(params) {
					return xVal.Messages.DataType_CreditCardLuhn || "Please enter a valid credit card number.";
				});

				jQuery.validator.addMethod("xVal_comparison", function(value, element, params) {
					if (this.optional(element)) return true;
					var elemToCompare = params[1];
					var comparisonOperator = params[2];
					switch (comparisonOperator) {
						case "Equals": return value == elemToCompare.value;
						case "DoesNotEqual": return value != elemToCompare.value;
					}
					return true; // Ignore unrecognized operator
				}, function(params) {
					var propertyToCompareName = params[0];
					var comparisonOperator = params[2];
					switch (comparisonOperator) {
						case "Equals": return $.format(xVal.Messages.Comparison_Equals || "This value must be the same as {0}.", propertyToCompareName);
						case "DoesNotEqual": return $.format(xVal.Messages.Comparison_DoesNotEqual || "This value must be different from {0}.", propertyToCompareName);
					}
				});

				$.expr[":"].displayableValidationSummaryMessage = function(object) {
					var span = $(object).find("span:first");
					if (span.length == 0)
						return true;
					return !(span.css("display") === "none") && !span.is(":empty");
				};
			}
		},

		_ensureValidationSummaryContainerExistsIfRequired: function(options) {
			// If we're using a validation summary, make sure the container contains an UL
			// (If there were no server-generated errors, there won't be until we create one)
			if (options.ValidationSummary) {
				var validationSummaryContainer = $("#" + options.ValidationSummary.ElementID);
				if (validationSummaryContainer.length == 0)
					alert("Cannot find validation summary element \"" + options.ValidationSummary.ElementID + "\". Make sure you've put an element with this ID into your HTML document.");
				if (!validationSummaryContainer.is(":has(ul)")) {
					validationSummaryContainer.append($("<span class='validation-summary-errors' />").text(options.ValidationSummary.HeaderMessage))
                                              .append($("<ul />"))
                                              .hide();
				}
			}
		},

		_modifyJQueryValidationElementHidingBehaviourToSupportValidationSummary: function(validator, options) {
			// Intercept the hideErrors() and showErrors() methods. Pity there isn't a proper API for this.
			var originalHideErrorsMethod = validator.hideErrors;
			var originalShowErrorsMethod = validator.showErrors;
			validator.hideErrors = function() {
				this.toHide = this.toHide.not("ul"); // Don't ever hide ULs, because these might still contain server-generated error messages
				originalHideErrorsMethod.apply(this, arguments);
				// If the summary container contains no displayable messages, hide the whole thing
				$("#" + options.ValidationSummary.ElementID + ":not(:has(li:displayableValidationSummaryMessage))").hide();
			};
			validator.showErrors = function() {
				originalShowErrorsMethod.apply(this, arguments);
				// If the summary container contains any displayable messages, show it
				$("#" + options.ValidationSummary.ElementID + ":has(li:displayableValidationSummaryMessage)").show();
			};
		},

		_modifyJQueryValidationElementHidingBehaviourToSupportGlobalValidationMessage: function(validator, options) {
			var originalHideErrorsMethod = validator.hideErrors;
			var originalShowErrorsMethod = validator.showErrors;
			validator.hideErrors = function() {
				originalHideErrorsMethod.apply(this, arguments);
				if ($('.validationError:visible', validator.errorContext).length == 0)
					$('#validationErrorMessage').hide();
			};
			validator.showErrors = function() {
				originalShowErrorsMethod.apply(this, arguments);
				if ($('.validationError:visible', validator.errorContext).length > 0)
					$('#validationErrorMessage').show();
			};
		}
	};
})(jQuery);

