// SprySlidingPanels.js - version 0.5 - Spry Pre-Release 1.6.1//// Copyright (c) 2006. Adobe Systems Incorporated.// All rights reserved.//// Redistribution and use in source and binary forms, with or without// modification, are permitted provided that the following conditions are met:////   * Redistributions of source code must retain the above copyright notice,//     this list of conditions and the following disclaimer.//   * Redistributions in binary form must reproduce the above copyright notice,//     this list of conditions and the following disclaimer in the documentation//     and/or other materials provided with the distribution.//   * Neither the name of Adobe Systems Incorporated nor the names of its//     contributors may be used to endorse or promote products derived from this//     software without specific prior written permission.//// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE// POSSIBILITY OF SUCH DAMAGE.var Spry;if (!Spry) Spry = {};if (!Spry.Widget) Spry.Widget = {};Spry.Widget.SlidingPanels = function(element, opts){	this.element = this.getElement(element);	this.enableAnimation = true;	this.currentPanel = null;	this.enableKeyboardNavigation = true;	this.hasFocus = false;	this.previousPanelKeyCode = Spry.Widget.SlidingPanels.KEY_LEFT;	this.nextPanelKeyCode = Spry.Widget.SlidingPanels.KEY_RIGHT;	this.currentPanelClass = "SlidingPanelsCurrentPanel";	this.focusedClass = "SlidingPanelsFocused";	this.animatingClass = "SlidingPanelsAnimating";	Spry.Widget.SlidingPanels.setOptions(this, opts);	if (this.element)		this.element.style.overflow = "hidden";	// Developers can specify the default panel as an index,	// id or an actual element node. Make sure to normalize	// it into an element node because that is what we expect	// internally.	if (this.defaultPanel)	{		if (typeof this.defaultPanel == "number")			this.currentPanel = this.getContentPanels()[this.defaultPanel];		else			this.currentPanel = this.getElement(this.defaultPanel);	}	// If we still don't have a current panel, use the first one!	if (!this.currentPanel)		this.currentPanel = this.getContentPanels()[0];	// Since we rely on the positioning information of the	// panels, we need to wait for the onload event to fire before	// we can attempt to show the initial panel. Once the onload	// fires, we know that all CSS files have loaded. This is	// especially important for Safari.	if (Spry.Widget.SlidingPanels.onloadDidFire)		this.attachBehaviors();	else		Spry.Widget.SlidingPanels.loadQueue.push(this);};Spry.Widget.SlidingPanels.prototype.onFocus = function(e){	this.hasFocus = true;	this.addClassName(this.element, this.focusedClass);	return false;};Spry.Widget.SlidingPanels.prototype.onBlur = function(e){	this.hasFocus = false;	this.removeClassName(this.element, this.focusedClass);	return false;};Spry.Widget.SlidingPanels.KEY_LEFT = 37;Spry.Widget.SlidingPanels.KEY_UP = 38;Spry.Widget.SlidingPanels.KEY_RIGHT = 39;Spry.Widget.SlidingPanels.KEY_DOWN = 40;Spry.Widget.SlidingPanels.prototype.onKeyDown = function(e){	var key = e.keyCode;	if (!this.hasFocus || (key != this.previousPanelKeyCode && key != this.nextPanelKeyCode))		return true;	if (key == this.nextPanelKeyCode)		this.showNextPanel();	else /* if (key == this.previousPanelKeyCode) */		this.showPreviousPanel();	if (e.preventDefault) e.preventDefault();	else e.returnValue = false;	if (e.stopPropagation) e.stopPropagation();	else e.cancelBubble = true;	return false;};Spry.Widget.SlidingPanels.prototype.attachBehaviors = function(){	var ele = this.element;	if (!ele)		return;	if (this.enableKeyboardNavigation)	{		var focusEle = null;		var tabIndexAttr = ele.attributes.getNamedItem("tabindex");		if (tabIndexAttr || ele.nodeName.toLowerCase() == "a")			focusEle = ele;			if (focusEle)		{			var self = this;			Spry.Widget.SlidingPanels.addEventListener(focusEle, "focus", function(e) { return self.onFocus(e || window.event); }, false);			Spry.Widget.SlidingPanels.addEventListener(focusEle, "blur", function(e) { return self.onBlur(e || window.event); }, false);			Spry.Widget.SlidingPanels.addEventListener(focusEle, "keydown", function(e) { return self.onKeyDown(e || window.event); }, false);		}	}	if (this.currentPanel)	{		// Temporarily turn off animation when showing the		// initial panel.		var ea = this.enableAnimation;		this.enableAnimation = false;		this.showPanel(this.currentPanel);		this.enableAnimation = ea;	}};Spry.Widget.SlidingPanels.prototype.getElement = function(ele){	if (ele && typeof ele == "string")		return document.getElementById(ele);	return ele;};Spry.Widget.SlidingPanels.prototype.addClassName = function(ele, className){	if (!ele || !className || (ele.className && ele.className.search(new RegExp("\\b" + className + "\\b")) != -1))		return;	ele.className += (ele.className ? " " : "") + className;};Spry.Widget.SlidingPanels.prototype.removeClassName = function(ele, className){	if (!ele || !className || (ele.className && ele.className.search(new RegExp("\\b" + className + "\\b")) == -1))		return;	ele.className = ele.className.replace(new RegExp("\\s*\\b" + className + "\\b", "g"), "");};Spry.Widget.SlidingPanels.setOptions = function(obj, optionsObj, ignoreUndefinedProps){	if (!optionsObj)		return;	for (var optionName in optionsObj)	{		if (ignoreUndefinedProps && optionsObj[optionName] == undefined)			continue;		obj[optionName] = optionsObj[optionName];	}};Spry.Widget.SlidingPanels.prototype.getElementChildren = function(element){	var children = [];	var child = element.firstChild;	while (child)	{		if (child.nodeType == 1 /* Node.ELEMENT_NODE */)			children.push(child);		child = child.nextSibling;	}	return children;};Spry.Widget.SlidingPanels.prototype.getCurrentPanel = function(){	return this.currentPanel;};Spry.Widget.SlidingPanels.prototype.getContentGroup = function(){	return this.getElementChildren(this.element)[0];};Spry.Widget.SlidingPanels.prototype.getContentPanels = function(){	return this.getElementChildren(this.getContentGroup());};Spry.Widget.SlidingPanels.prototype.getContentPanelsCount = function(){	return this.getContentPanels().length;};Spry.Widget.SlidingPanels.onloadDidFire = false;Spry.Widget.SlidingPanels.loadQueue = [];Spry.Widget.SlidingPanels.addLoadListener = function(handler){	if (typeof window.addEventListener != 'undefined')		window.addEventListener('load', handler, false);	else if (typeof document.addEventListener != 'undefined')		document.addEventListener('load', handler, false);	else if (typeof window.attachEvent != 'undefined')		window.attachEvent('onload', handler);};Spry.Widget.SlidingPanels.processLoadQueue = function(handler){	Spry.Widget.SlidingPanels.onloadDidFire = true;	var q = Spry.Widget.SlidingPanels.loadQueue;	var qlen = q.length;	for (var i = 0; i < qlen; i++)		q[i].attachBehaviors();};Spry.Widget.SlidingPanels.addLoadListener(Spry.Widget.SlidingPanels.processLoadQueue);Spry.Widget.SlidingPanels.addEventListener = function(element, eventType, handler, capture){	try	{		if (element.addEventListener)			element.addEventListener(eventType, handler, capture);		else if (element.attachEvent)			element.attachEvent("on" + eventType, handler);	}	catch (e) {}};Spry.Widget.SlidingPanels.prototype.getContentPanelIndex = function(ele){	if (ele)	{		ele = this.getElement(ele);		var panels = this.getContentPanels();		var numPanels = panels.length;		for (var i = 0; i < numPanels; i++)		{			if (panels[i] == ele)				return i;		}	}	return -1;};Spry.Widget.SlidingPanels.prototype.showPanel = function(elementOrIndex){	var pIndex = -1;		if (typeof elementOrIndex == "number")		pIndex = elementOrIndex;	else // Must be the element for the content panel.		pIndex = this.getContentPanelIndex(elementOrIndex);	var numPanels = this.getContentPanelsCount();	if (numPanels > 0)		pIndex = (pIndex >= numPanels) ? numPanels - 1 : pIndex;	else		pIndex = 0;	var panel = this.getContentPanels()[pIndex];	var contentGroup = this.getContentGroup();	if (panel && contentGroup)	{		if (this.currentPanel)			this.removeClassName(this.currentPanel, this.currentPanelClass);		this.currentPanel = panel;		var nx = -panel.offsetLeft;		var ny = -panel.offsetTop;		if (this.enableAnimation)		{			if (this.animator)				this.animator.stop();			var cx = contentGroup.offsetLeft;			var cy = contentGroup.offsetTop;			if (cx != nx || cy != ny)			{				var self = this;				this.addClassName(this.element, this.animatingClass);				this.animator = new Spry.Widget.SlidingPanels.PanelAnimator(contentGroup, cx, cy, nx, ny, { duration: this.duration, fps: this.fps, transition: this.transition, finish: function()				{					self.removeClassName(self.element, self.animatingClass);					self.addClassName(panel, self.currentPanelClass);				} });				this.animator.start();			}		}		else		{			contentGroup.style.left = nx + "px";			contentGroup.style.top = ny + "px";			this.addClassName(panel, this.currentPanelClass);		}	}	return panel;};Spry.Widget.SlidingPanels.prototype.showFirstPanel = function(){	return this.showPanel(0);};Spry.Widget.SlidingPanels.prototype.showLastPanel = function(){	return this.showPanel(this.getContentPanels().length - 1);};Spry.Widget.SlidingPanels.prototype.showPreviousPanel = function(){	return this.showPanel(this.getContentPanelIndex(this.currentPanel) - 1);};Spry.Widget.SlidingPanels.prototype.showNextPanel = function(){	return this.showPanel(this.getContentPanelIndex(this.currentPanel) + 1);};Spry.Widget.SlidingPanels.PanelAnimator = function(ele, curX, curY, dstX, dstY, opts){	this.element = ele;	this.curX = curX;	this.curY = curY;	this.dstX = dstX;	this.dstY = dstY;	this.fps = 60;	this.duration = 500;	this.transition = Spry.Widget.SlidingPanels.PanelAnimator.defaultTransition;	this.startTime = 0;	this.timerID = 0;	this.finish = null;	var self = this;	this.intervalFunc = function() { self.step(); };		Spry.Widget.SlidingPanels.setOptions(this, opts, true);	this.interval = 1000/this.fps;};Spry.Widget.SlidingPanels.PanelAnimator.defaultTransition = function(time, begin, finish, duration) { time /= duration; return begin + ((2 - time) * time * finish); };Spry.Widget.SlidingPanels.PanelAnimator.prototype.start = function(){	this.stop();	this.startTime = (new Date()).getTime();	this.timerID = setTimeout(this.intervalFunc, this.interval);};Spry.Widget.SlidingPanels.PanelAnimator.prototype.stop = function(){	if (this.timerID)		clearTimeout(this.timerID);	this.timerID = 0;};Spry.Widget.SlidingPanels.PanelAnimator.prototype.step = function(){	var elapsedTime = (new Date()).getTime() - this.startTime;	var done = elapsedTime >= this.duration;	var x, y;	if (done)	{		x = this.curX = this.dstX;		y = this.curY = this.dstY;	}	else	{		x = this.transition(elapsedTime, this.curX, this.dstX - this.curX, this.duration);		y = this.transition(elapsedTime, this.curY, this.dstY - this.curY, this.duration);	}	this.element.style.left = x + "px";	this.element.style.top = y + "px";	if (!done)		this.timerID = setTimeout(this.intervalFunc, this.interval);	else if (this.finish)		this.finish();};