/**
 * 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: 20, 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;