
/*
 * jQuery UI Accordion 1.6
 * 
 * Copyright (c) 2007 Jörn Zaefferer
 *
 * http://docs.jquery.com/UI/Accordion
 *
 * Dual licensed under the MIT and GPL licenses:
 *   http://www.opensource.org/licenses/mit-license.php
 *   http://www.gnu.org/licenses/gpl.html
 *
 * Revision: $Id: jquery.accordion.js 4876 2008-03-08 11:49:04Z joern.zaefferer $
 *
 */

;(function($) {
	
	// If the UI scope is not available, add it
	$.ui = $.ui || {};

	$.fn.extend({
		accordion: function(options, data) {
			var args = Array.prototype.slice.call(arguments, 1);

			return this.each(function() {
				if (typeof options == "string") {
					var accordion = $.data(this, "ui-accordion");
					accordion[options].apply(accordion, args);
				// INIT with optional options
				} else if (!$(this).is(".ui-accordion"))
					$.data(this, "ui-accordion", new $.ui.accordion(this, options));
			});
		},
		// deprecated, use accordion("activate", index) instead
		activate: function(index) {
			return this.accordion("activate", index);
		}
	});

	$.ui.accordion = function(container, options) {
		
		// setup configuration
		this.options = options = $.extend({}, $.ui.accordion.defaults, options);
		this.element = container;
		
		$(container).addClass("ui-accordion");
		
		if ( options.navigation ) {
			var current = $(container).find("a").filter(options.navigationFilter);
			if ( current.length ) {
				if ( current.filter(options.header).length ) {
					options.active = current;
				} else {
					options.active = current.parent().parent().prev();
					current.addClass("current");
				}
			}
		}
		
		// calculate active if not specified, using the first header
		options.headers = $(container).find(options.header);
		options.active = findActive(options.headers, options.active);

		if ( options.fillSpace ) {
			var maxHeight = $(container).parent().height();
			options.headers.each(function() {
				maxHeight -= $(this).outerHeight();
			});
			var maxPadding = 0;
			options.headers.next().each(function() {
				maxPadding = Math.max(maxPadding, $(this).innerHeight() - $(this).height());
			}).height(maxHeight - maxPadding);
		} else if ( options.autoheight ) {
			var maxHeight = 0;
			options.headers.next().each(function() {
				maxHeight = Math.max(maxHeight, $(this).outerHeight());
			}).height(maxHeight);
		}

		options.headers
			.not(options.active || "")
			.next()
			.hide();
		options.active.parent().andSelf().addClass(options.selectedClass);
		
		if (options.event)
			$(container).bind((options.event) + ".ui-accordion", clickHandler);
	};

	$.ui.accordion.prototype = {
		activate: function(index) {
			// call clickHandler with custom event
			clickHandler.call(this.element, {
				target: findActive( this.options.headers, index )[0]
			});
		},
		
		enable: function() {
			this.options.disabled = false;
		},
		disable: function() {
			this.options.disabled = true;
		},
		destroy: function() {
			this.options.headers.next().css("display", "");
			if ( this.options.fillSpace || this.options.autoheight ) {
				this.options.headers.next().css("height", "");
			}
			$.removeData(this.element, "ui-accordion");
			$(this.element).removeClass("ui-accordion").unbind(".ui-accordion");
		}
	}

	function scopeCallback(callback, scope) {
		return function() {
			return callback.apply(scope, arguments);
		};
	}

	function completed(cancel) {
		// if removed while animated data can be empty
		if (!$.data(this, "ui-accordion"))
			return;
		var instance = $.data(this, "ui-accordion");
		var options = instance.options;
		options.running = cancel ? 0 : --options.running;
		if ( options.running )
			return;
		if ( options.clearStyle ) {
			options.toShow.add(options.toHide).css({
				height: "",
				overflow: ""
			});
		}
		$(this).triggerHandler("change.ui-accordion", [options.data], options.change);
	}

	function toggle(toShow, toHide, data, clickedActive, down) {
		var options = $.data(this, "ui-accordion").options;
		options.toShow = toShow;
		options.toHide = toHide;
		options.data = data;
		var complete = scopeCallback(completed, this);
		
		// count elements to animate
		options.running = toHide.size() == 0 ? toShow.size() : toHide.size();
		
		if ( options.animated ) {
			if ( !options.alwaysOpen && clickedActive ) {
				$.ui.accordion.animations[options.animated]({
					toShow: jQuery([]),
					toHide: toHide,
					complete: complete,
					down: down,
					autoheight: options.autoheight
				});
			} else {
				$.ui.accordion.animations[options.animated]({
					toShow: toShow,
					toHide: toHide,
					complete: complete,
					down: down,
					autoheight: options.autoheight
				});
			}
		} else {
			if ( !options.alwaysOpen && clickedActive ) {
				toShow.toggle();
			} else {
				toHide.hide();
				toShow.show();
			}
			complete(true);
		}
	}

	function clickHandler(event) {
		var options = $.data(this, "ui-accordion").options;
		if (options.disabled)
			return false;
		
		// called only when using activate(false) to close all parts programmatically
		if ( !event.target && !options.alwaysOpen ) {
			options.active.parent().andSelf().toggleClass(options.selectedClass);
			var toHide = options.active.next(),
				data = {
					instance: this,
					options: options,
					newHeader: jQuery([]),
					oldHeader: options.active,
					newContent: jQuery([]),
					oldContent: toHide
				},
				toShow = options.active = $([]);
			toggle.call(this, toShow, toHide, data );
			return false;
		}
		// get the click target
		var clicked = $(event.target);
		
		// due to the event delegation model, we have to check if one
		// of the parent elements is our actual header, and find that
		if ( clicked.parents(options.header).length )
			while ( !clicked.is(options.header) )
				clicked = clicked.parent();
		
		var clickedActive = clicked[0] == options.active[0];
		
		// if animations are still active, or the active header is the target, ignore click
		if (options.running || (options.alwaysOpen && clickedActive))
			return false;
		if (!clicked.is(options.header))
			return;

		// switch classes
		options.active.parent().andSelf().toggleClass(options.selectedClass);
		if ( !clickedActive ) {
			clicked.parent().andSelf().addClass(options.selectedClass);
		}

		// find elements to show and hide
		var toShow = clicked.next(),
			toHide = options.active.next(),
			//data = [clicked, options.active, toShow, toHide],
			data = {
				instance: this,
				options: options,
				newHeader: clicked,
				oldHeader: options.active,
				newContent: toShow,
				oldContent: toHide
			},
			down = options.headers.index( options.active[0] ) > options.headers.index( clicked[0] );
		
		options.active = clickedActive ? $([]) : clicked;
		toggle.call(this, toShow, toHide, data, clickedActive, down );

		return false;
	};

	function findActive(headers, selector) {
		return selector != undefined
			? typeof selector == "number"
				? headers.filter(":eq(" + selector + ")")
				: headers.not(headers.not(selector))
			: selector === false
				? $([])
				: headers.filter(":eq(0)");
	}

	$.extend($.ui.accordion, {
		defaults: {
			selectedClass: "selected",
			alwaysOpen: true,
			animated: 'slide',
			event: "click",
			header: "a",
			autoheight: true,
			running: 0,
			navigationFilter: function() {
				return this.href.toLowerCase() == location.href.toLowerCase();
			}
		},
		animations: {
			slide: function(options, additions) {
				options = $.extend({
					easing: "swing",
					duration: 300
				}, options, additions);
				if ( !options.toHide.size() ) {
					options.toShow.animate({height: "show"}, options);
					return;
				}
				var hideHeight = options.toHide.height(),
					showHeight = options.toShow.height(),
					difference = showHeight / hideHeight;
				options.toShow.css({ height: 0, overflow: 'hidden' }).show();
				options.toHide.filter(":hidden").each(options.complete).end().filter(":visible").animate({height:"hide"},{
					step: function(now) {
						var current = (hideHeight - now) * difference;
						if ($.browser.msie || $.browser.opera) {
							current = Math.ceil(current);
						}
						options.toShow.height( current );
					},
					duration: options.duration,
					easing: options.easing,
					complete: function() {
						if ( !options.autoheight ) {
							options.toShow.css("height", "auto");
						}
						options.complete();
					}
				});
			},
			bounceslide: function(options) {
				this.slide(options, {
					easing: options.down ? "bounceout" : "swing",
					duration: options.down ? 1000 : 200
				});
			},
			easeslide: function(options) {
				this.slide(options, {
					easing: "easeinout",
					duration: 700
				})
			}
		}
	});

})(jQuery);

$(function(){
	$('#accordion').accordion({ 
		header: 'h3', 
		active: false, 
		alwaysOpen: false, 
		animated: false, 
		autoheight: false 
	});
});