/** * SqueezeBox - Expandable Lightbox * * Allows to open various content as modal, * centered and animated box. * * Dependencies: MooTools 1.2 * * Inspired by *  ... Lokesh Dhakar	- The original Lightbox v2 * * @version		1.1 rc4 * * @license		MIT-style license * @author		Harald Kirschner <mail [at] digitarald.de> * @copyright	Author */var SqueezeBox = {	presets: {		onOpen: $empty,		onClose: $empty,		onUpdate: $empty,		onResize: $empty,		onMove: $empty,		onShow: $empty,		onHide: $empty,		size: {x: 600, y: 450},		sizeLoading: {x: 200, y: 150},		marginInner: {x: 10, y: 20},		marginImage: {x: 50, y: 75},		handler: false,		target: null,		closable: true,		closeBtn: true,		zIndex: 65555,		overlayOpacity: 0.7,		classWindow: '',		classOverlay: '',		overlayFx: {},		resizeFx: {},		contentFx: {},		parse: false, // 'rel'		parseSecure: false,		shadow: true,		document: null,		ajaxOptions: {}	},	initialize: function(presets) {		if (this.options) return this;		this.presets = $merge(this.presets, presets);		this.doc = this.presets.document || document;		this.options = {};		this.setOptions(this.presets).build();		this.bound = {			window: this.reposition.bind(this, [null]),			scroll: this.checkTarget.bind(this),			close: this.close.bind(this),			key: this.onKey.bind(this)		};		this.isOpen = this.isLoading = false;		return this;	},	build: function() {		this.overlay = new Element('div', {			id: 'sbox-overlay',			styles: {display: 'none', zIndex: this.options.zIndex}		});		this.win = new Element('div', {			id: 'sbox-window',			styles: {display: 'none', zIndex: this.options.zIndex + 2}		});		if (this.options.shadow) {			if (Browser.Engine.webkit420) {				this.win.setStyle('-webkit-box-shadow', '0 0 10px rgba(0, 0, 0, 0.7)');			} else if (!Browser.Engine.trident4) {				var shadow = new Element('div', {'class': 'sbox-bg-wrap'}).inject(this.win);				var relay = function(e) {					this.overlay.fireEvent('click', [e]);				}.bind(this);				['n', 'ne', 'e', 'se', 's', 'sw', 'w', 'nw'].each(function(dir) {					new Element('div', {'class': 'sbox-bg sbox-bg-' + dir}).inject(shadow).addEvent('click', relay);				});			}		}		this.content = new Element('div', {id: 'sbox-content'}).inject(this.win);		this.closeBtn = new Element('a', {id: 'sbox-btn-close', href: '#'}).inject(this.win);		this.fx = {			overlay: new Fx.Tween(this.overlay, $merge({				property: 'opacity',				onStart: Events.prototype.clearChain,				duration: 250,				link: 'cancel'			}, this.options.overlayFx)).set(0),			win: new Fx.Morph(this.win, $merge({				onStart: Events.prototype.clearChain,				unit: 'px',				duration: 750,				transition: Fx.Transitions.Quint.easeOut,				link: 'cancel',				unit: 'px'			}, this.options.resizeFx)),			content: new Fx.Tween(this.content, $merge({				property: 'opacity',				duration: 250,				link: 'cancel'			}, this.options.contentFx)).set(0)		};		$(this.doc.body).adopt(this.overlay, this.win);	},	assign: function(to, options) {		return ($(to) || $$(to)).addEvent('click', function() {			return !SqueezeBox.fromElement(this, options);		});	},		open: function(subject, options) {		this.initialize();		if (this.element != null) this.trash();		this.element = $(subject) || false;				this.setOptions($merge(this.presets, options || {}));				if (this.element && this.options.parse) {			var obj = this.element.getProperty(this.options.parse);			if (obj && (obj = JSON.decode(obj, this.options.parseSecure))) this.setOptions(obj);		}		this.url = ((this.element) ? (this.element.get('href')) : subject) || this.options.url || '';		this.assignOptions();				var handler = handler || this.options.handler;		if (handler) return this.setContent(handler, this.parsers[handler].call(this, true));		var ret = false;		return this.parsers.some(function(parser, key) {			var content = parser.call(this);			if (content) {				ret = this.setContent(key, content);				return true;			}			return false;		}, this);	},		fromElement: function(from, options) {		return this.open(from, options);	},	assignOptions: function() {		this.overlay.set('class', this.options.classOverlay);		this.win.set('class', this.options.classWindow);		if (Browser.Engine.trident4) this.win.addClass('sbox-window-ie6');	},	close: function(e) {		var stoppable = ($type(e) == 'event');		if (stoppable) e.stop();		if (!this.isOpen || (stoppable && !$lambda(this.options.closable).call(this, e))) return this;		this.fx.overlay.start(0).chain(this.toggleOverlay.bind(this));		this.win.setStyle('display', 'none');		this.fireEvent('onClose', [this.content]);		this.trash();		this.toggleListeners();		this.isOpen = false;		return this;	},	trash: function() {		this.element = this.asset = null;		this.content.empty();		this.options = {};		this.removeEvents().setOptions(this.presets).callChain();	},	onError: function() {		this.asset = null;		this.setContent('string', this.options.errorMsg || 'An error occurred');	},	setContent: function(handler, content) {		if (!this.handlers[handler]) return false;		this.content.className = 'sbox-content-' + handler;		this.applyTimer = this.applyContent.delay(this.fx.overlay.options.duration, this, this.handlers[handler].call(this, content));		if (this.overlay.retrieve('opacity')) return this;		this.toggleOverlay(true);		this.fx.overlay.start(this.options.overlayOpacity);		return this.reposition();	},	applyContent: function(content, size) {		if (!this.isOpen && !this.applyTimer) return;		this.applyTimer = $clear(this.applyTimer);		this.hideContent();		if (!content) {			this.toggleLoading(true);		} else {			if (this.isLoading) this.toggleLoading(false);			this.fireEvent('onUpdate', [this.content], 20);		}		if (content) {			if (['string', 'array'].contains($type(content))) this.content.set('html', content);			else if (!this.content.hasChild(content)) this.content.adopt(content);		}		this.callChain();		if (!this.isOpen) {			this.toggleListeners(true);			this.resize(size, true);			this.isOpen = true;			this.fireEvent('onOpen', [this.content]);		} else {			this.resize(size);		}	},	resize: function(size, instantly) {		this.showTimer = $clear(this.showTimer || null);		var box = this.doc.getSize(), scroll = this.doc.getScroll();		this.size = $merge((this.isLoading) ? this.options.sizeLoading : this.options.size, size);		var to = {			width: this.size.x,			height: this.size.y,			left: (scroll.x + (box.x - this.size.x - this.options.marginInner.x) / 2).toInt(),			top: (scroll.y + (box.y - this.size.y - this.options.marginInner.y) / 2).toInt()		};		this.hideContent();		if (!instantly) {			this.fx.win.start(to).chain(this.showContent.bind(this));		} else {			this.win.setStyles(to).setStyle('display', '');			this.showTimer = this.showContent.delay(50, this);		}		return this.reposition();	},	toggleListeners: function(state) {		var fn = (state) ? 'addEvent' : 'removeEvent';		this.closeBtn[fn]('click', this.bound.close);		this.overlay[fn]('click', this.bound.close);		this.doc[fn]('keydown', this.bound.key)[fn]('mousewheel', this.bound.scroll);		this.doc.getWindow()[fn]('resize', this.bound.window)[fn]('scroll', this.bound.window);	},	toggleLoading: function(state) {		this.isLoading = state;		this.win[(state) ? 'addClass' : 'removeClass']('sbox-loading');		if (state) this.fireEvent('onLoading', [this.win]);	},	toggleOverlay: function(state) {		var full = this.doc.getSize().x;		this.overlay.setStyle('display', (state) ? '' : 'none');		this.doc.body[(state) ? 'addClass' : 'removeClass']('body-overlayed');		if (state) {			this.scrollOffset = this.doc.getWindow().getSize().x - full;			this.doc.body.setStyle('margin-right', this.scrollOffset);		} else {			this.doc.body.setStyle('margin-right', '');		}	},	showContent: function() {		if (this.content.get('opacity')) this.fireEvent('onShow', [this.win]);		this.fx.content.start(1);	},	hideContent: function() {		if (!this.content.get('opacity')) this.fireEvent('onHide', [this.win]);		this.fx.content.cancel().set(0);	},	onKey: function(e) {		switch (e.key) {			case 'esc': this.close(e);			case 'up': case 'down': return false;		}	},	checkTarget: function(e) {		return this.content.hasChild(e.target);	},	reposition: function() {		var size = this.doc.getSize(), scroll = this.doc.getScroll(), ssize = this.doc.getScrollSize();		this.overlay.setStyles({			width: ssize.x + 'px',			height: ssize.y + 'px'		});		this.win.setStyles({			left: (scroll.x + (size.x - this.win.offsetWidth) / 2 - this.scrollOffset).toInt() + 'px',			top: (scroll.y + (size.y - this.win.offsetHeight) / 2).toInt() + 'px'		});		return this.fireEvent('onMove', [this.overlay, this.win]);	},	removeEvents: function(type){		if (!this.$events) return this;		if (!type) this.$events = null;		else if (this.$events[type]) this.$events[type] = null;		return this;	},	extend: function(properties) {		return $extend(this, properties);	},	handlers: new Hash(),	parsers: new Hash()};SqueezeBox.extend(new Events($empty)).extend(new Options($empty)).extend(new Chain($empty));SqueezeBox.parsers.extend({	image: function(preset) {		return (preset || (/\.(?:jpg|png|gif)$/i).test(this.url)) ? this.url : false;	},	clone: function(preset) {		if ($(this.options.target)) return $(this.options.target);		if (this.element && !this.element.parentNode) return this.element;		var bits = this.url.match(/#([\w-]+)$/);		return (bits) ? $(bits[1]) : (preset ? this.element : false);	},	ajax: function(preset) {		return (preset || (this.url && !(/^(?:javascript|#)/i).test(this.url))) ? this.url : false;	},	iframe: function(preset) {		return (preset || this.url) ? this.url : false;	},	string: function(preset) {		return true;	}});SqueezeBox.handlers.extend({	image: function(url) {		var size, tmp = new Image();		this.asset = null;		tmp.onload = tmp.onabort = tmp.onerror = (function() {			tmp.onload = tmp.onabort = tmp.onerror = null;			if (!tmp.width) {				this.onError.delay(10, this);				return;			}			var box = this.doc.getSize();			box.x -= this.options.marginImage.x;			box.y -= this.options.marginImage.y;			size = {x: tmp.width, y: tmp.height};			for (var i = 2; i--;) {				if (size.x > box.x) {					size.y *= box.x / size.x;					size.x = box.x;				} else if (size.y > box.y) {					size.x *= box.y / size.y;					size.y = box.y;				}			}			size.x = size.x.toInt();			size.y = size.y.toInt();			this.asset = $(tmp);			tmp = null;			this.asset.width = size.x;			this.asset.height = size.y;			this.applyContent(this.asset, size);		}).bind(this);		tmp.src = url;		if (tmp && tmp.onload && tmp.complete) tmp.onload();		return (this.asset) ? [this.asset, size] : null;	},	clone: function(el) {		if (el) return el.clone();		return this.onError();	},	adopt: function(el) {		if (el) return el;		return this.onError();	},	ajax: function(url) {		var options = this.options.ajaxOptions || {};		this.asset = new Request.HTML($merge({			method: 'get',			evalScripts: false		}, this.options.ajaxOptions)).addEvents({			onSuccess: function(resp) {				this.applyContent(resp);				if (options.evalScripts !== null && !options.evalScripts) $exec(this.asset.response.javascript);				this.fireEvent('onAjax', [resp, this.asset]);				this.asset = null;			}.bind(this),			onFailure: this.onError.bind(this)		});		this.asset.send.delay(10, this.asset, [{url: url}]);	},	iframe: function(url) {		this.asset = new Element('iframe', $merge({			src: url,			frameBorder: 0,			width: this.options.size.x,			height: this.options.size.y		}, this.options.iframeOptions));		if (this.options.iframePreload) {			this.asset.addEvent('load', function() {				this.applyContent(this.asset.setStyle('display', ''));			}.bind(this));			this.asset.setStyle('display', 'none').inject(this.content);			return false;		}		return this.asset;	},	string: function(str) {		return str;	}});SqueezeBox.handlers.url = SqueezeBox.handlers.ajax;SqueezeBox.parsers.url = SqueezeBox.parsers.ajax;SqueezeBox.parsers.adopt = SqueezeBox.parsers.clone;
