/* http://keith-wood.name/svg.html
   jQuery DOM compatibility for jQuery SVG v1.5.0.
   Written by Keith Wood (kbwood{at}iinet.com.au) April 2009.
   Available under the MIT (http://keith-wood.name/licence.html) license. 
   Please attribute the author if you use it. */

(function($) { // Hide scope, no $ conflict
 
var rclass = /[\t\r\n]/g,
	rspace = /\s+/,
	rwhitespace = "[\\x20\\t\\r\\n\\f]";

/** Retrieve the element classes.
	@private
	@param elem {Element} The element to examine.
	@return {string} The class names. */
function getClassNames(elem) {
	return (!$.svg.isSVGElem(elem) ? elem.className :
		(elem.className ? elem.className.baseVal : elem.getAttribute('class'))) || '';
}

/** Set the element classes.
	@private
	@param elem {Element} The element to update.
	@param classes {string} The new class names. */
function setClassNames(elem, classes) {
	(elem.className ? elem.className.baseVal = classes : elem.setAttribute('class',  classes));
}

/** Support adding class names to SVG nodes.
	@param classNames {string} The classes to add. */
$.fn.addClass = function(origAddClass) {
	return function(classNames) {
		if ($.isFunction(classNames)) {
			return this.each(function(i) {
				$(this).addClass(classNames.call(this, i, getClassNames(this)));
			});
		}
		var origArgs = arguments;
		classNames = classNames || '';
		return this.each(function() {
			if ($.svg.isSVGElem(this)) {
				var node = this;
				$.each(classNames.split(/\s+/), function(i, className) {
					var classes = getClassNames(node);
					if ($.inArray(className, classes.split(/\s+/)) === -1) {
						setClassNames(node,  classes += (classes ? ' ' : '') + className);
					}
				});
			}
			else {
				origAddClass.apply($(this), origArgs);
			}
		});
	};
}($.fn.addClass);

/** Support removing class names from SVG nodes.
	@param classNames {string} The classes to remove. */
$.fn.removeClass = function(origRemoveClass) {
	return function(classNames) {
		if ($.isFunction(classNames)) {
			return this.each(function(i) {
				$(this).removeClass(classNames.call(this, i, getClassNames(this)));
			});
		}
		var origArgs = arguments;
		classNames = classNames || '';
		return this.each(function() {
			if ($.svg.isSVGElem(this)) {
				var node = this;
				$.each(classNames.split(/\s+/), function(i, className) {
					var classes = getClassNames(node);
					classes = $.grep(classes.split(/\s+/), function(n, i) { return n !== className; }).join(' ');
					setClassNames(node, classes);
				});
			}
			else {
				origRemoveClass.apply($(this), origArgs);
			}
		});
	};
}($.fn.removeClass);

/** Support toggling class names on SVG nodes.
	@param classNames {string} The classes to toggle. */
$.fn.toggleClass = function(origToggleClass) {
	return function(classNames, state) {
		if ($.isFunction(classNames)) {
			return this.each(function(i) {
				$(this).toggleClass(classNames.call(this, i, getClassNames(this), state), state);
			});
		}
		var origArgs = arguments;
		var hasState = (typeof state === 'boolean');
		return this.each(function() {
			if ($.svg.isSVGElem(this)) {
				if (typeof classNames === 'string') {
					var node = $(this);
					$.each(classNames.split(/\s+/), function(i, className) {
						if (!hasState) {
							state = !node.hasClass(className);
						}
						node[(state ? 'add' : 'remove') + 'Class'](className);
					});
				}
				else {
					var classes = getClassNames(this);
					if (classes) {
						$._data(this, '__className__', classes); // store className if set
					}
					// toggle whole className
					setClassNames(this, classes || classNames === false ? '' : $._data(this, '__className__') || '');
				}
			}
			else {
				origToggleClass.apply($(this), origArgs);
			}
		});
	};
}($.fn.toggleClass);

/** Support checking class names on SVG nodes.
	@param className {string} The class to check.
	@return {boolean} <code>true</code> if this class is present, <code>false</code> if not. */
$.fn.hasClass = function(origHasClass) {
	return function(className) {
		className = className || '';
		var found = false;
		this.each(function() {
			if ($.svg.isSVGElem(this)) {
				found = ($.inArray(className, getClassNames(this).split(/\s+/)) > -1);
			}
			else {
				found = (origHasClass.apply($(this), [className]));
			}
			return !found;
		});
		return found;
	};
}($.fn.hasClass);

/** Support attributes on SVG nodes.
	@param name {string} The attribute name.
	@param [value] {any} The new attribute value.
	@param type {boolean} Internal flag.
	@return {any} If an attribute value is requested. */
$.fn.attr = function(origAttr) {
	return function(name, value, type) {
		if (typeof name === 'string' && value === undefined) { // Return attribute value
			var val = origAttr.apply(this, arguments);
			if (val && val.baseVal && val.baseVal.numberOfItems != null) { // Multiple values
				value = '';
				val = val.baseVal;
				if (name === 'transform') {
					for (var i = 0; i < val.numberOfItems; i++) {
						var item = val.getItem(i);
						switch (item.type) {
							case 1: value += ' matrix(' + item.matrix.a + ',' + item.matrix.b + ',' +
										item.matrix.c + ',' + item.matrix.d + ',' +
										item.matrix.e + ',' + item.matrix.f + ')';
									break;
							case 2: value += ' translate(' + item.matrix.e + ',' + item.matrix.f + ')'; break;
							case 3: value += ' scale(' + item.matrix.a + ',' + item.matrix.d + ')'; break;
							case 4: value += ' rotate(' + item.angle + ')'; break; // Doesn't handle new origin
							case 5: value += ' skewX(' + item.angle + ')'; break;
							case 6: value += ' skewY(' + item.angle + ')'; break;
						}
					}
					val = value.substring(1);
				}
				else {
					val = val.getItem(0).valueAsString;
				}
			}
			return (val && val.baseVal ? val.baseVal.valueAsString : val);
		}

		var options = name;
		if (typeof name === 'string') {
			options = {};
			options[name] = value;
		}
		if ($.isFunction(value)) {
			return $(this).each(function(i) {
				$(this).attr(name, value.call(this, i, $(this).attr(name)));
			});
		}
		var origArgs = arguments;
		return $(this).each(function() {
			if ($.svg.isSVGElem(this)) {
				for (var n in options) {
					(type ? this.style[n] = options[n] : this.setAttribute(n, options[n]));
				}
			}
			else {
				origAttr.apply($(this), origArgs);
			}
		});
	};
}($.fn.attr);

/** Support removing attributes on SVG nodes.
	@param names {string} The names of the attributes to remove. */
$.fn.removeAttr = function(origRemoveAttr) {
	return function(names) {
		var origArgs = arguments;
		return this.each(function() {
			if ($.svg.isSVGElem(this)) {
				var node = this;
				$.each(names.split(/\s+/), function(i, name) {
					(node[name] && node[name].baseVal ? node[name].baseVal.value = null : node.removeAttribute(name));
				});
			}
			else {
				origRemoveAttr.apply($(this), origArgs);
			}
		});
	};
}($.fn.removeAttr);

/* Add numeric only properties. */
$.extend($.cssNumber, {
	'stopOpacity': true,
	'strokeMitrelimit': true,
	'strokeOpacity': true
});

/* Support retrieving CSS/attribute values on SVG nodes. */
if ($.cssProps) {
	$.css = function(origCSS) {
		return function(elem, name, numeric, extra) {
			var value = (name.match(/^svg.*/) ? $(elem).attr($.cssProps[name] || name) : '');
			return value || origCSS(elem, name, numeric, extra);
		};
	}($.css);
}

})(jQuery);