//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);

    }



});
