//MooTools More, <http://mootools.net/more>. Copyright (c) 2006-2008 Valerio Proietti, <http://mad4milk.net>, MIT Style License./*Script: Fx.Slide.js	Effect to slide an element in and out of view.License:	MIT-style license.*/Fx.Slide = new Class({	Extends: Fx,	options: {		mode: 'vertical'	},	initialize: function(element, options){		this.addEvent('complete', function(){			this.open = (this.wrapper['offset' + this.layout.capitalize()] != 0);			if (this.open && Browser.Engine.webkit419) this.element.dispose().inject(this.wrapper);		}, true);		this.element = this.subject = $(element);		this.parent(options);		var wrapper = this.element.retrieve('wrapper');		this.wrapper = wrapper || new Element('div', {			styles: $extend(this.element.getStyles('margin', 'position'), {'overflow': 'hidden'})		}).wraps(this.element);		this.element.store('wrapper', this.wrapper).setStyle('margin', 0);		this.now = [];		this.open = true;	},	vertical: function(){		this.margin = 'margin-top';		this.layout = 'height';		this.offset = this.element.offsetHeight;	},	horizontal: function(){		this.margin = 'margin-left';		this.layout = 'width';		this.offset = this.element.offsetWidth;	},	set: function(now){		this.element.setStyle(this.margin, now[0]);		this.wrapper.setStyle(this.layout, now[1]);		return this;	},	compute: function(from, to, delta){		var now = [];		var x = 2;		x.times(function(i){			now[i] = Fx.compute(from[i], to[i], delta);		});		return now;	},	start: function(how, mode){		if (!this.check(arguments.callee, how, mode)) return this;		this[mode || this.options.mode]();		var margin = this.element.getStyle(this.margin).toInt();		var layout = this.wrapper.getStyle(this.layout).toInt();		var caseIn = [[margin, layout], [0, this.offset]];		var caseOut = [[margin, layout], [-this.offset, 0]];		var start;		switch (how){			case 'in': start = caseIn; break;			case 'out': start = caseOut; break;			case 'toggle': start = (this.wrapper['offset' + this.layout.capitalize()] == 0) ? caseIn : caseOut;		}		return this.parent(start[0], start[1]);	},	slideIn: function(mode){		return this.start('in', mode);	},	slideOut: function(mode){		return this.start('out', mode);	},	hide: function(mode){		this[mode || this.options.mode]();		this.open = false;		return this.set([-this.offset, 0]);	},	show: function(mode){		this[mode || this.options.mode]();		this.open = true;		return this.set([0, this.offset]);	},	toggle: function(mode){		return this.start('toggle', mode);	}});Element.Properties.slide = {	set: function(options){		var slide = this.retrieve('slide');		if (slide) slide.cancel();		return this.eliminate('slide').store('slide:options', $extend({link: 'cancel'}, options));	},		get: function(options){		if (options || !this.retrieve('slide')){			if (options || !this.retrieve('slide:options')) this.set('slide', options);			this.store('slide', new Fx.Slide(this, this.retrieve('slide:options')));		}		return this.retrieve('slide');	}};Element.implement({	slide: function(how, mode){		how = how || 'toggle';		var slide = this.get('slide'), toggle;		switch (how){			case 'hide': slide.hide(mode); break;			case 'show': slide.show(mode); break;			case 'toggle':				var flag = this.retrieve('slide:flag', slide.open);				slide[(flag) ? 'slideOut' : 'slideIn'](mode);				this.store('slide:flag', !flag);				toggle = true;			break;			default: slide.start(how, mode);		}		if (!toggle) this.eliminate('slide:flag');		return this;	}});/*Script: Fx.Scroll.js	Effect to smoothly scroll any element, including the window.License:	MIT-style license.*/Fx.Scroll = new Class({	Extends: Fx,	options: {		offset: {'x': 0, 'y': 0},		wheelStops: true	},	initialize: function(element, options){		this.element = this.subject = $(element);		this.parent(options);		var cancel = this.cancel.bind(this, false);		if ($type(this.element) != 'element') this.element = $(this.element.getDocument().body);		var stopper = this.element;		if (this.options.wheelStops){			this.addEvent('start', function(){				stopper.addEvent('mousewheel', cancel);			}, true);			this.addEvent('complete', function(){				stopper.removeEvent('mousewheel', cancel);			}, true);		}	},	set: function(){		var now = Array.flatten(arguments);		this.element.scrollTo(now[0], now[1]);	},	compute: function(from, to, delta){		var now = [];		var x = 2;		x.times(function(i){			now.push(Fx.compute(from[i], to[i], delta));		});		return now;	},	start: function(x, y){		if (!this.check(arguments.callee, x, y)) return this;		var offsetSize = this.element.getSize(), scrollSize = this.element.getScrollSize();		var scroll = this.element.getScroll(), values = {x: x, y: y};		for (var z in values){			var max = scrollSize[z] - offsetSize[z];			if ($chk(values[z])) values[z] = ($type(values[z]) == 'number') ? values[z].limit(0, max) : max;			else values[z] = scroll[z];			values[z] += this.options.offset[z];		}		return this.parent([scroll.x, scroll.y], [values.x, values.y]);	},	toTop: function(){		return this.start(false, 0);	},	toLeft: function(){		return this.start(0, false);	},	toRight: function(){		return this.start('right', false);	},	toBottom: function(){		return this.start(false, 'bottom');	},	toElement: function(el){		var position = $(el).getPosition(this.element);		return this.start(position.x, position.y);	}});/*Script: Fx.Elements.js	Effect to change any number of CSS properties of any number of Elements.License:	MIT-style license.*/Fx.Elements = new Class({	Extends: Fx.CSS,	initialize: function(elements, options){		this.elements = this.subject = $$(elements);		this.parent(options);	},	compute: function(from, to, delta){		var now = {};		for (var i in from){			var iFrom = from[i], iTo = to[i], iNow = now[i] = {};			for (var p in iFrom) iNow[p] = this.parent(iFrom[p], iTo[p], delta);		}		return now;	},	set: function(now){		for (var i in now){			var iNow = now[i];			for (var p in iNow) this.render(this.elements[i], p, iNow[p], this.options.unit);		}		return this;	},	start: function(obj){		if (!this.check(arguments.callee, obj)) return this;		var from = {}, to = {};		for (var i in obj){			var iProps = obj[i], iFrom = from[i] = {}, iTo = to[i] = {};			for (var p in iProps){				var parsed = this.prepare(this.elements[i], p, iProps[p]);				iFrom[p] = parsed.from;				iTo[p] = parsed.to;			}		}		return this.parent(from, to);	}});/*Script: Drag.js	The base Drag Class. Can be used to drag and resize Elements using mouse events.License:	MIT-style license.*/var Drag = new Class({	Implements: [Events, Options],	options: {/*		onBeforeStart: $empty,		onStart: $empty,		onDrag: $empty,		onCancel: $empty,		onComplete: $empty,*/		snap: 6,		unit: 'px',		grid: false,		style: true,		limit: false,		handle: false,		invert: false,		preventDefault: false,		modifiers: {x: 'left', y: 'top'}	},	initialize: function(){		var params = Array.link(arguments, {'options': Object.type, 'element': $defined});		this.element = $(params.element);		this.document = this.element.getDocument();		this.setOptions(params.options || {});		var htype = $type(this.options.handle);		this.handles = (htype == 'array' || htype == 'collection') ? $$(this.options.handle) : $(this.options.handle) || this.element;		this.mouse = {'now': {}, 'pos': {}};		this.value = {'start': {}, 'now': {}};				this.selection = (Browser.Engine.trident) ? 'selectstart' : 'mousedown';				this.bound = {			start: this.start.bind(this),			check: this.check.bind(this),			drag: this.drag.bind(this),			stop: this.stop.bind(this),			cancel: this.cancel.bind(this),			eventStop: $lambda(false)		};		this.attach();	},	attach: function(){		this.handles.addEvent('mousedown', this.bound.start);		return this;	},	detach: function(){		this.handles.removeEvent('mousedown', this.bound.start);		return this;	},	start: function(event){		if (this.options.preventDefault) event.preventDefault();		this.fireEvent('beforeStart', this.element);		this.mouse.start = event.page;		var limit = this.options.limit;		this.limit = {'x': [], 'y': []};		for (var z in this.options.modifiers){			if (!this.options.modifiers[z]) continue;			if (this.options.style) this.value.now[z] = this.element.getStyle(this.options.modifiers[z]).toInt();			else this.value.now[z] = this.element[this.options.modifiers[z]];			if (this.options.invert) this.value.now[z] *= -1;			this.mouse.pos[z] = event.page[z] - this.value.now[z];			if (limit && limit[z]){				for (var i = 2; i--; i){					if ($chk(limit[z][i])) this.limit[z][i] = $lambda(limit[z][i])();				}			}		}		if ($type(this.options.grid) == 'number') this.options.grid = {'x': this.options.grid, 'y': this.options.grid};		this.document.addEvents({mousemove: this.bound.check, mouseup: this.bound.cancel});		this.document.addEvent(this.selection, this.bound.eventStop);	},	check: function(event){		if (this.options.preventDefault) event.preventDefault();		var distance = Math.round(Math.sqrt(Math.pow(event.page.x - this.mouse.start.x, 2) + Math.pow(event.page.y - this.mouse.start.y, 2)));		if (distance > this.options.snap){			this.cancel();			this.document.addEvents({				mousemove: this.bound.drag,				mouseup: this.bound.stop			});			this.fireEvent('start', this.element).fireEvent('snap', this.element);		}	},	drag: function(event){		if (this.options.preventDefault) event.preventDefault();		this.mouse.now = event.page;		for (var z in this.options.modifiers){			if (!this.options.modifiers[z]) continue;			this.value.now[z] = this.mouse.now[z] - this.mouse.pos[z];			if (this.options.invert) this.value.now[z] *= -1;			if (this.options.limit && this.limit[z]){				if ($chk(this.limit[z][1]) && (this.value.now[z] > this.limit[z][1])){					this.value.now[z] = this.limit[z][1];				} else if ($chk(this.limit[z][0]) && (this.value.now[z] < this.limit[z][0])){					this.value.now[z] = this.limit[z][0];				}			}			if (this.options.grid[z]) this.value.now[z] -= (this.value.now[z] % this.options.grid[z]);			if (this.options.style) this.element.setStyle(this.options.modifiers[z], this.value.now[z] + this.options.unit);			else this.element[this.options.modifiers[z]] = this.value.now[z];		}		this.fireEvent('drag', this.element);	},	cancel: function(event){		this.document.removeEvent('mousemove', this.bound.check);		this.document.removeEvent('mouseup', this.bound.cancel);		if (event){			this.document.removeEvent(this.selection, this.bound.eventStop);			this.fireEvent('cancel', this.element);		}	},	stop: function(event){		this.document.removeEvent(this.selection, this.bound.eventStop);		this.document.removeEvent('mousemove', this.bound.drag);		this.document.removeEvent('mouseup', this.bound.stop);		if (event) this.fireEvent('complete', this.element);	}});Element.implement({		makeResizable: function(options){		return new Drag(this, $merge({modifiers: {'x': 'width', 'y': 'height'}}, options));	}});/*Script: Drag.Move.js	A Drag extension that provides support for the constraining of draggables to containers and droppables.License:	MIT-style license.*/Drag.Move = new Class({	Extends: Drag,	options: {		droppables: [],		container: false	},	initialize: function(element, options){		this.parent(element, options);		this.droppables = $$(this.options.droppables);		this.container = $(this.options.container);		if (this.container && $type(this.container) != 'element') this.container = $(this.container.getDocument().body);		element = this.element;				var current = element.getStyle('position');		var position = (current != 'static') ? current : 'absolute';		if (element.getStyle('left') == 'auto' || element.getStyle('top') == 'auto') element.position(element.getPosition(element.offsetParent));				element.setStyle('position', position);				this.addEvent('start', function(){			this.checkDroppables();		}, true);	},	start: function(event){		if (this.container){			var el = this.element, cont = this.container, ccoo = cont.getCoordinates(el.offsetParent), cps = {}, ems = {};			['top', 'right', 'bottom', 'left'].each(function(pad){				cps[pad] = cont.getStyle('padding-' + pad).toInt();				ems[pad] = el.getStyle('margin-' + pad).toInt();			}, this);			var width = el.offsetWidth + ems.left + ems.right, height = el.offsetHeight + ems.top + ems.bottom;			var x = [ccoo.left + cps.left, ccoo.right - cps.right - width];			var y = [ccoo.top + cps.top, ccoo.bottom - cps.bottom - height];			this.options.limit = {x: x, y: y};		}		this.parent(event);	},	checkAgainst: function(el){		el = el.getCoordinates();		var now = this.mouse.now;		return (now.x > el.left && now.x < el.right && now.y < el.bottom && now.y > el.top);	},	checkDroppables: function(){		var overed = this.droppables.filter(this.checkAgainst, this).getLast();		if (this.overed != overed){			if (this.overed) this.fireEvent('leave', [this.element, this.overed]);			if (overed){				this.overed = overed;				this.fireEvent('enter', [this.element, overed]);			} else {				this.overed = null;			}		}	},	drag: function(event){		this.parent(event);		if (this.droppables.length) this.checkDroppables();	},	stop: function(event){		this.checkDroppables();		this.fireEvent('drop', [this.element, this.overed]);		this.overed = null;		return this.parent(event);	}});Element.implement({	makeDraggable: function(options){		return new Drag.Move(this, options);	}});/*Script: Hash.Cookie.js	Class for creating, reading, and deleting Cookies in JSON format.License:	MIT-style license.*/Hash.Cookie = new Class({	Extends: Cookie,	options: {		autoSave: true	},	initialize: function(name, options){		this.parent(name, options);		this.load();	},	save: function(){		var value = JSON.encode(this.hash);		if (!value || value.length > 4096) return false; //cookie would be truncated!		if (value == '{}') this.dispose();		else this.write(value);		return true;	},	load: function(){		this.hash = new Hash(JSON.decode(this.read(), true));		return this;	}});Hash.Cookie.implement((function(){		var methods = {};		Hash.each(Hash.prototype, function(method, name){		methods[name] = function(){			var value = method.apply(this.hash, arguments);			if (this.options.autoSave) this.save();			return value;		};	});		return methods;	})());/*Script: Color.js	Class for creating and manipulating colors in JavaScript. Supports HSB -> RGB Conversions and vice versa.License:	MIT-style license.*/var Color = new Native({  	initialize: function(color, type){		if (arguments.length >= 3){			type = "rgb"; color = Array.slice(arguments, 0, 3);		} else if (typeof color == 'string'){			if (color.match(/rgb/)) color = color.rgbToHex().hexToRgb(true);			else if (color.match(/hsb/)) color = color.hsbToRgb();			else color = color.hexToRgb(true);		}		type = type || 'rgb';		switch (type){			case 'hsb':				var old = color;				color = color.hsbToRgb();				color.hsb = old;			break;			case 'hex': color = color.hexToRgb(true); break;		}		color.rgb = color.slice(0, 3);		color.hsb = color.hsb || color.rgbToHsb();		color.hex = color.rgbToHex();		return $extend(color, this);	}});Color.implement({	mix: function(){		var colors = Array.slice(arguments);		var alpha = ($type(colors.getLast()) == 'number') ? colors.pop() : 50;		var rgb = this.slice();		colors.each(function(color){			color = new Color(color);			for (var i = 0; i < 3; i++) rgb[i] = Math.round((rgb[i] / 100 * (100 - alpha)) + (color[i] / 100 * alpha));		});		return new Color(rgb, 'rgb');	},	invert: function(){		return new Color(this.map(function(value){			return 255 - value;		}));	},	setHue: function(value){		return new Color([value, this.hsb[1], this.hsb[2]], 'hsb');	},	setSaturation: function(percent){		return new Color([this.hsb[0], percent, this.hsb[2]], 'hsb');	},	setBrightness: function(percent){		return new Color([this.hsb[0], this.hsb[1], percent], 'hsb');	}});function $RGB(r, g, b){	return new Color([r, g, b], 'rgb');};function $HSB(h, s, b){	return new Color([h, s, b], 'hsb');};function $HEX(hex){	return new Color(hex, 'hex');};Array.implement({	rgbToHsb: function(){		var red = this[0], green = this[1], blue = this[2];		var hue, saturation, brightness;		var max = Math.max(red, green, blue), min = Math.min(red, green, blue);		var delta = max - min;		brightness = max / 255;		saturation = (max != 0) ? delta / max : 0;		if (saturation == 0){			hue = 0;		} else {			var rr = (max - red) / delta;			var gr = (max - green) / delta;			var br = (max - blue) / delta;			if (red == max) hue = br - gr;			else if (green == max) hue = 2 + rr - br;			else hue = 4 + gr - rr;			hue /= 6;			if (hue < 0) hue++;		}		return [Math.round(hue * 360), Math.round(saturation * 100), Math.round(brightness * 100)];	},	hsbToRgb: function(){		var br = Math.round(this[2] / 100 * 255);		if (this[1] == 0){			return [br, br, br];		} else {			var hue = this[0] % 360;			var f = hue % 60;			var p = Math.round((this[2] * (100 - this[1])) / 10000 * 255);			var q = Math.round((this[2] * (6000 - this[1] * f)) / 600000 * 255);			var t = Math.round((this[2] * (6000 - this[1] * (60 - f))) / 600000 * 255);			switch (Math.floor(hue / 60)){				case 0: return [br, t, p];				case 1: return [q, br, p];				case 2: return [p, br, t];				case 3: return [p, q, br];				case 4: return [t, p, br];				case 5: return [br, p, q];			}		}		return false;	}});String.implement({	rgbToHsb: function(){		var rgb = this.match(/\d{1,3}/g);		return (rgb) ? hsb.rgbToHsb() : null;	},		hsbToRgb: function(){		var hsb = this.match(/\d{1,3}/g);		return (hsb) ? hsb.hsbToRgb() : null;	}});/*Script: Group.js	Class for monitoring collections of eventsLicense:	MIT-style license.*/var Group = new Class({	initialize: function(){		this.instances = Array.flatten(arguments);		this.events = {};		this.checker = {};	},	addEvent: function(type, fn){		this.checker[type] = this.checker[type] || {};		this.events[type] = this.events[type] || [];		if (this.events[type].contains(fn)) return false;		else this.events[type].push(fn);		this.instances.each(function(instance, i){			instance.addEvent(type, this.check.bind(this, [type, instance, i]));		}, this);		return this;	},	check: function(type, instance, i){		this.checker[type][i] = true;		var every = this.instances.every(function(current, j){			return this.checker[type][j] || false;		}, this);		if (!every) return;		this.checker[type] = {};		this.events[type].each(function(event){			event.call(this, this.instances, instance);		}, this);	}});/*Script: Assets.js	Provides methods to dynamically load JavaScript, CSS, and Image files into the document.License:	MIT-style license.*/var Asset = new Hash({	javascript: function(source, properties){		properties = $extend({			onload: $empty,			document: document,			check: $lambda(true)		}, properties);				var script = new Element('script', {'src': source, 'type': 'text/javascript'});				var load = properties.onload.bind(script), check = properties.check, doc = properties.document;		delete properties.onload; delete properties.check; delete properties.document;				script.addEvents({			load: load,			readystatechange: function(){				if (['loaded', 'complete'].contains(this.readyState)) load();			}		}).setProperties(properties);						if (Browser.Engine.webkit419) var checker = (function(){			if (!$try(check)) return;			$clear(checker);			load();		}).periodical(50);				return script.inject(doc.head);	},	css: function(source, properties){		return new Element('link', $merge({			'rel': 'stylesheet', 'media': 'screen', 'type': 'text/css', 'href': source		}, properties)).inject(document.head);	},	image: function(source, properties){		properties = $merge({			'onload': $empty,			'onabort': $empty,			'onerror': $empty		}, properties);		var image = new Image();		var element = $(image) || new Element('img');		['load', 'abort', 'error'].each(function(name){			var type = 'on' + name;			var event = properties[type];			delete properties[type];			image[type] = function(){				if (!image) return;				if (!element.parentNode){					element.width = image.width;					element.height = image.height;				}				image = image.onload = image.onabort = image.onerror = null;				event.delay(1, element, element);				element.fireEvent(name, element, 1);			};		});		image.src = element.src = source;		if (image && image.complete) image.onload.delay(1);		return element.setProperties(properties);	},	images: function(sources, options){		options = $merge({			onComplete: $empty,			onProgress: $empty		}, options);		if (!sources.push) sources = [sources];		var images = [];		var counter = 0;		sources.each(function(source){			var img = new Asset.image(source, {				'onload': function(){					options.onProgress.call(this, counter, sources.indexOf(source));					counter++;					if (counter == sources.length) options.onComplete();				}			});			images.push(img);		});		return new Elements(images);	}});/*Script: Sortables.js	Class for creating a drag and drop sorting interface for lists of items.License:	MIT-style license.*/var Sortables = new Class({	Implements: [Events, Options],	options: {/*		onSort: $empty,		onStart: $empty,		onComplete: $empty,*/		snap: 4,		opacity: 1,		clone: false,		revert: false,		handle: false,		constrain: false	},	initialize: function(lists, options){		this.setOptions(options);		this.elements = [];		this.lists = [];		this.idle = true;				this.addLists($$($(lists) || lists));		if (!this.options.clone) this.options.revert = false;		if (this.options.revert) this.effect = new Fx.Morph(null, $merge({duration: 250, link: 'cancel'}, this.options.revert));	},	attach: function(){		this.addLists(this.lists);		return this;	},	detach: function(){		this.lists = this.removeLists(this.lists);		return this;	},	addItems: function(){		Array.flatten(arguments).each(function(element){			this.elements.push(element);			var start = element.retrieve('sortables:start', this.start.bindWithEvent(this, element));			(this.options.handle ? element.getElement(this.options.handle) || element : element).addEvent('mousedown', start);		}, this);		return this;	},	addLists: function(){		Array.flatten(arguments).each(function(list){			this.lists.push(list);			this.addItems(list.getChildren());		}, this);		return this;	},	removeItems: function(){		var elements = [];		Array.flatten(arguments).each(function(element){			elements.push(element);			this.elements.erase(element);			var start = element.retrieve('sortables:start');			(this.options.handle ? element.getElement(this.options.handle) || element : element).removeEvent('mousedown', start);		}, this);		return $$(elements);	},	removeLists: function(){		var lists = [];		Array.flatten(arguments).each(function(list){			lists.push(list);			this.lists.erase(list);			this.removeItems(list.getChildren());		}, this);		return $$(lists);	},	getClone: function(event, element){		if (!this.options.clone) return new Element('div').inject(document.body);		if ($type(this.options.clone) == 'function') return this.options.clone.call(this, event, element, this.list);		return element.clone(true).setStyles({			'margin': '0px',			'position': 'absolute',			'visibility': 'hidden',			'width': element.getStyle('width')		}).inject(this.list).position(element.getPosition(element.getOffsetParent()));	},	getDroppables: function(){		var droppables = this.list.getChildren();		if (!this.options.constrain) droppables = this.lists.concat(droppables).erase(this.list);		return droppables.erase(this.clone).erase(this.element);	},	insert: function(dragging, element){		var where = 'inside';		if (this.lists.contains(element)){			this.list = element;			this.drag.droppables = this.getDroppables();		} else {			where = this.element.getAllPrevious().contains(element) ? 'before' : 'after';		}		this.element.inject(element, where);		this.fireEvent('sort', [this.element, this.clone]);	},	start: function(event, element){		if (!this.idle) return;		this.idle = false;		this.element = element;		this.opacity = element.get('opacity');		this.list = element.getParent();		this.clone = this.getClone(event, element);				this.drag = new Drag.Move(this.clone, {			snap: this.options.snap,			container: this.options.constrain && this.element.getParent(),			droppables: this.getDroppables(),			onSnap: function(){				event.stop();				this.clone.setStyle('visibility', 'visible');				this.element.set('opacity', this.options.opacity || 0);				this.fireEvent('start', [this.element, this.clone]);			}.bind(this),			onEnter: this.insert.bind(this),			onCancel: this.reset.bind(this),			onComplete: this.end.bind(this)		});				this.clone.inject(this.element, 'before');		this.drag.start(event);	},	end: function(){		this.drag.detach();		this.element.set('opacity', this.opacity);		if (this.effect){			var dim = this.element.getStyles('width', 'height');			var pos = this.clone.computePosition(this.element.getPosition(this.clone.offsetParent));			this.effect.element = this.clone;			this.effect.start({				top: pos.top,				left: pos.left,				width: dim.width,				height: dim.height,				opacity: 0.25			}).chain(this.reset.bind(this));		} else {			this.reset();		}	},	reset: function(){		this.idle = true;		this.clone.destroy();		this.fireEvent('complete', this.element);	},	serialize: function(){		var params = Array.link(arguments, {modifier: Function.type, index: $defined});		var serial = this.lists.map(function(list){			return list.getChildren().map(params.modifier || function(element){				return element.get('id');			}, this);		}, this);				var index = params.index;		if (this.lists.length == 1) index = 0;		return $chk(index) && index >= 0 && index < this.lists.length ? serial[index] : serial;	}});/*Script: Tips.js	Class for creating nice tips that follow the mouse cursor when hovering an element.License:	MIT-style license.*/var Tips = new Class({	Implements: [Events, Options],	options: {		onShow: function(tip){			tip.setStyle('visibility', 'visible');		},		onHide: function(tip){			tip.setStyle('visibility', 'hidden');		},		showDelay: 100,		hideDelay: 100,		className: null,		offsets: {x: 16, y: 16},		fixed: false	},	initialize: function(){		var params = Array.link(arguments, {options: Object.type, elements: $defined});		this.setOptions(params.options || null);				this.tip = new Element('div').inject(document.body);				if (this.options.className) this.tip.addClass(this.options.className);				var top = new Element('div', {'class': 'tip-top'}).inject(this.tip);		this.container = new Element('div', {'class': 'tip'}).inject(this.tip);		var bottom = new Element('div', {'class': 'tip-bottom'}).inject(this.tip);		this.tip.setStyles({position: 'absolute', top: 0, left: 0, visibility: 'hidden'});				if (params.elements) this.attach(params.elements);	},		attach: function(elements){		$$(elements).each(function(element){			var title = element.retrieve('tip:title', element.get('title'));			var text = element.retrieve('tip:text', element.get('rel') || element.get('href'));			var enter = element.retrieve('tip:enter', this.elementEnter.bindWithEvent(this, element));			var leave = element.retrieve('tip:leave', this.elementLeave.bindWithEvent(this, element));			element.addEvents({mouseenter: enter, mouseleave: leave});			if (!this.options.fixed){				var move = element.retrieve('tip:move', this.elementMove.bindWithEvent(this, element));				element.addEvent('mousemove', move);			}			element.store('tip:native', element.get('title'));			element.erase('title');		}, this);		return this;	},		detach: function(elements){		$$(elements).each(function(element){			element.removeEvent('mouseenter', element.retrieve('tip:enter') || $empty);			element.removeEvent('mouseleave', element.retrieve('tip:leave') || $empty);			element.removeEvent('mousemove', element.retrieve('tip:move') || $empty);			element.eliminate('tip:enter').eliminate('tip:leave').eliminate('tip:move');			var original = element.retrieve('tip:native');			if (original) element.set('title', original);		});		return this;	},		elementEnter: function(event, element){				$A(this.container.childNodes).each(Element.dispose);				var title = element.retrieve('tip:title');				if (title){			this.titleElement = new Element('div', {'class': 'tip-title'}).inject(this.container);			this.fill(this.titleElement, title);		}				var text = element.retrieve('tip:text');		if (text){			this.textElement = new Element('div', {'class': 'tip-text'}).inject(this.container);			this.fill(this.textElement, text);		}				this.timer = $clear(this.timer);		this.timer = this.show.delay(this.options.showDelay, this);		this.position((!this.options.fixed) ? event : {page: element.getPosition()});	},		elementLeave: function(event){		$clear(this.timer);		this.timer = this.hide.delay(this.options.hideDelay, this);	},		elementMove: function(event){		this.position(event);	},		position: function(event){		var size = window.getSize(), scroll = window.getScroll();		var tip = {x: this.tip.offsetWidth, y: this.tip.offsetHeight};		var props = {x: 'left', y: 'top'};		for (var z in props){			var pos = event.page[z] + this.options.offsets[z];			if ((pos + tip[z] - scroll[z]) > size[z]) pos = event.page[z] - this.options.offsets[z] - tip[z];			this.tip.setStyle(props[z], pos);		}	},		fill: function(element, contents){		(typeof contents == 'string') ? element.set('html', contents) : element.adopt(contents);	},	show: function(){		this.fireEvent('show', this.tip);	},	hide: function(){		this.fireEvent('hide', this.tip);	}});/*Script: SmoothScroll.js	Class for creating a smooth scrolling effect to all internal links on the page.License:	MIT-style license.*/var SmoothScroll = new Class({	Extends: Fx.Scroll,	initialize: function(options, context){		context = context || document;		var doc = context.getDocument(), win = context.getWindow();		this.parent(doc, options);		this.links = (this.options.links) ? $$(this.options.links) : $$(doc.links);		var location = win.location.href.match(/^[^#]*/)[0] + '#';		this.links.each(function(link){			if (link.href.indexOf(location) != 0) return;			var anchor = link.href.substr(location.length);			if (anchor && $(anchor)) this.useLink(link, anchor);		}, this);		if (!Browser.Engine.webkit419) this.addEvent('complete', function(){			win.location.hash = this.anchor;		}, true);	},	useLink: function(link, anchor){		link.addEvent('click', function(event){			this.anchor = anchor;			this.toElement(anchor);			event.stop();		}.bind(this));	}});/*Script: Slider.js	Class for creating horizontal and vertical slider controls.License:	MIT-style license.*/var Slider = new Class({	Implements: [Events, Options],	options: {/*		onChange: $empty,		onComplete: $empty,*/		onTick: function(position){			if(this.options.snap) position = this.toPosition(this.step);			this.knob.setStyle(this.property, position);		},		snap: false,		offset: 0,		range: false,		wheel: false,		steps: 100,		mode: 'horizontal'	},	initialize: function(element, knob, options){		this.setOptions(options);		this.element = $(element);		this.knob = $(knob);		this.previousChange = this.previousEnd = this.step = -1;		this.element.addEvent('mousedown', this.clickedElement.bind(this));		if (this.options.wheel) this.element.addEvent('mousewheel', this.scrolledElement.bindWithEvent(this));		var offset, limit = {}, modifiers = {'x': false, 'y': false};		switch (this.options.mode){			case 'vertical':				this.axis = 'y';				this.property = 'top';				offset = 'offsetHeight';				break;			case 'horizontal':				this.axis = 'x';				this.property = 'left';				offset = 'offsetWidth';		}		this.half = this.knob[offset] / 2;		this.full = this.element[offset] - this.knob[offset] + (this.options.offset * 2);		this.min = $chk(this.options.range[0]) ? this.options.range[0] : 0;		this.max = $chk(this.options.range[1]) ? this.options.range[1] : this.options.steps;		this.range = this.max - this.min;		this.steps = this.options.steps || this.full;		this.stepSize = Math.abs(this.range) / this.steps;		this.stepWidth = this.stepSize * this.full / Math.abs(this.range) ;				this.knob.setStyle('position', 'relative').setStyle(this.property, - this.options.offset);		modifiers[this.axis] = this.property;		limit[this.axis] = [- this.options.offset, this.full - this.options.offset];		this.drag = new Drag(this.knob, {			snap: 0,			limit: limit,			modifiers: modifiers,			onDrag: this.draggedKnob.bind(this),			onStart: this.draggedKnob.bind(this),			onComplete: function(){				this.draggedKnob();				this.end();			}.bind(this)		});		if (this.options.snap) {			this.drag.options.grid = Math.ceil(this.stepWidth);			this.drag.options.limit[this.axis][1] = this.full;		}	},	set: function(step){		if (!((this.range > 0) ^ (step < this.min))) step = this.min;		if (!((this.range > 0) ^ (step > this.max))) step = this.max;				this.step = Math.round(step);		this.checkStep();		this.end();		this.fireEvent('tick', this.toPosition(this.step));		return this;	},	clickedElement: function(event){		var dir = this.range < 0 ? -1 : 1;		var position = event.page[this.axis] - this.element.getPosition()[this.axis] - this.half;		position = position.limit(-this.options.offset, this.full -this.options.offset);				this.step = Math.round(this.min + dir * this.toStep(position));		this.checkStep();		this.end();		this.fireEvent('tick', position);	},		scrolledElement: function(event){		var mode = (this.options.mode == 'horizontal') ? (event.wheel < 0) : (event.wheel > 0);		this.set(mode ? this.step - this.stepSize : this.step + this.stepSize);		event.stop();	},
	draggedKnob: function(){		var dir = this.range < 0 ? -1 : 1;		var position = this.drag.value.now[this.axis];		position = position.limit(-this.options.offset, this.full -this.options.offset);		this.step = Math.round(this.min + dir * this.toStep(position));		this.checkStep();	},	checkStep: function(){		if (this.previousChange != this.step){			this.previousChange = this.step;			this.fireEvent('change', this.step);		}	},	end: function(){		if (this.previousEnd !== this.step){			this.previousEnd = this.step;			this.fireEvent('complete', this.step + '');		}	},	toStep: function(position){		var step = (position + this.options.offset) * this.stepSize / this.full * this.steps;		return this.options.steps ? Math.round(step -= step % this.stepSize) : step;	},	toPosition: function(step){		return (this.full * Math.abs(this.min - step)) / (this.steps * this.stepSize) - this.options.offset;	}});/*Script: Scroller.js	Class which scrolls the contents of any Element (including the window) when the mouse reaches the Element's boundaries.License:	MIT-style license.*/var Scroller = new Class({	Implements: [Events, Options],	options: {		area: 20,		velocity: 1,		onChange: function(x, y){			this.element.scrollTo(x, y);		}	},	initialize: function(element, options){		this.setOptions(options);		this.element = $(element);		this.listener = ($type(this.element) != 'element') ? $(this.element.getDocument().body) : this.element;		this.timer = null;		this.coord = this.getCoords.bind(this);	},	start: function(){		this.listener.addEvent('mousemove', this.coord);	},	stop: function(){		this.listener.removeEvent('mousemove', this.coord);		this.timer = $clear(this.timer);	},	getCoords: function(event){		this.page = (this.listener.get('tag') == 'body') ? event.client : event.page;		if (!this.timer) this.timer = this.scroll.periodical(50, this);	},	scroll: function(){		var size = this.element.getSize(), scroll = this.element.getScroll(), pos = this.element.getPosition(), change = {'x': 0, 'y': 0};		for (var z in this.page){			if (this.page[z] < (this.options.area + pos[z]) && scroll[z] != 0)				change[z] = (this.page[z] - this.options.area - pos[z]) * this.options.velocity;			else if (this.page[z] + this.options.area > (size[z] + pos[z]) && size[z] + size[z] != scroll[z])				change[z] = (this.page[z] - size[z] + this.options.area - pos[z]) * this.options.velocity;		}		if (change.y || change.x) this.fireEvent('change', [scroll.x + change.x, scroll.y + change.y]);	}});/*Script: Accordion.js	An Fx.Elements extension which allows you to easily create accordion type controls.License:	MIT-style license.*/var Accordion = new Class({	Extends: Fx.Elements,	options: {/*		onActive: $empty,		onBackground: $empty,*/		display: 0,		show: false,		height: true,		width: false,		opacity: true,		fixedHeight: false,		fixedWidth: false,		wait: false,		alwaysHide: false	},	initialize: function(){		var params = Array.link(arguments, {'container': Element.type, 'options': Object.type, 'togglers': $defined, 'elements': $defined});		this.parent(params.elements, params.options);		this.togglers = $$(params.togglers);		this.container = $(params.container);		this.previous = -1;		if (this.options.alwaysHide) this.options.wait = true;		if ($chk(this.options.show)){			this.options.display = false;			this.previous = this.options.show;		}		if (this.options.start){			this.options.display = false;			this.options.show = false;		}		this.effects = {};		if (this.options.opacity) this.effects.opacity = 'fullOpacity';		if (this.options.width) this.effects.width = this.options.fixedWidth ? 'fullWidth' : 'offsetWidth';		if (this.options.height) this.effects.height = this.options.fixedHeight ? 'fullHeight' : 'scrollHeight';		for (var i = 0, l = this.togglers.length; i < l; i++) this.addSection(this.togglers[i], this.elements[i]);		this.elements.each(function(el, i){			if (this.options.show === i){				this.fireEvent('active', [this.togglers[i], el]);			} else {				for (var fx in this.effects) el.setStyle(fx, 0);			}		}, this);		if ($chk(this.options.display)) this.display(this.options.display);	},	addSection: function(toggler, element, pos){		toggler = $(toggler);		element = $(element);		var test = this.togglers.contains(toggler);		var len = this.togglers.length;		this.togglers.include(toggler);		this.elements.include(element);		if (len && (!test || pos)){			pos = $pick(pos, len - 1);			toggler.inject(this.togglers[pos], 'before');			element.inject(toggler, 'after');		} else if (this.container && !test){			toggler.inject(this.container);			element.inject(this.container);		}		var idx = this.togglers.indexOf(toggler);		toggler.addEvent('click', this.display.bind(this, idx));		if (this.options.height) element.setStyles({'padding-top': 0, 'border-top': 'none', 'padding-bottom': 0, 'border-bottom': 'none'});		if (this.options.width) element.setStyles({'padding-left': 0, 'border-left': 'none', 'padding-right': 0, 'border-right': 'none'});		element.fullOpacity = 1;		if (this.options.fixedWidth) element.fullWidth = this.options.fixedWidth;		if (this.options.fixedHeight) element.fullHeight = this.options.fixedHeight;		element.setStyle('overflow', 'hidden');		if (!test){			for (var fx in this.effects) element.setStyle(fx, 0);		}		return this;	},	display: function(index){		index = ($type(index) == 'element') ? this.elements.indexOf(index) : index;		if ((this.timer && this.options.wait) || (index === this.previous && !this.options.alwaysHide)) return this;		this.previous = index;		var obj = {};		this.elements.each(function(el, i){			obj[i] = {};			var hide = (i != index) || (this.options.alwaysHide && (el.offsetHeight > 0));			this.fireEvent(hide ? 'background' : 'active', [this.togglers[i], el]);			for (var fx in this.effects) obj[i][fx] = hide ? 0 : el[this.effects[fx]];		}, this);		return this.start(obj);	}});
