if(typeof (Adito)=="undefined"){
Adito={};
}
Adito.Detector={useragent:navigator.userAgent.toLowerCase(),DOM:document.all?(document.getElementById?2:1):(document.getElementById?4:(document.layers?3:0)),cached:null,getAgent:function(){
return this.useragent;
},isMac:function(_1){
var _2=_1||this.useragent;
return !!_2.match(/mac/i);
},isWin:function(_3){
var _4=_3||this.useragent;
return !!_4.match(/win/i);
},isWin2k:function(_5){
var _6=_5||this.useragent;
return this.isWin(_6)&&(_6.match(/nt\s*5/i));
},isWinVista:function(_7){
var _8=_7||this.useragent;
return this.isWin(_8)&&(_8.match(/nt\s*6/i));
},isWebKit:function(_9){
var _a=_9||this.useragent;
return !!_a.match(/AppleWebKit/i);
},isOpera:function(_b){
var _c=_b||this.useragent;
return !!_c.match(/opera/i);
},isIE:function(_d){
var _e=_d||this.useragent;
return !!_e.match(/msie/i);
},isIE6:function(_f){
var _10=_f||this.useragent;
return (_10.indexOf("msie 6")!=-1);
},isIE7:function(_11){
var _12=_11||this.useragent;
return (_12.indexOf("msie 7")!=-1);
},isOldIE:function(_13){
var _14=_13||this.useragent;
return !this.isIE7(_14);
},isMSIE:function(_15){
var _16=_15||this.useragent;
return this.isIE(_16)&&this.isWin(_16);
},isIE512:function(_17){
var _18=_17||this.useragent;
return (_18.indexOf("msie 5.12")!=-1);
},isOther:function(_19){
var _1a=_19||this.useragent;
return (_1a.indexOf("unix")!=-1||_1a.indexOf("sunos")!=-1||_1a.indexOf("bsd")!=-1||_1a.indexOf("x11")!=-1||_1a.indexOf("linux")!=-1);
},isIEStrict:function(_1b){
var _1c=_1b||this.useragent;
return _1c.match(/msie/i)&&!this.isOpera(_1c);
},isNewIE:function(_1d){
var _1e=_1d||this.useragent;
return (this.isIE(_1e)&&!this.isMac(_1e)&&(!_1e.match(/msie (5|6)/)));
},isFirefox:function(_1f){
var _20=_1f||this.useragent;
return !!_20.match(/firefox/i);
},onMobileFinished:function(_21,_22){
this.cached=_22.ismobiledevice;
},isMobile:function(_23){
var _24=_23||this.useragent;
if(this.cached==null){
new Ajax.YOComm("/remote/agent.php",{onComplete:this.onMobileFinished.bind(this),asynchronous:false,parameters:"agent="+encodeURIComponent(_24)});
}
return this.cached;
},isiPhone:function(_25){
var _26=_25||this.useragent;
return !!_26.match(/iPhone/i);
},isiTunesOK:function(_27){
var _28=_27||this.useragent;
return this.isMac(_28)||this.isWin2k(_28);
},isQTInstalled:function(){
var _29=false;
if(navigator.plugins&&navigator.plugins.length){
for(var i=0;i<navigator.plugins.length;i++){
var _2b=navigator.plugins[i];
if(_2b.name.indexOf("QuickTime")>-1){
_29=true;
}
}
}else{
qtObj=false;
execScript("on error resume next: qtObj = IsObject(CreateObject(\"QuickTimeCheckObject.QuickTimeCheck.1\"))","VBScript");
_29=qtObj;
}
return _29;
},getQTVersion:function(){
var _2c="0";
if(navigator.plugins&&navigator.plugins.length){
for(var i=0;i<navigator.plugins.length;i++){
var _2e=navigator.plugins[i];
var _2f=_2e.name.match(/quicktime\D*([\.\d]*)/i);
if(_2f&&_2f[1]){
_2c=_2f[1];
}
}
}else{
ieQTVersion=null;
execScript("on error resume next: ieQTVersion = CreateObject(\"QuickTimeCheckObject.QuickTimeCheck.1\").QuickTimeVersion","VBScript");
if(ieQTVersion){
_2c=(ieQTVersion>>24).toString(16);
}
}
return _2c;
},isQTCompatible:function(_30,_31){
function areCompatible(_32,_33){
var _34=parseInt(_32[0]);
if(isNaN(_34)){
_34=0;
}
var _35=parseInt(_33[0]);
if(isNaN(_35)){
_35=0;
}
if(_34==_35){
if(_32.length>1){
return areCompatible(_32.slice(1),_33.slice(1));
}else{
return true;
}
}else{
if(_34<_35){
return true;
}else{
return false;
}
}
}
var _36=_30.split(/\./);
var _37=_31?_31.split(/\./):this.getQTVersion().split(/\./);
return areCompatible(_36,_37);
},isValidQTAvailable:function(_38){
return this.isQTInstalled()&&this.isQTCompatible(_38);
}};

/*  Prototype JavaScript framework, version 1.6.1
 *  (c) 2005-2009 Sam Stephenson
 *
 *  Prototype is freely distributable under the terms of an MIT-style license.
 *  For details, see the Prototype web site: http://www.prototypejs.org/
 *
 *--------------------------------------------------------------------------*/

var Prototype = {
  Version: '1.6.1',

  Browser: (function(){
    var ua = navigator.userAgent;
    var isOpera = Object.prototype.toString.call(window.opera) == '[object Opera]';
    return {
      IE:             !!window.attachEvent && !isOpera,
      Opera:          isOpera,
      WebKit:         ua.indexOf('AppleWebKit/') > -1,
      Gecko:          ua.indexOf('Gecko') > -1 && ua.indexOf('KHTML') === -1,
      MobileSafari:   /Apple.*Mobile.*Safari/.test(ua)
    }
  })(),

  BrowserFeatures: {
    XPath: !!document.evaluate,
    SelectorsAPI: !!document.querySelector,
    ElementExtensions: (function() {
      var constructor = window.Element || window.HTMLElement;
      return !!(constructor && constructor.prototype);
    })(),
    SpecificElementExtensions: (function() {
      if (typeof window.HTMLDivElement !== 'undefined')
        return true;

      var div = document.createElement('div');
      var form = document.createElement('form');
      var isSupported = false;

      if (div['__proto__'] && (div['__proto__'] !== form['__proto__'])) {
        isSupported = true;
      }

      div = form = null;

      return isSupported;
    })()
  },

  ScriptFragment: '<script[^>]*>([\\S\\s]*?)<\/script>',
  JSONFilter: /^\/\*-secure-([\s\S]*)\*\/\s*$/,

  emptyFunction: function() { },
  K: function(x) { return x }
};

if (Prototype.Browser.MobileSafari)
  Prototype.BrowserFeatures.SpecificElementExtensions = false;


var Abstract = { };


var Try = {
  these: function() {
    var returnValue;

    for (var i = 0, length = arguments.length; i < length; i++) {
      var lambda = arguments[i];
      try {
        returnValue = lambda();
        break;
      } catch (e) { }
    }

    return returnValue;
  }
};

/* Based on Alex Arnell's inheritance implementation. */

var Class = (function() {
  function subclass() {};
  function create() {
    var parent = null, properties = $A(arguments);
    if (Object.isFunction(properties[0]))
      parent = properties.shift();

    function klass() {
      this.initialize.apply(this, arguments);
    }

    Object.extend(klass, Class.Methods);
    klass.superclass = parent;
    klass.subclasses = [];

    if (parent) {
      subclass.prototype = parent.prototype;
      klass.prototype = new subclass;
      parent.subclasses.push(klass);
    }

    for (var i = 0; i < properties.length; i++)
      klass.addMethods(properties[i]);

    if (!klass.prototype.initialize)
      klass.prototype.initialize = Prototype.emptyFunction;

    klass.prototype.constructor = klass;
    return klass;
  }

  function addMethods(source) {
    var ancestor   = this.superclass && this.superclass.prototype;
    var properties = Object.keys(source);

    if (!Object.keys({ toString: true }).length) {
      if (source.toString != Object.prototype.toString)
        properties.push("toString");
      if (source.valueOf != Object.prototype.valueOf)
        properties.push("valueOf");
    }

    for (var i = 0, length = properties.length; i < length; i++) {
      var property = properties[i], value = source[property];
      if (ancestor && Object.isFunction(value) &&
          value.argumentNames().first() == "$super") {
        var method = value;
        value = (function(m) {
          return function() { return ancestor[m].apply(this, arguments); };
        })(property).wrap(method);

        value.valueOf = method.valueOf.bind(method);
        value.toString = method.toString.bind(method);
      }
      this.prototype[property] = value;
    }

    return this;
  }

  return {
    create: create,
    Methods: {
      addMethods: addMethods
    }
  };
})();
(function() {

  var _toString = Object.prototype.toString;

  function extend(destination, source) {
    for (var property in source)
      destination[property] = source[property];
    return destination;
  }

  function inspect(object) {
    try {
      if (isUndefined(object)) return 'undefined';
      if (object === null) return 'null';
      return object.inspect ? object.inspect() : String(object);
    } catch (e) {
      if (e instanceof RangeError) return '...';
      throw e;
    }
  }

  function toJSON(object) {
    var type = typeof object;
    switch (type) {
      case 'undefined':
      case 'function':
      case 'unknown': return;
      case 'boolean': return object.toString();
    }

    if (object === null) return 'null';
    if (object.toJSON) return object.toJSON();
    if (isElement(object)) return;

    var results = [];
    for (var property in object) {
      var value = toJSON(object[property]);
      if (!isUndefined(value))
        results.push(property.toJSON() + ': ' + value);
    }

    return '{' + results.join(', ') + '}';
  }

  function toQueryString(object) {
    return $H(object).toQueryString();
  }

  function toHTML(object) {
    return object && object.toHTML ? object.toHTML() : String.interpret(object);
  }

  function keys(object) {
    var results = [];
    for (var property in object)
      results.push(property);
    return results;
  }

  function values(object) {
    var results = [];
    for (var property in object)
      results.push(object[property]);
    return results;
  }

  function clone(object) {
    return extend({ }, object);
  }

  function isElement(object) {
    return !!(object && object.nodeType == 1);
  }

  function isArray(object) {
    return _toString.call(object) == "[object Array]";
  }


  function isHash(object) {
    return object instanceof Hash;
  }

  function isFunction(object) {
    return typeof object === "function";
  }

  function isString(object) {
    return _toString.call(object) == "[object String]";
  }

  function isNumber(object) {
    return _toString.call(object) == "[object Number]";
  }

  function isUndefined(object) {
    return typeof object === "undefined";
  }

  extend(Object, {
    extend:        extend,
    inspect:       inspect,
    toJSON:        toJSON,
    toQueryString: toQueryString,
    toHTML:        toHTML,
    keys:          keys,
    values:        values,
    clone:         clone,
    isElement:     isElement,
    isArray:       isArray,
    isHash:        isHash,
    isFunction:    isFunction,
    isString:      isString,
    isNumber:      isNumber,
    isUndefined:   isUndefined
  });
})();
Object.extend(Function.prototype, (function() {
  var slice = Array.prototype.slice;

  function update(array, args) {
    var arrayLength = array.length, length = args.length;
    while (length--) array[arrayLength + length] = args[length];
    return array;
  }

  function merge(array, args) {
    array = slice.call(array, 0);
    return update(array, args);
  }

  function argumentNames() {
    var names = this.toString().match(/^[\s\(]*function[^(]*\(([^)]*)\)/)[1]
      .replace(/\/\/.*?[\r\n]|\/\*(?:.|[\r\n])*?\*\//g, '')
      .replace(/\s+/g, '').split(',');
    return names.length == 1 && !names[0] ? [] : names;
  }

  function bind(context) {
    if (arguments.length < 2 && Object.isUndefined(arguments[0])) return this;
    var __method = this, args = slice.call(arguments, 1);
    return function() {
      var a = merge(args, arguments);
      return __method.apply(context, a);
    }
  }

  function bindAsEventListener(context) {
    var __method = this, args = slice.call(arguments, 1);
    return function(event) {
      var a = update([event || window.event], args);
      return __method.apply(context, a);
    }
  }

  function curry() {
    if (!arguments.length) return this;
    var __method = this, args = slice.call(arguments, 0);
    return function() {
      var a = merge(args, arguments);
      return __method.apply(this, a);
    }
  }

  function delay(timeout) {
    var __method = this, args = slice.call(arguments, 1);
    timeout = timeout * 1000
    return window.setTimeout(function() {
      return __method.apply(__method, args);
    }, timeout);
  }

  function defer() {
    var args = update([0.01], arguments);
    return this.delay.apply(this, args);
  }

  function wrap(wrapper) {
    var __method = this;
    return function() {
      var a = update([__method.bind(this)], arguments);
      return wrapper.apply(this, a);
    }
  }

  function methodize() {
    if (this._methodized) return this._methodized;
    var __method = this;
    return this._methodized = function() {
      var a = update([this], arguments);
      return __method.apply(null, a);
    };
  }

  return {
    argumentNames:       argumentNames,
    bind:                bind,
    bindAsEventListener: bindAsEventListener,
    curry:               curry,
    delay:               delay,
    defer:               defer,
    wrap:                wrap,
    methodize:           methodize
  }
})());


Date.prototype.toJSON = function() {
  return '"' + this.getUTCFullYear() + '-' +
    (this.getUTCMonth() + 1).toPaddedString(2) + '-' +
    this.getUTCDate().toPaddedString(2) + 'T' +
    this.getUTCHours().toPaddedString(2) + ':' +
    this.getUTCMinutes().toPaddedString(2) + ':' +
    this.getUTCSeconds().toPaddedString(2) + 'Z"';
};


RegExp.prototype.match = RegExp.prototype.test;

RegExp.escape = function(str) {
  return String(str).replace(/([.*+?^=!:${}()|[\]\/\\])/g, '\\$1');
};
var PeriodicalExecuter = Class.create({
  initialize: function(callback, frequency) {
    this.callback = callback;
    this.frequency = frequency;
    this.currentlyExecuting = false;

    this.registerCallback();
  },

  registerCallback: function() {
    this.timer = setInterval(this.onTimerEvent.bind(this), this.frequency * 1000);
  },

  execute: function() {
    this.callback(this);
  },

  stop: function() {
    if (!this.timer) return;
    clearInterval(this.timer);
    this.timer = null;
  },

  onTimerEvent: function() {
    if (!this.currentlyExecuting) {
      try {
        this.currentlyExecuting = true;
        this.execute();
        this.currentlyExecuting = false;
      } catch(e) {
        this.currentlyExecuting = false;
        throw e;
      }
    }
  }
});
Object.extend(String, {
  interpret: function(value) {
    return value == null ? '' : String(value);
  },
  specialChar: {
    '\b': '\\b',
    '\t': '\\t',
    '\n': '\\n',
    '\f': '\\f',
    '\r': '\\r',
    '\\': '\\\\'
  }
});

Object.extend(String.prototype, (function() {

  function prepareReplacement(replacement) {
    if (Object.isFunction(replacement)) return replacement;
    var template = new Template(replacement);
    return function(match) { return template.evaluate(match) };
  }

  function gsub(pattern, replacement) {
    var result = '', source = this, match;
    replacement = prepareReplacement(replacement);

    if (Object.isString(pattern))
      pattern = RegExp.escape(pattern);

    if (!(pattern.length || pattern.source)) {
      replacement = replacement('');
      return replacement + source.split('').join(replacement) + replacement;
    }

    while (source.length > 0) {
      if (match = source.match(pattern)) {
        result += source.slice(0, match.index);
        result += String.interpret(replacement(match));
        source  = source.slice(match.index + match[0].length);
      } else {
        result += source, source = '';
      }
    }
    return result;
  }

  function sub(pattern, replacement, count) {
    replacement = prepareReplacement(replacement);
    count = Object.isUndefined(count) ? 1 : count;

    return this.gsub(pattern, function(match) {
      if (--count < 0) return match[0];
      return replacement(match);
    });
  }

  function scan(pattern, iterator) {
    this.gsub(pattern, iterator);
    return String(this);
  }

  function truncate(length, truncation) {
    length = length || 30;
    truncation = Object.isUndefined(truncation) ? '...' : truncation;
    return this.length > length ?
      this.slice(0, length - truncation.length) + truncation : String(this);
  }

  function strip() {
    return this.replace(/^\s+/, '').replace(/\s+$/, '');
  }

  function stripTags() {
    return this.replace(/<\w+(\s+("[^"]*"|'[^']*'|[^>])+)?>|<\/\w+>/gi, '');
  }

  function stripScripts() {
    return this.replace(new RegExp(Prototype.ScriptFragment, 'img'), '');
  }

  function extractScripts() {
    var matchAll = new RegExp(Prototype.ScriptFragment, 'img');
    var matchOne = new RegExp(Prototype.ScriptFragment, 'im');
    return (this.match(matchAll) || []).map(function(scriptTag) {
      return (scriptTag.match(matchOne) || ['', ''])[1];
    });
  }

  function evalScripts() {
    return this.extractScripts().map(function(script) { return eval(script) });
  }

  function escapeHTML() {
    return this.replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;');
  }

  function unescapeHTML() {
    return this.stripTags().replace(/&lt;/g,'<').replace(/&gt;/g,'>').replace(/&amp;/g,'&');
  }


  function toQueryParams(separator) {
    var match = this.strip().match(/([^?#]*)(#.*)?$/);
    if (!match) return { };

    return match[1].split(separator || '&').inject({ }, function(hash, pair) {
      if ((pair = pair.split('='))[0]) {
        var key = decodeURIComponent(pair.shift());
        var value = pair.length > 1 ? pair.join('=') : pair[0];
        if (value != undefined) value = decodeURIComponent(value);

        if (key in hash) {
          if (!Object.isArray(hash[key])) hash[key] = [hash[key]];
          hash[key].push(value);
        }
        else hash[key] = value;
      }
      return hash;
    });
  }

  function toArray() {
    return this.split('');
  }

  function succ() {
    return this.slice(0, this.length - 1) +
      String.fromCharCode(this.charCodeAt(this.length - 1) + 1);
  }

  function times(count) {
    return count < 1 ? '' : new Array(count + 1).join(this);
  }

  function camelize() {
    var parts = this.split('-'), len = parts.length;
    if (len == 1) return parts[0];

    var camelized = this.charAt(0) == '-'
      ? parts[0].charAt(0).toUpperCase() + parts[0].substring(1)
      : parts[0];

    for (var i = 1; i < len; i++)
      camelized += parts[i].charAt(0).toUpperCase() + parts[i].substring(1);

    return camelized;
  }

  function capitalize() {
    return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase();
  }

  function underscore() {
    return this.replace(/::/g, '/')
               .replace(/([A-Z]+)([A-Z][a-z])/g, '$1_$2')
               .replace(/([a-z\d])([A-Z])/g, '$1_$2')
               .replace(/-/g, '_')
               .toLowerCase();
  }

  function dasherize() {
    return this.replace(/_/g, '-');
  }

  function inspect(useDoubleQuotes) {
    var escapedString = this.replace(/[\x00-\x1f\\]/g, function(character) {
      if (character in String.specialChar) {
        return String.specialChar[character];
      }
      return '\\u00' + character.charCodeAt().toPaddedString(2, 16);
    });
    if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"';
    return "'" + escapedString.replace(/'/g, '\\\'') + "'";
  }

  function toJSON() {
    return this.inspect(true);
  }

  function unfilterJSON(filter) {
    return this.replace(filter || Prototype.JSONFilter, '$1');
  }

  function isJSON() {
    var str = this;
    if (str.blank()) return false;
    str = this.replace(/\\./g, '@').replace(/"[^"\\\n\r]*"/g, '');
    return (/^[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t]*$/).test(str);
  }

  function evalJSON(sanitize) {
    var json = this.unfilterJSON();
    try {
      if (!sanitize || json.isJSON()) return eval('(' + json + ')');
    } catch (e) { }
    throw new SyntaxError('Badly formed JSON string: ' + this.inspect());
  }

  function include(pattern) {
    return this.indexOf(pattern) > -1;
  }

  function startsWith(pattern) {
    return this.indexOf(pattern) === 0;
  }

  function endsWith(pattern) {
    var d = this.length - pattern.length;
    return d >= 0 && this.lastIndexOf(pattern) === d;
  }

  function empty() {
    return this == '';
  }

  function blank() {
    return /^\s*$/.test(this);
  }

  function interpolate(object, pattern) {
    return new Template(this, pattern).evaluate(object);
  }

  return {
    gsub:           gsub,
    sub:            sub,
    scan:           scan,
    truncate:       truncate,
    strip:          String.prototype.trim ? String.prototype.trim : strip,
    stripTags:      stripTags,
    stripScripts:   stripScripts,
    extractScripts: extractScripts,
    evalScripts:    evalScripts,
    escapeHTML:     escapeHTML,
    unescapeHTML:   unescapeHTML,
    toQueryParams:  toQueryParams,
    parseQuery:     toQueryParams,
    toArray:        toArray,
    succ:           succ,
    times:          times,
    camelize:       camelize,
    capitalize:     capitalize,
    underscore:     underscore,
    dasherize:      dasherize,
    inspect:        inspect,
    toJSON:         toJSON,
    unfilterJSON:   unfilterJSON,
    isJSON:         isJSON,
    evalJSON:       evalJSON,
    include:        include,
    startsWith:     startsWith,
    endsWith:       endsWith,
    empty:          empty,
    blank:          blank,
    interpolate:    interpolate
  };
})());

var Template = Class.create({
  initialize: function(template, pattern) {
    this.template = template.toString();
    this.pattern = pattern || Template.Pattern;
  },

  evaluate: function(object) {
    if (object && Object.isFunction(object.toTemplateReplacements))
      object = object.toTemplateReplacements();

    return this.template.gsub(this.pattern, function(match) {
      if (object == null) return (match[1] + '');

      var before = match[1] || '';
      if (before == '\\') return match[2];

      var ctx = object, expr = match[3];
      var pattern = /^([^.[]+|\[((?:.*?[^\\])?)\])(\.|\[|$)/;
      match = pattern.exec(expr);
      if (match == null) return before;

      while (match != null) {
        var comp = match[1].startsWith('[') ? match[2].replace(/\\\\]/g, ']') : match[1];
        ctx = ctx[comp];
        if (null == ctx || '' == match[3]) break;
        expr = expr.substring('[' == match[3] ? match[1].length : match[0].length);
        match = pattern.exec(expr);
      }

      return before + String.interpret(ctx);
    });
  }
});
Template.Pattern = /(^|.|\r|\n)(#\{(.*?)\})/;

var $break = { };

var Enumerable = (function() {
  function each(iterator, context) {
    var index = 0;
    try {
      this._each(function(value) {
        iterator.call(context, value, index++);
      });
    } catch (e) {
      if (e != $break) throw e;
    }
    return this;
  }

  function eachSlice(number, iterator, context) {
    var index = -number, slices = [], array = this.toArray();
    if (number < 1) return array;
    while ((index += number) < array.length)
      slices.push(array.slice(index, index+number));
    return slices.collect(iterator, context);
  }

  function all(iterator, context) {
    iterator = iterator || Prototype.K;
    var result = true;
    this.each(function(value, index) {
      result = result && !!iterator.call(context, value, index);
      if (!result) throw $break;
    });
    return result;
  }

  function any(iterator, context) {
    iterator = iterator || Prototype.K;
    var result = false;
    this.each(function(value, index) {
      if (result = !!iterator.call(context, value, index))
        throw $break;
    });
    return result;
  }

  function collect(iterator, context) {
    iterator = iterator || Prototype.K;
    var results = [];
    this.each(function(value, index) {
      results.push(iterator.call(context, value, index));
    });
    return results;
  }

  function detect(iterator, context) {
    var result;
    this.each(function(value, index) {
      if (iterator.call(context, value, index)) {
        result = value;
        throw $break;
      }
    });
    return result;
  }

  function findAll(iterator, context) {
    var results = [];
    this.each(function(value, index) {
      if (iterator.call(context, value, index))
        results.push(value);
    });
    return results;
  }

  function grep(filter, iterator, context) {
    iterator = iterator || Prototype.K;
    var results = [];

    if (Object.isString(filter))
      filter = new RegExp(RegExp.escape(filter));

    this.each(function(value, index) {
      if (filter.match(value))
        results.push(iterator.call(context, value, index));
    });
    return results;
  }

  function include(object) {
    if (Object.isFunction(this.indexOf))
      if (this.indexOf(object) != -1) return true;

    var found = false;
    this.each(function(value) {
      if (value == object) {
        found = true;
        throw $break;
      }
    });
    return found;
  }

  function inGroupsOf(number, fillWith) {
    fillWith = Object.isUndefined(fillWith) ? null : fillWith;
    return this.eachSlice(number, function(slice) {
      while(slice.length < number) slice.push(fillWith);
      return slice;
    });
  }

  function inject(memo, iterator, context) {
    this.each(function(value, index) {
      memo = iterator.call(context, memo, value, index);
    });
    return memo;
  }

  function invoke(method) {
    var args = $A(arguments).slice(1);
    return this.map(function(value) {
      return value[method].apply(value, args);
    });
  }

  function max(iterator, context) {
    iterator = iterator || Prototype.K;
    var result;
    this.each(function(value, index) {
      value = iterator.call(context, value, index);
      if (result == null || value >= result)
        result = value;
    });
    return result;
  }

  function min(iterator, context) {
    iterator = iterator || Prototype.K;
    var result;
    this.each(function(value, index) {
      value = iterator.call(context, value, index);
      if (result == null || value < result)
        result = value;
    });
    return result;
  }

  function partition(iterator, context) {
    iterator = iterator || Prototype.K;
    var trues = [], falses = [];
    this.each(function(value, index) {
      (iterator.call(context, value, index) ?
        trues : falses).push(value);
    });
    return [trues, falses];
  }

  function pluck(property) {
    var results = [];
    this.each(function(value) {
      results.push(value[property]);
    });
    return results;
  }

  function reject(iterator, context) {
    var results = [];
    this.each(function(value, index) {
      if (!iterator.call(context, value, index))
        results.push(value);
    });
    return results;
  }

  function sortBy(iterator, context) {
    return this.map(function(value, index) {
      return {
        value: value,
        criteria: iterator.call(context, value, index)
      };
    }).sort(function(left, right) {
      var a = left.criteria, b = right.criteria;
      return a < b ? -1 : a > b ? 1 : 0;
    }).pluck('value');
  }

  function toArray() {
    return this.map();
  }

  function zip() {
    var iterator = Prototype.K, args = $A(arguments);
    if (Object.isFunction(args.last()))
      iterator = args.pop();

    var collections = [this].concat(args).map($A);
    return this.map(function(value, index) {
      return iterator(collections.pluck(index));
    });
  }

  function size() {
    return this.toArray().length;
  }

  function inspect() {
    return '#<Enumerable:' + this.toArray().inspect() + '>';
  }









  return {
    each:       each,
    eachSlice:  eachSlice,
    all:        all,
    every:      all,
    any:        any,
    some:       any,
    collect:    collect,
    map:        collect,
    detect:     detect,
    findAll:    findAll,
    select:     findAll,
    filter:     findAll,
    grep:       grep,
    include:    include,
    member:     include,
    inGroupsOf: inGroupsOf,
    inject:     inject,
    invoke:     invoke,
    max:        max,
    min:        min,
    partition:  partition,
    pluck:      pluck,
    reject:     reject,
    sortBy:     sortBy,
    toArray:    toArray,
    entries:    toArray,
    zip:        zip,
    size:       size,
    inspect:    inspect,
    find:       detect
  };
})();
function $A(iterable) {
  if (!iterable) return [];
  if ('toArray' in Object(iterable)) return iterable.toArray();
  var length = iterable.length || 0, results = new Array(length);
  while (length--) results[length] = iterable[length];
  return results;
}

function $w(string) {
  if (!Object.isString(string)) return [];
  string = string.strip();
  return string ? string.split(/\s+/) : [];
}

Array.from = $A;


(function() {
  var arrayProto = Array.prototype,
      slice = arrayProto.slice,
      _each = arrayProto.forEach; // use native browser JS 1.6 implementation if available

  function each(iterator) {
    for (var i = 0, length = this.length; i < length; i++)
      iterator(this[i]);
  }
  if (!_each) _each = each;

  function clear() {
    this.length = 0;
    return this;
  }

  function first() {
    return this[0];
  }

  function last() {
    return this[this.length - 1];
  }

  function compact() {
    return this.select(function(value) {
      return value != null;
    });
  }

  function flatten() {
    return this.inject([], function(array, value) {
      if (Object.isArray(value))
        return array.concat(value.flatten());
      array.push(value);
      return array;
    });
  }

  function without() {
    var values = slice.call(arguments, 0);
    return this.select(function(value) {
      return !values.include(value);
    });
  }

  function reverse(inline) {
    return (inline !== false ? this : this.toArray())._reverse();
  }

  function uniq(sorted) {
    return this.inject([], function(array, value, index) {
      if (0 == index || (sorted ? array.last() != value : !array.include(value)))
        array.push(value);
      return array;
    });
  }

  function intersect(array) {
    return this.uniq().findAll(function(item) {
      return array.detect(function(value) { return item === value });
    });
  }


  function clone() {
    return slice.call(this, 0);
  }

  function size() {
    return this.length;
  }

  function inspect() {
    return '[' + this.map(Object.inspect).join(', ') + ']';
  }

  function toJSON() {
    var results = [];
    this.each(function(object) {
      var value = Object.toJSON(object);
      if (!Object.isUndefined(value)) results.push(value);
    });
    return '[' + results.join(', ') + ']';
  }

  function indexOf(item, i) {
    i || (i = 0);
    var length = this.length;
    if (i < 0) i = length + i;
    for (; i < length; i++)
      if (this[i] === item) return i;
    return -1;
  }

  function lastIndexOf(item, i) {
    i = isNaN(i) ? this.length : (i < 0 ? this.length + i : i) + 1;
    var n = this.slice(0, i).reverse().indexOf(item);
    return (n < 0) ? n : i - n - 1;
  }

  function concat() {
    var array = slice.call(this, 0), item;
    for (var i = 0, length = arguments.length; i < length; i++) {
      item = arguments[i];
      if (Object.isArray(item) && !('callee' in item)) {
        for (var j = 0, arrayLength = item.length; j < arrayLength; j++)
          array.push(item[j]);
      } else {
        array.push(item);
      }
    }
    return array;
  }

  Object.extend(arrayProto, Enumerable);

  if (!arrayProto._reverse)
    arrayProto._reverse = arrayProto.reverse;

  Object.extend(arrayProto, {
    _each:     _each,
    clear:     clear,
    first:     first,
    last:      last,
    compact:   compact,
    flatten:   flatten,
    without:   without,
    reverse:   reverse,
    uniq:      uniq,
    intersect: intersect,
    clone:     clone,
    toArray:   clone,
    size:      size,
    inspect:   inspect,
    toJSON:    toJSON
  });

  var CONCAT_ARGUMENTS_BUGGY = (function() {
    return [].concat(arguments)[0][0] !== 1;
  })(1,2)

  if (CONCAT_ARGUMENTS_BUGGY) arrayProto.concat = concat;

  if (!arrayProto.indexOf) arrayProto.indexOf = indexOf;
  if (!arrayProto.lastIndexOf) arrayProto.lastIndexOf = lastIndexOf;
})();
function $H(object) {
  return new Hash(object);
};

var Hash = Class.create(Enumerable, (function() {
  function initialize(object) {
    this._object = Object.isHash(object) ? object.toObject() : Object.clone(object);
  }

  function _each(iterator) {
    for (var key in this._object) {
      var value = this._object[key], pair = [key, value];
      pair.key = key;
      pair.value = value;
      iterator(pair);
    }
  }

  function set(key, value) {
    return this._object[key] = value;
  }

  function get(key) {
    if (this._object[key] !== Object.prototype[key])
      return this._object[key];
  }

  function unset(key) {
    var value = this._object[key];
    delete this._object[key];
    return value;
  }

  function toObject() {
    return Object.clone(this._object);
  }

  function keys() {
    return this.pluck('key');
  }

  function values() {
    return this.pluck('value');
  }

  function index(value) {
    var match = this.detect(function(pair) {
      return pair.value === value;
    });
    return match && match.key;
  }

  function merge(object) {
    return this.clone().update(object);
  }

  function update(object) {
    return new Hash(object).inject(this, function(result, pair) {
      result.set(pair.key, pair.value);
      return result;
    });
  }

  function toQueryPair(key, value) {
    if (Object.isUndefined(value)) return key;
    return key + '=' + encodeURIComponent(String.interpret(value));
  }

  function toQueryString() {
    return this.inject([], function(results, pair) {
      var key = encodeURIComponent(pair.key), values = pair.value;

      if (values && typeof values == 'object') {
        if (Object.isArray(values))
          return results.concat(values.map(toQueryPair.curry(key)));
      } else results.push(toQueryPair(key, values));
      return results;
    }).join('&');
  }

  function inspect() {
    return '#<Hash:{' + this.map(function(pair) {
      return pair.map(Object.inspect).join(': ');
    }).join(', ') + '}>';
  }

  function toJSON() {
    return Object.toJSON(this.toObject());
  }

  function clone() {
    return new Hash(this);
  }

  return {
    initialize:             initialize,
    _each:                  _each,
    set:                    set,
    get:                    get,
    unset:                  unset,
    toObject:               toObject,
    toTemplateReplacements: toObject,
    keys:                   keys,
    values:                 values,
    index:                  index,
    merge:                  merge,
    update:                 update,
    toQueryString:          toQueryString,
    inspect:                inspect,
    toJSON:                 toJSON,
    clone:                  clone
  };
})());

Hash.from = $H;
Object.extend(Number.prototype, (function() {
  function toColorPart() {
    return this.toPaddedString(2, 16);
  }

  function succ() {
    return this + 1;
  }

  function times(iterator, context) {
    $R(0, this, true).each(iterator, context);
    return this;
  }

  function toPaddedString(length, radix) {
    var string = this.toString(radix || 10);
    return '0'.times(length - string.length) + string;
  }

  function toJSON() {
    return isFinite(this) ? this.toString() : 'null';
  }

  function abs() {
    return Math.abs(this);
  }

  function round() {
    return Math.round(this);
  }

  function ceil() {
    return Math.ceil(this);
  }

  function floor() {
    return Math.floor(this);
  }

  return {
    toColorPart:    toColorPart,
    succ:           succ,
    times:          times,
    toPaddedString: toPaddedString,
    toJSON:         toJSON,
    abs:            abs,
    round:          round,
    ceil:           ceil,
    floor:          floor
  };
})());

function $R(start, end, exclusive) {
  return new ObjectRange(start, end, exclusive);
}

var ObjectRange = Class.create(Enumerable, (function() {
  function initialize(start, end, exclusive) {
    this.start = start;
    this.end = end;
    this.exclusive = exclusive;
  }

  function _each(iterator) {
    var value = this.start;
    while (this.include(value)) {
      iterator(value);
      value = value.succ();
    }
  }

  function include(value) {
    if (value < this.start)
      return false;
    if (this.exclusive)
      return value < this.end;
    return value <= this.end;
  }

  return {
    initialize: initialize,
    _each:      _each,
    include:    include
  };
})());



var Ajax = {
  getTransport: function() {
    return Try.these(
      function() {return new XMLHttpRequest()},
      function() {return new ActiveXObject('Msxml2.XMLHTTP')},
      function() {return new ActiveXObject('Microsoft.XMLHTTP')}
    ) || false;
  },

  activeRequestCount: 0
};

Ajax.Responders = {
  responders: [],

  _each: function(iterator) {
    this.responders._each(iterator);
  },

  register: function(responder) {
    if (!this.include(responder))
      this.responders.push(responder);
  },

  unregister: function(responder) {
    this.responders = this.responders.without(responder);
  },

  dispatch: function(callback, request, transport, json) {
    this.each(function(responder) {
      if (Object.isFunction(responder[callback])) {
        try {
          responder[callback].apply(responder, [request, transport, json]);
        } catch (e) { }
      }
    });
  }
};

Object.extend(Ajax.Responders, Enumerable);

Ajax.Responders.register({
  onCreate:   function() { Ajax.activeRequestCount++ },
  onComplete: function() { Ajax.activeRequestCount-- }
});
Ajax.Base = Class.create({
  initialize: function(options) {
    this.options = {
      method:       'post',
      asynchronous: true,
      contentType:  'application/x-www-form-urlencoded',
      encoding:     'UTF-8',
      parameters:   '',
      evalJSON:     true,
      evalJS:       true
    };
    Object.extend(this.options, options || { });

    this.options.method = this.options.method.toLowerCase();

    if (Object.isString(this.options.parameters))
      this.options.parameters = this.options.parameters.toQueryParams();
    else if (Object.isHash(this.options.parameters))
      this.options.parameters = this.options.parameters.toObject();
  }
});
Ajax.Request = Class.create(Ajax.Base, {
  _complete: false,

  initialize: function($super, url, options) {
    $super(options);
    this.transport = Ajax.getTransport();
    this.request(url);
  },

  request: function(url) {
    this.url = url;
    this.method = this.options.method;
    var params = Object.clone(this.options.parameters);

    if (!['get', 'post'].include(this.method)) {
      params['_method'] = this.method;
      this.method = 'post';
    }

    this.parameters = params;

    if (params = Object.toQueryString(params)) {
      if (this.method == 'get')
        this.url += (this.url.include('?') ? '&' : '?') + params;
      else if (/Konqueror|Safari|KHTML/.test(navigator.userAgent))
        params += '&_=';
    }

    try {
      var response = new Ajax.Response(this);
      if (this.options.onCreate) this.options.onCreate(response);
      Ajax.Responders.dispatch('onCreate', this, response);

      this.transport.open(this.method.toUpperCase(), this.url,
        this.options.asynchronous);

      if (this.options.asynchronous) this.respondToReadyState.bind(this).defer(1);

      this.transport.onreadystatechange = this.onStateChange.bind(this);
      this.setRequestHeaders();

      this.body = this.method == 'post' ? (this.options.postBody || params) : null;
      this.transport.send(this.body);

      /* Force Firefox to handle ready state 4 for synchronous requests */
      if (!this.options.asynchronous && this.transport.overrideMimeType)
        this.onStateChange();

    }
    catch (e) {
      this.dispatchException(e);
    }
  },

  onStateChange: function() {
    var readyState = this.transport.readyState;
    if (readyState > 1 && !((readyState == 4) && this._complete))
      this.respondToReadyState(this.transport.readyState);
  },

  setRequestHeaders: function() {
    var headers = {
      'X-Requested-With': 'XMLHttpRequest',
      'X-Prototype-Version': Prototype.Version,
      'Accept': 'text/javascript, text/html, application/xml, text/xml, */*'
    };

    if (this.method == 'post') {
      headers['Content-type'] = this.options.contentType +
        (this.options.encoding ? '; charset=' + this.options.encoding : '');

      /* Force "Connection: close" for older Mozilla browsers to work
       * around a bug where XMLHttpRequest sends an incorrect
       * Content-length header. See Mozilla Bugzilla #246651.
       */
      if (this.transport.overrideMimeType &&
          (navigator.userAgent.match(/Gecko\/(\d{4})/) || [0,2005])[1] < 2005)
            headers['Connection'] = 'close';
    }

    if (typeof this.options.requestHeaders == 'object') {
      var extras = this.options.requestHeaders;

      if (Object.isFunction(extras.push))
        for (var i = 0, length = extras.length; i < length; i += 2)
          headers[extras[i]] = extras[i+1];
      else
        $H(extras).each(function(pair) { headers[pair.key] = pair.value });
    }

    for (var name in headers)
      this.transport.setRequestHeader(name, headers[name]);
  },

  success: function() {
    var status = this.getStatus();
    return !status || (status >= 200 && status < 300);
  },

  getStatus: function() {
    try {
      return this.transport.status || 0;
    } catch (e) { return 0 }
  },

  respondToReadyState: function(readyState) {
    var state = Ajax.Request.Events[readyState], response = new Ajax.Response(this);

    if (state == 'Complete') {
      try {
        this._complete = true;
        (this.options['on' + response.status]
         || this.options['on' + (this.success() ? 'Success' : 'Failure')]
         || Prototype.emptyFunction)(response, response.headerJSON);
      } catch (e) {
        this.dispatchException(e);
      }

      var contentType = response.getHeader('Content-type');
      if (this.options.evalJS == 'force'
          || (this.options.evalJS && this.isSameOrigin() && contentType
          && contentType.match(/^\s*(text|application)\/(x-)?(java|ecma)script(;.*)?\s*$/i)))
        this.evalResponse();
    }

    try {
      (this.options['on' + state] || Prototype.emptyFunction)(response, response.headerJSON);
      Ajax.Responders.dispatch('on' + state, this, response, response.headerJSON);
    } catch (e) {
      this.dispatchException(e);
    }

    if (state == 'Complete') {
      this.transport.onreadystatechange = Prototype.emptyFunction;
    }
  },

  isSameOrigin: function() {
    var m = this.url.match(/^\s*https?:\/\/[^\/]*/);
    return !m || (m[0] == '#{protocol}//#{domain}#{port}'.interpolate({
      protocol: location.protocol,
      domain: document.domain,
      port: location.port ? ':' + location.port : ''
    }));
  },

  getHeader: function(name) {
    try {
      return this.transport.getResponseHeader(name) || null;
    } catch (e) { return null; }
  },

  evalResponse: function() {
    try {
      return eval((this.transport.responseText || '').unfilterJSON());
    } catch (e) {
      this.dispatchException(e);
    }
  },

  dispatchException: function(exception) {
    (this.options.onException || Prototype.emptyFunction)(this, exception);
    Ajax.Responders.dispatch('onException', this, exception);
  }
});

Ajax.Request.Events =
  ['Uninitialized', 'Loading', 'Loaded', 'Interactive', 'Complete'];








Ajax.Response = Class.create({
  initialize: function(request){
    this.request = request;
    var transport  = this.transport  = request.transport,
        readyState = this.readyState = transport.readyState;

    if((readyState > 2 && !Prototype.Browser.IE) || readyState == 4) {
      this.status       = this.getStatus();
      this.statusText   = this.getStatusText();
      this.responseText = String.interpret(transport.responseText);
      this.headerJSON   = this._getHeaderJSON();
    }

    if(readyState == 4) {
      var xml = transport.responseXML;
      this.responseXML  = Object.isUndefined(xml) ? null : xml;
      this.responseJSON = this._getResponseJSON();
    }
  },

  status:      0,

  statusText: '',

  getStatus: Ajax.Request.prototype.getStatus,

  getStatusText: function() {
    try {
      return this.transport.statusText || '';
    } catch (e) { return '' }
  },

  getHeader: Ajax.Request.prototype.getHeader,

  getAllHeaders: function() {
    try {
      return this.getAllResponseHeaders();
    } catch (e) { return null }
  },

  getResponseHeader: function(name) {
    return this.transport.getResponseHeader(name);
  },

  getAllResponseHeaders: function() {
    return this.transport.getAllResponseHeaders();
  },

  _getHeaderJSON: function() {
    var json = this.getHeader('X-JSON');
    if (!json) return null;
    json = decodeURIComponent(escape(json));
    try {
      return json.evalJSON(this.request.options.sanitizeJSON ||
        !this.request.isSameOrigin());
    } catch (e) {
      this.request.dispatchException(e);
    }
  },

  _getResponseJSON: function() {
    var options = this.request.options;
    if (!options.evalJSON || (options.evalJSON != 'force' &&
      !(this.getHeader('Content-type') || '').include('application/json')) ||
        this.responseText.blank())
          return null;
    try {
      return this.responseText.evalJSON(options.sanitizeJSON ||
        !this.request.isSameOrigin());
    } catch (e) {
      this.request.dispatchException(e);
    }
  }
});

Ajax.Updater = Class.create(Ajax.Request, {
  initialize: function($super, container, url, options) {
    this.container = {
      success: (container.success || container),
      failure: (container.failure || (container.success ? null : container))
    };

    options = Object.clone(options);
    var onComplete = options.onComplete;
    options.onComplete = (function(response, json) {
      this.updateContent(response.responseText);
      if (Object.isFunction(onComplete)) onComplete(response, json);
    }).bind(this);

    $super(url, options);
  },

  updateContent: function(responseText) {
    var receiver = this.container[this.success() ? 'success' : 'failure'],
        options = this.options;

    if (!options.evalScripts) responseText = responseText.stripScripts();

    if (receiver = $(receiver)) {
      if (options.insertion) {
        if (Object.isString(options.insertion)) {
          var insertion = { }; insertion[options.insertion] = responseText;
          receiver.insert(insertion);
        }
        else options.insertion(receiver, responseText);
      }
      else receiver.update(responseText);
    }
  }
});

Ajax.PeriodicalUpdater = Class.create(Ajax.Base, {
  initialize: function($super, container, url, options) {
    $super(options);
    this.onComplete = this.options.onComplete;

    this.frequency = (this.options.frequency || 2);
    this.decay = (this.options.decay || 1);

    this.updater = { };
    this.container = container;
    this.url = url;

    this.start();
  },

  start: function() {
    this.options.onComplete = this.updateComplete.bind(this);
    this.onTimerEvent();
  },

  stop: function() {
    this.updater.options.onComplete = undefined;
    clearTimeout(this.timer);
    (this.onComplete || Prototype.emptyFunction).apply(this, arguments);
  },

  updateComplete: function(response) {
    if (this.options.decay) {
      this.decay = (response.responseText == this.lastText ?
        this.decay * this.options.decay : 1);

      this.lastText = response.responseText;
    }
    this.timer = this.onTimerEvent.bind(this).delay(this.decay * this.frequency);
  },

  onTimerEvent: function() {
    this.updater = new Ajax.Updater(this.container, this.url, this.options);
  }
});



function $(element) {
  if (arguments.length > 1) {
    for (var i = 0, elements = [], length = arguments.length; i < length; i++)
      elements.push($(arguments[i]));
    return elements;
  }
  if (Object.isString(element))
    element = document.getElementById(element);
  return Element.extend(element);
}

if (Prototype.BrowserFeatures.XPath) {
  document._getElementsByXPath = function(expression, parentElement) {
    var results = [];
    var query = document.evaluate(expression, $(parentElement) || document,
      null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
    for (var i = 0, length = query.snapshotLength; i < length; i++)
      results.push(Element.extend(query.snapshotItem(i)));
    return results;
  };
}

/*--------------------------------------------------------------------------*/

if (!window.Node) var Node = { };

if (!Node.ELEMENT_NODE) {
  Object.extend(Node, {
    ELEMENT_NODE: 1,
    ATTRIBUTE_NODE: 2,
    TEXT_NODE: 3,
    CDATA_SECTION_NODE: 4,
    ENTITY_REFERENCE_NODE: 5,
    ENTITY_NODE: 6,
    PROCESSING_INSTRUCTION_NODE: 7,
    COMMENT_NODE: 8,
    DOCUMENT_NODE: 9,
    DOCUMENT_TYPE_NODE: 10,
    DOCUMENT_FRAGMENT_NODE: 11,
    NOTATION_NODE: 12
  });
}


(function(global) {

  var SETATTRIBUTE_IGNORES_NAME = (function(){
    var elForm = document.createElement("form");
    var elInput = document.createElement("input");
    var root = document.documentElement;
    elInput.setAttribute("name", "test");
    elForm.appendChild(elInput);
    root.appendChild(elForm);
    var isBuggy = elForm.elements
      ? (typeof elForm.elements.test == "undefined")
      : null;
    root.removeChild(elForm);
    elForm = elInput = null;
    return isBuggy;
  })();

  var element = global.Element;
  global.Element = function(tagName, attributes) {
    attributes = attributes || { };
    tagName = tagName.toLowerCase();
    var cache = Element.cache;
    if (SETATTRIBUTE_IGNORES_NAME && attributes.name) {
      tagName = '<' + tagName + ' name="' + attributes.name + '">';
      delete attributes.name;
      return Element.writeAttribute(document.createElement(tagName), attributes);
    }
    if (!cache[tagName]) cache[tagName] = Element.extend(document.createElement(tagName));
    return Element.writeAttribute(cache[tagName].cloneNode(false), attributes);
  };
  Object.extend(global.Element, element || { });
  if (element) global.Element.prototype = element.prototype;
})(this);

Element.cache = { };
Element.idCounter = 1;

Element.Methods = {
  visible: function(element) {
    return $(element).style.display != 'none';
  },

  toggle: function(element) {
    element = $(element);
    Element[Element.visible(element) ? 'hide' : 'show'](element);
    return element;
  },


  hide: function(element) {
    element = $(element);
    element.style.display = 'none';
    return element;
  },

  show: function(element) {
    element = $(element);
    element.style.display = '';
    return element;
  },

  remove: function(element) {
    element = $(element);
    element.parentNode.removeChild(element);
    return element;
  },

  update: (function(){

    var SELECT_ELEMENT_INNERHTML_BUGGY = (function(){
      var el = document.createElement("select"),
          isBuggy = true;
      el.innerHTML = "<option value=\"test\">test</option>";
      if (el.options && el.options[0]) {
        isBuggy = el.options[0].nodeName.toUpperCase() !== "OPTION";
      }
      el = null;
      return isBuggy;
    })();

    var TABLE_ELEMENT_INNERHTML_BUGGY = (function(){
      try {
        var el = document.createElement("table");
        if (el && el.tBodies) {
          el.innerHTML = "<tbody><tr><td>test</td></tr></tbody>";
          var isBuggy = typeof el.tBodies[0] == "undefined";
          el = null;
          return isBuggy;
        }
      } catch (e) {
        return true;
      }
    })();

    var SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING = (function () {
      var s = document.createElement("script"),
          isBuggy = false;
      try {
        s.appendChild(document.createTextNode(""));
        isBuggy = !s.firstChild ||
          s.firstChild && s.firstChild.nodeType !== 3;
      } catch (e) {
        isBuggy = true;
      }
      s = null;
      return isBuggy;
    })();

    function update(element, content) {
      element = $(element);

      if (content && content.toElement)
        content = content.toElement();

      if (Object.isElement(content))
        return element.update().insert(content);

      content = Object.toHTML(content);

      var tagName = element.tagName.toUpperCase();

      if (tagName === 'SCRIPT' && SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING) {
        element.text = content;
        return element;
      }

      if (SELECT_ELEMENT_INNERHTML_BUGGY || TABLE_ELEMENT_INNERHTML_BUGGY) {
        if (tagName in Element._insertionTranslations.tags) {
          while (element.firstChild) {
            element.removeChild(element.firstChild);
          }
          Element._getContentFromAnonymousElement(tagName, content.stripScripts())
            .each(function(node) {
              element.appendChild(node)
            });
        }
        else {
          element.innerHTML = content.stripScripts();
        }
      }
      else {
        element.innerHTML = content.stripScripts();
      }

      content.evalScripts.bind(content).defer();
      return element;
    }

    return update;
  })(),

  replace: function(element, content) {
    element = $(element);
    if (content && content.toElement) content = content.toElement();
    else if (!Object.isElement(content)) {
      content = Object.toHTML(content);
      var range = element.ownerDocument.createRange();
      range.selectNode(element);
      content.evalScripts.bind(content).defer();
      content = range.createContextualFragment(content.stripScripts());
    }
    element.parentNode.replaceChild(content, element);
    return element;
  },

  insert: function(element, insertions) {
    element = $(element);

    if (Object.isString(insertions) || Object.isNumber(insertions) ||
        Object.isElement(insertions) || (insertions && (insertions.toElement || insertions.toHTML)))
          insertions = {bottom:insertions};

    var content, insert, tagName, childNodes;

    for (var position in insertions) {
      content  = insertions[position];
      position = position.toLowerCase();
      insert = Element._insertionTranslations[position];

      if (content && content.toElement) content = content.toElement();
      if (Object.isElement(content)) {
        insert(element, content);
        continue;
      }

      content = Object.toHTML(content);

      tagName = ((position == 'before' || position == 'after')
        ? element.parentNode : element).tagName.toUpperCase();

      childNodes = Element._getContentFromAnonymousElement(tagName, content.stripScripts());

      if (position == 'top' || position == 'after') childNodes.reverse();
      childNodes.each(insert.curry(element));

      content.evalScripts.bind(content).defer();
    }

    return element;
  },

  wrap: function(element, wrapper, attributes) {
    element = $(element);
    if (Object.isElement(wrapper))
      $(wrapper).writeAttribute(attributes || { });
    else if (Object.isString(wrapper)) wrapper = new Element(wrapper, attributes);
    else wrapper = new Element('div', wrapper);
    if (element.parentNode)
      element.parentNode.replaceChild(wrapper, element);
    wrapper.appendChild(element);
    return wrapper;
  },

  inspect: function(element) {
    element = $(element);
    var result = '<' + element.tagName.toLowerCase();
    $H({'id': 'id', 'className': 'class'}).each(function(pair) {
      var property = pair.first(), attribute = pair.last();
      var value = (element[property] || '').toString();
      if (value) result += ' ' + attribute + '=' + value.inspect(true);
    });
    return result + '>';
  },

  recursivelyCollect: function(element, property) {
    element = $(element);
    var elements = [];
    while (element = element[property])
      if (element.nodeType == 1)
        elements.push(Element.extend(element));
    return elements;
  },

  ancestors: function(element) {
    return Element.recursivelyCollect(element, 'parentNode');
  },

  descendants: function(element) {
    return Element.select(element, "*");
  },

  firstDescendant: function(element) {
    element = $(element).firstChild;
    while (element && element.nodeType != 1) element = element.nextSibling;
    return $(element);
  },

  immediateDescendants: function(element) {
    if (!(element = $(element).firstChild)) return [];
    while (element && element.nodeType != 1) element = element.nextSibling;
    if (element) return [element].concat($(element).nextSiblings());
    return [];
  },

  previousSiblings: function(element) {
    return Element.recursivelyCollect(element, 'previousSibling');
  },

  nextSiblings: function(element) {
    return Element.recursivelyCollect(element, 'nextSibling');
  },

  siblings: function(element) {
    element = $(element);
    return Element.previousSiblings(element).reverse()
      .concat(Element.nextSiblings(element));
  },

  match: function(element, selector) {
    if (Object.isString(selector))
      selector = new Selector(selector);
    return selector.match($(element));
  },

  up: function(element, expression, index) {
    element = $(element);
    if (arguments.length == 1) return $(element.parentNode);
    var ancestors = Element.ancestors(element);
    return Object.isNumber(expression) ? ancestors[expression] :
      Selector.findElement(ancestors, expression, index);
  },

  down: function(element, expression, index) {
    element = $(element);
    if (arguments.length == 1) return Element.firstDescendant(element);
    return Object.isNumber(expression) ? Element.descendants(element)[expression] :
      Element.select(element, expression)[index || 0];
  },

  previous: function(element, expression, index) {
    element = $(element);
    if (arguments.length == 1) return $(Selector.handlers.previousElementSibling(element));
    var previousSiblings = Element.previousSiblings(element);
    return Object.isNumber(expression) ? previousSiblings[expression] :
      Selector.findElement(previousSiblings, expression, index);
  },

  next: function(element, expression, index) {
    element = $(element);
    if (arguments.length == 1) return $(Selector.handlers.nextElementSibling(element));
    var nextSiblings = Element.nextSiblings(element);
    return Object.isNumber(expression) ? nextSiblings[expression] :
      Selector.findElement(nextSiblings, expression, index);
  },


  select: function(element) {
    var args = Array.prototype.slice.call(arguments, 1);
    return Selector.findChildElements(element, args);
  },

  adjacent: function(element) {
    var args = Array.prototype.slice.call(arguments, 1);
    return Selector.findChildElements(element.parentNode, args).without(element);
  },

  identify: function(element) {
    element = $(element);
    var id = Element.readAttribute(element, 'id');
    if (id) return id;
    do { id = 'anonymous_element_' + Element.idCounter++ } while ($(id));
    Element.writeAttribute(element, 'id', id);
    return id;
  },

  readAttribute: function(element, name) {
    element = $(element);
    if (Prototype.Browser.IE) {
      var t = Element._attributeTranslations.read;
      if (t.values[name]) return t.values[name](element, name);
      if (t.names[name]) name = t.names[name];
      if (name.include(':')) {
        return (!element.attributes || !element.attributes[name]) ? null :
         element.attributes[name].value;
      }
    }
    return element.getAttribute(name);
  },

  writeAttribute: function(element, name, value) {
    element = $(element);
    var attributes = { }, t = Element._attributeTranslations.write;

    if (typeof name == 'object') attributes = name;
    else attributes[name] = Object.isUndefined(value) ? true : value;

    for (var attr in attributes) {
      name = t.names[attr] || attr;
      value = attributes[attr];
      if (t.values[attr]) name = t.values[attr](element, value);
      if (value === false || value === null)
        element.removeAttribute(name);
      else if (value === true)
        element.setAttribute(name, name);
      else element.setAttribute(name, value);
    }
    return element;
  },

  getHeight: function(element) {
    return Element.getDimensions(element).height;
  },

  getWidth: function(element) {
    return Element.getDimensions(element).width;
  },

  classNames: function(element) {
    return new Element.ClassNames(element);
  },

  hasClassName: function(element, className) {
    if (!(element = $(element))) return;
    var elementClassName = element.className;
    return (elementClassName.length > 0 && (elementClassName == className ||
      new RegExp("(^|\\s)" + className + "(\\s|$)").test(elementClassName)));
  },

  addClassName: function(element, className) {
    if (!(element = $(element))) return;
    if (!Element.hasClassName(element, className))
      element.className += (element.className ? ' ' : '') + className;
    return element;
  },

  removeClassName: function(element, className) {
    if (!(element = $(element))) return;
    element.className = element.className.replace(
      new RegExp("(^|\\s+)" + className + "(\\s+|$)"), ' ').strip();
    return element;
  },

  toggleClassName: function(element, className) {
    if (!(element = $(element))) return;
    return Element[Element.hasClassName(element, className) ?
      'removeClassName' : 'addClassName'](element, className);
  },

  cleanWhitespace: function(element) {
    element = $(element);
    var node = element.firstChild;
    while (node) {
      var nextNode = node.nextSibling;
      if (node.nodeType == 3 && !/\S/.test(node.nodeValue))
        element.removeChild(node);
      node = nextNode;
    }
    return element;
  },

  empty: function(element) {
    return $(element).innerHTML.blank();
  },

  descendantOf: function(element, ancestor) {
    element = $(element), ancestor = $(ancestor);

    if (element.compareDocumentPosition)
      return (element.compareDocumentPosition(ancestor) & 8) === 8;

    if (ancestor.contains)
      return ancestor.contains(element) && ancestor !== element;

    while (element = element.parentNode)
      if (element == ancestor) return true;

    return false;
  },

  scrollTo: function(element) {
    element = $(element);
    var pos = Element.cumulativeOffset(element);
    window.scrollTo(pos[0], pos[1]);
    return element;
  },

  getStyle: function(element, style) {
    element = $(element);
    style = style == 'float' ? 'cssFloat' : style.camelize();
    var value = element.style[style];
    if (!value || value == 'auto') {
      var css = document.defaultView.getComputedStyle(element, null);
      value = css ? css[style] : null;
    }
    if (style == 'opacity') return value ? parseFloat(value) : 1.0;
    return value == 'auto' ? null : value;
  },

  getOpacity: function(element) {
    return $(element).getStyle('opacity');
  },

  setStyle: function(element, styles) {
    element = $(element);
    var elementStyle = element.style, match;
    if (Object.isString(styles)) {
      element.style.cssText += ';' + styles;
      return styles.include('opacity') ?
        element.setOpacity(styles.match(/opacity:\s*(\d?\.?\d*)/)[1]) : element;
    }
    for (var property in styles)
      if (property == 'opacity') element.setOpacity(styles[property]);
      else
        elementStyle[(property == 'float' || property == 'cssFloat') ?
          (Object.isUndefined(elementStyle.styleFloat) ? 'cssFloat' : 'styleFloat') :
            property] = styles[property];

    return element;
  },

  setOpacity: function(element, value) {
    element = $(element);
    element.style.opacity = (value == 1 || value === '') ? '' :
      (value < 0.00001) ? 0 : value;
    return element;
  },

  getDimensions: function(element) {
    element = $(element);
    var display = Element.getStyle(element, 'display');
    if (display != 'none' && display != null) // Safari bug
      return {width: element.offsetWidth, height: element.offsetHeight};

    var els = element.style;
    var originalVisibility = els.visibility;
    var originalPosition = els.position;
    var originalDisplay = els.display;
    els.visibility = 'hidden';
    if (originalPosition != 'fixed') // Switching fixed to absolute causes issues in Safari
      els.position = 'absolute';
    els.display = 'block';
    var originalWidth = element.clientWidth;
    var originalHeight = element.clientHeight;
    els.display = originalDisplay;
    els.position = originalPosition;
    els.visibility = originalVisibility;
    return {width: originalWidth, height: originalHeight};
  },

  makePositioned: function(element) {
    element = $(element);
    var pos = Element.getStyle(element, 'position');
    if (pos == 'static' || !pos) {
      element._madePositioned = true;
      element.style.position = 'relative';
      if (Prototype.Browser.Opera) {
        element.style.top = 0;
        element.style.left = 0;
      }
    }
    return element;
  },

  undoPositioned: function(element) {
    element = $(element);
    if (element._madePositioned) {
      element._madePositioned = undefined;
      element.style.position =
        element.style.top =
        element.style.left =
        element.style.bottom =
        element.style.right = '';
    }
    return element;
  },

  makeClipping: function(element) {
    element = $(element);
    if (element._overflow) return element;
    element._overflow = Element.getStyle(element, 'overflow') || 'auto';
    if (element._overflow !== 'hidden')
      element.style.overflow = 'hidden';
    return element;
  },

  undoClipping: function(element) {
    element = $(element);
    if (!element._overflow) return element;
    element.style.overflow = element._overflow == 'auto' ? '' : element._overflow;
    element._overflow = null;
    return element;
  },

  cumulativeOffset: function(element) {
    var valueT = 0, valueL = 0;
    do {
      valueT += element.offsetTop  || 0;
      valueL += element.offsetLeft || 0;
      element = element.offsetParent;
    } while (element);
    return Element._returnOffset(valueL, valueT);
  },

  positionedOffset: function(element) {
    var valueT = 0, valueL = 0;
    do {
      valueT += element.offsetTop  || 0;
      valueL += element.offsetLeft || 0;
      element = element.offsetParent;
      if (element) {
        if (element.tagName.toUpperCase() == 'BODY') break;
        var p = Element.getStyle(element, 'position');
        if (p !== 'static') break;
      }
    } while (element);
    return Element._returnOffset(valueL, valueT);
  },

  absolutize: function(element) {
    element = $(element);
    if (Element.getStyle(element, 'position') == 'absolute') return element;

    var offsets = Element.positionedOffset(element);
    var top     = offsets[1];
    var left    = offsets[0];
    var width   = element.clientWidth;
    var height  = element.clientHeight;

    element._originalLeft   = left - parseFloat(element.style.left  || 0);
    element._originalTop    = top  - parseFloat(element.style.top || 0);
    element._originalWidth  = element.style.width;
    element._originalHeight = element.style.height;

    element.style.position = 'absolute';
    element.style.top    = top + 'px';
    element.style.left   = left + 'px';
    element.style.width  = width + 'px';
    element.style.height = height + 'px';
    return element;
  },

  relativize: function(element) {
    element = $(element);
    if (Element.getStyle(element, 'position') == 'relative') return element;

    element.style.position = 'relative';
    var top  = parseFloat(element.style.top  || 0) - (element._originalTop || 0);
    var left = parseFloat(element.style.left || 0) - (element._originalLeft || 0);

    element.style.top    = top + 'px';
    element.style.left   = left + 'px';
    element.style.height = element._originalHeight;
    element.style.width  = element._originalWidth;
    return element;
  },

  cumulativeScrollOffset: function(element) {
    var valueT = 0, valueL = 0;
    do {
      valueT += element.scrollTop  || 0;
      valueL += element.scrollLeft || 0;
      element = element.parentNode;
    } while (element);
    return Element._returnOffset(valueL, valueT);
  },

  getOffsetParent: function(element) {
    if (element.offsetParent) return $(element.offsetParent);
    if (element == document.body) return $(element);

    while ((element = element.parentNode) && element != document.body)
      if (Element.getStyle(element, 'position') != 'static')
        return $(element);

    return $(document.body);
  },

  viewportOffset: function(forElement) {
    var valueT = 0, valueL = 0;

    var element = forElement;
    do {
      valueT += element.offsetTop  || 0;
      valueL += element.offsetLeft || 0;

      if (element.offsetParent == document.body &&
        Element.getStyle(element, 'position') == 'absolute') break;

    } while (element = element.offsetParent);

    element = forElement;
    do {
      if (!Prototype.Browser.Opera || (element.tagName && (element.tagName.toUpperCase() == 'BODY'))) {
        valueT -= element.scrollTop  || 0;
        valueL -= element.scrollLeft || 0;
      }
    } while (element = element.parentNode);

    return Element._returnOffset(valueL, valueT);
  },

  clonePosition: function(element, source) {
    var options = Object.extend({
      setLeft:    true,
      setTop:     true,
      setWidth:   true,
      setHeight:  true,
      offsetTop:  0,
      offsetLeft: 0
    }, arguments[2] || { });

    source = $(source);
    var p = Element.viewportOffset(source);

    element = $(element);
    var delta = [0, 0];
    var parent = null;
    if (Element.getStyle(element, 'position') == 'absolute') {
      parent = Element.getOffsetParent(element);
      delta = Element.viewportOffset(parent);
    }

    if (parent == document.body) {
      delta[0] -= document.body.offsetLeft;
      delta[1] -= document.body.offsetTop;
    }

    if (options.setLeft)   element.style.left  = (p[0] - delta[0] + options.offsetLeft) + 'px';
    if (options.setTop)    element.style.top   = (p[1] - delta[1] + options.offsetTop) + 'px';
    if (options.setWidth)  element.style.width = source.offsetWidth + 'px';
    if (options.setHeight) element.style.height = source.offsetHeight + 'px';
    return element;
  }
};

Object.extend(Element.Methods, {
  getElementsBySelector: Element.Methods.select,

  childElements: Element.Methods.immediateDescendants
});

Element._attributeTranslations = {
  write: {
    names: {
      className: 'class',
      htmlFor:   'for'
    },
    values: { }
  }
};

if (Prototype.Browser.Opera) {
  Element.Methods.getStyle = Element.Methods.getStyle.wrap(
    function(proceed, element, style) {
      switch (style) {
        case 'left': case 'top': case 'right': case 'bottom':
          if (proceed(element, 'position') === 'static') return null;
        case 'height': case 'width':
          if (!Element.visible(element)) return null;

          var dim = parseInt(proceed(element, style), 10);

          if (dim !== element['offset' + style.capitalize()])
            return dim + 'px';

          var properties;
          if (style === 'height') {
            properties = ['border-top-width', 'padding-top',
             'padding-bottom', 'border-bottom-width'];
          }
          else {
            properties = ['border-left-width', 'padding-left',
             'padding-right', 'border-right-width'];
          }
          return properties.inject(dim, function(memo, property) {
            var val = proceed(element, property);
            return val === null ? memo : memo - parseInt(val, 10);
          }) + 'px';
        default: return proceed(element, style);
      }
    }
  );

  Element.Methods.readAttribute = Element.Methods.readAttribute.wrap(
    function(proceed, element, attribute) {
      if (attribute === 'title') return element.title;
      return proceed(element, attribute);
    }
  );
}

else if (Prototype.Browser.IE) {
  Element.Methods.getOffsetParent = Element.Methods.getOffsetParent.wrap(
    function(proceed, element) {
      element = $(element);
      try { element.offsetParent }
      catch(e) { return $(document.body) }
      var position = element.getStyle('position');
      if (position !== 'static') return proceed(element);
      element.setStyle({ position: 'relative' });
      var value = proceed(element);
      element.setStyle({ position: position });
      return value;
    }
  );

  $w('positionedOffset viewportOffset').each(function(method) {
    Element.Methods[method] = Element.Methods[method].wrap(
      function(proceed, element) {
        element = $(element);
        try { element.offsetParent }
        catch(e) { return Element._returnOffset(0,0) }
        var position = element.getStyle('position');
        if (position !== 'static') return proceed(element);
        var offsetParent = element.getOffsetParent();
        if (offsetParent && offsetParent.getStyle('position') === 'fixed')
          offsetParent.setStyle({ zoom: 1 });
        element.setStyle({ position: 'relative' });
        var value = proceed(element);
        element.setStyle({ position: position });
        return value;
      }
    );
  });

  Element.Methods.cumulativeOffset = Element.Methods.cumulativeOffset.wrap(
    function(proceed, element) {
      try { element.offsetParent }
      catch(e) { return Element._returnOffset(0,0) }
      return proceed(element);
    }
  );

  Element.Methods.getStyle = function(element, style) {
    element = $(element);
    style = (style == 'float' || style == 'cssFloat') ? 'styleFloat' : style.camelize();
    var value = element.style[style];
    if (!value && element.currentStyle) value = element.currentStyle[style];

    if (style == 'opacity') {
      if (value = (element.getStyle('filter') || '').match(/alpha\(opacity=(.*)\)/))
        if (value[1]) return parseFloat(value[1]) / 100;
      return 1.0;
    }

    if (value == 'auto') {
      if ((style == 'width' || style == 'height') && (element.getStyle('display') != 'none'))
        return element['offset' + style.capitalize()] + 'px';
      return null;
    }
    return value;
  };

  Element.Methods.setOpacity = function(element, value) {
    function stripAlpha(filter){
      return filter.replace(/alpha\([^\)]*\)/gi,'');
    }
    element = $(element);
    var currentStyle = element.currentStyle;
    if ((currentStyle && !currentStyle.hasLayout) ||
      (!currentStyle && element.style.zoom == 'normal'))
        element.style.zoom = 1;

    var filter = element.getStyle('filter'), style = element.style;
    if (value == 1 || value === '') {
      (filter = stripAlpha(filter)) ?
        style.filter = filter : style.removeAttribute('filter');
      return element;
    } else if (value < 0.00001) value = 0;
    style.filter = stripAlpha(filter) +
      'alpha(opacity=' + (value * 100) + ')';
    return element;
  };

  Element._attributeTranslations = (function(){

    var classProp = 'className';
    var forProp = 'for';

    var el = document.createElement('div');

    el.setAttribute(classProp, 'x');

    if (el.className !== 'x') {
      el.setAttribute('class', 'x');
      if (el.className === 'x') {
        classProp = 'class';
      }
    }
    el = null;

    el = document.createElement('label');
    el.setAttribute(forProp, 'x');
    if (el.htmlFor !== 'x') {
      el.setAttribute('htmlFor', 'x');
      if (el.htmlFor === 'x') {
        forProp = 'htmlFor';
      }
    }
    el = null;

    return {
      read: {
        names: {
          'class':      classProp,
          'className':  classProp,
          'for':        forProp,
          'htmlFor':    forProp
        },
        values: {
          _getAttr: function(element, attribute) {
            return element.getAttribute(attribute);
          },
          _getAttr2: function(element, attribute) {
            return element.getAttribute(attribute, 2);
          },
          _getAttrNode: function(element, attribute) {
            var node = element.getAttributeNode(attribute);
            return node ? node.value : "";
          },
          _getEv: (function(){

            var el = document.createElement('div');
            el.onclick = Prototype.emptyFunction;
            var value = el.getAttribute('onclick');
            var f;

            if (String(value).indexOf('{') > -1) {
              f = function(element, attribute) {
                attribute = element.getAttribute(attribute);
                if (!attribute) return null;
                attribute = attribute.toString();
                attribute = attribute.split('{')[1];
                attribute = attribute.split('}')[0];
                return attribute.strip();
              };
            }
            else if (value === '') {
              f = function(element, attribute) {
                attribute = element.getAttribute(attribute);
                if (!attribute) return null;
                return attribute.strip();
              };
            }
            el = null;
            return f;
          })(),
          _flag: function(element, attribute) {
            return $(element).hasAttribute(attribute) ? attribute : null;
          },
          style: function(element) {
            return element.style.cssText.toLowerCase();
          },
          title: function(element) {
            return element.title;
          }
        }
      }
    }
  })();

  Element._attributeTranslations.write = {
    names: Object.extend({
      cellpadding: 'cellPadding',
      cellspacing: 'cellSpacing'
    }, Element._attributeTranslations.read.names),
    values: {
      checked: function(element, value) {
        element.checked = !!value;
      },

      style: function(element, value) {
        element.style.cssText = value ? value : '';
      }
    }
  };

  Element._attributeTranslations.has = {};

  $w('colSpan rowSpan vAlign dateTime accessKey tabIndex ' +
      'encType maxLength readOnly longDesc frameBorder').each(function(attr) {
    Element._attributeTranslations.write.names[attr.toLowerCase()] = attr;
    Element._attributeTranslations.has[attr.toLowerCase()] = attr;
  });

  (function(v) {
    Object.extend(v, {
      href:        v._getAttr2,
      src:         v._getAttr2,
      type:        v._getAttr,
      action:      v._getAttrNode,
      disabled:    v._flag,
      checked:     v._flag,
      readonly:    v._flag,
      multiple:    v._flag,
      onload:      v._getEv,
      onunload:    v._getEv,
      onclick:     v._getEv,
      ondblclick:  v._getEv,
      onmousedown: v._getEv,
      onmouseup:   v._getEv,
      onmouseover: v._getEv,
      onmousemove: v._getEv,
      onmouseout:  v._getEv,
      onfocus:     v._getEv,
      onblur:      v._getEv,
      onkeypress:  v._getEv,
      onkeydown:   v._getEv,
      onkeyup:     v._getEv,
      onsubmit:    v._getEv,
      onreset:     v._getEv,
      onselect:    v._getEv,
      onchange:    v._getEv
    });
  })(Element._attributeTranslations.read.values);

  if (Prototype.BrowserFeatures.ElementExtensions) {
    (function() {
      function _descendants(element) {
        var nodes = element.getElementsByTagName('*'), results = [];
        for (var i = 0, node; node = nodes[i]; i++)
          if (node.tagName !== "!") // Filter out comment nodes.
            results.push(node);
        return results;
      }

      Element.Methods.down = function(element, expression, index) {
        element = $(element);
        if (arguments.length == 1) return element.firstDescendant();
        return Object.isNumber(expression) ? _descendants(element)[expression] :
          Element.select(element, expression)[index || 0];
      }
    })();
  }

}

else if (Prototype.Browser.Gecko && /rv:1\.8\.0/.test(navigator.userAgent)) {
  Element.Methods.setOpacity = function(element, value) {
    element = $(element);
    element.style.opacity = (value == 1) ? 0.999999 :
      (value === '') ? '' : (value < 0.00001) ? 0 : value;
    return element;
  };
}

else if (Prototype.Browser.WebKit) {
  Element.Methods.setOpacity = function(element, value) {
    element = $(element);
    element.style.opacity = (value == 1 || value === '') ? '' :
      (value < 0.00001) ? 0 : value;

    if (value == 1)
      if(element.tagName.toUpperCase() == 'IMG' && element.width) {
        element.width++; element.width--;
      } else try {
        var n = document.createTextNode(' ');
        element.appendChild(n);
        element.removeChild(n);
      } catch (e) { }

    return element;
  };

  Element.Methods.cumulativeOffset = function(element) {
    var valueT = 0, valueL = 0;
    do {
      valueT += element.offsetTop  || 0;
      valueL += element.offsetLeft || 0;
      if (element.offsetParent == document.body)
        if (Element.getStyle(element, 'position') == 'absolute') break;

      element = element.offsetParent;
    } while (element);

    return Element._returnOffset(valueL, valueT);
  };
}

if ('outerHTML' in document.documentElement) {
  Element.Methods.replace = function(element, content) {
    element = $(element);

    if (content && content.toElement) content = content.toElement();
    if (Object.isElement(content)) {
      element.parentNode.replaceChild(content, element);
      return element;
    }

    content = Object.toHTML(content);
    var parent = element.parentNode, tagName = parent.tagName.toUpperCase();

    if (Element._insertionTranslations.tags[tagName]) {
      var nextSibling = element.next();
      var fragments = Element._getContentFromAnonymousElement(tagName, content.stripScripts());
      parent.removeChild(element);
      if (nextSibling)
        fragments.each(function(node) { parent.insertBefore(node, nextSibling) });
      else
        fragments.each(function(node) { parent.appendChild(node) });
    }
    else element.outerHTML = content.stripScripts();

    content.evalScripts.bind(content).defer();
    return element;
  };
}

Element._returnOffset = function(l, t) {
  var result = [l, t];
  result.left = l;
  result.top = t;
  return result;
};

Element._getContentFromAnonymousElement = function(tagName, html) {
  var div = new Element('div'), t = Element._insertionTranslations.tags[tagName];
  if (t) {
    div.innerHTML = t[0] + html + t[1];
    t[2].times(function() { div = div.firstChild });
  } else div.innerHTML = html;
  return $A(div.childNodes);
};

Element._insertionTranslations = {
  before: function(element, node) {
    element.parentNode.insertBefore(node, element);
  },
  top: function(element, node) {
    element.insertBefore(node, element.firstChild);
  },
  bottom: function(element, node) {
    element.appendChild(node);
  },
  after: function(element, node) {
    element.parentNode.insertBefore(node, element.nextSibling);
  },
  tags: {
    TABLE:  ['<table>',                '</table>',                   1],
    TBODY:  ['<table><tbody>',         '</tbody></table>',           2],
    TR:     ['<table><tbody><tr>',     '</tr></tbody></table>',      3],
    TD:     ['<table><tbody><tr><td>', '</td></tr></tbody></table>', 4],
    SELECT: ['<select>',               '</select>',                  1]
  }
};

(function() {
  var tags = Element._insertionTranslations.tags;
  Object.extend(tags, {
    THEAD: tags.TBODY,
    TFOOT: tags.TBODY,
    TH:    tags.TD
  });
})();

Element.Methods.Simulated = {
  hasAttribute: function(element, attribute) {
    attribute = Element._attributeTranslations.has[attribute] || attribute;
    var node = $(element).getAttributeNode(attribute);
    return !!(node && node.specified);
  }
};

Element.Methods.ByTag = { };

Object.extend(Element, Element.Methods);

(function(div) {

  if (!Prototype.BrowserFeatures.ElementExtensions && div['__proto__']) {
    window.HTMLElement = { };
    window.HTMLElement.prototype = div['__proto__'];
    Prototype.BrowserFeatures.ElementExtensions = true;
  }

  div = null;

})(document.createElement('div'))

Element.extend = (function() {

  function checkDeficiency(tagName) {
    if (typeof window.Element != 'undefined') {
      var proto = window.Element.prototype;
      if (proto) {
        var id = '_' + (Math.random()+'').slice(2);
        var el = document.createElement(tagName);
        proto[id] = 'x';
        var isBuggy = (el[id] !== 'x');
        delete proto[id];
        el = null;
        return isBuggy;
      }
    }
    return false;
  }

  function extendElementWith(element, methods) {
    for (var property in methods) {
      var value = methods[property];
      if (Object.isFunction(value) && !(property in element))
        element[property] = value.methodize();
    }
  }

  var HTMLOBJECTELEMENT_PROTOTYPE_BUGGY = checkDeficiency('object');

  if (Prototype.BrowserFeatures.SpecificElementExtensions) {
    if (HTMLOBJECTELEMENT_PROTOTYPE_BUGGY) {
      return function(element) {
        if (element && typeof element._extendedByPrototype == 'undefined') {
          var t = element.tagName;
          if (t && (/^(?:object|applet|embed)$/i.test(t))) {
            extendElementWith(element, Element.Methods);
            extendElementWith(element, Element.Methods.Simulated);
            extendElementWith(element, Element.Methods.ByTag[t.toUpperCase()]);
          }
        }
        return element;
      }
    }
    return Prototype.K;
  }

  var Methods = { }, ByTag = Element.Methods.ByTag;

  var extend = Object.extend(function(element) {
    if (!element || typeof element._extendedByPrototype != 'undefined' ||
        element.nodeType != 1 || element == window) return element;

    var methods = Object.clone(Methods),
        tagName = element.tagName.toUpperCase();

    if (ByTag[tagName]) Object.extend(methods, ByTag[tagName]);

    extendElementWith(element, methods);

    element._extendedByPrototype = Prototype.emptyFunction;
    return element;

  }, {
    refresh: function() {
      if (!Prototype.BrowserFeatures.ElementExtensions) {
        Object.extend(Methods, Element.Methods);
        Object.extend(Methods, Element.Methods.Simulated);
      }
    }
  });

  extend.refresh();
  return extend;
})();

Element.hasAttribute = function(element, attribute) {
  if (element.hasAttribute) return element.hasAttribute(attribute);
  return Element.Methods.Simulated.hasAttribute(element, attribute);
};

Element.addMethods = function(methods) {
  var F = Prototype.BrowserFeatures, T = Element.Methods.ByTag;

  if (!methods) {
    Object.extend(Form, Form.Methods);
    Object.extend(Form.Element, Form.Element.Methods);
    Object.extend(Element.Methods.ByTag, {
      "FORM":     Object.clone(Form.Methods),
      "INPUT":    Object.clone(Form.Element.Methods),
      "SELECT":   Object.clone(Form.Element.Methods),
      "TEXTAREA": Object.clone(Form.Element.Methods)
    });
  }

  if (arguments.length == 2) {
    var tagName = methods;
    methods = arguments[1];
  }

  if (!tagName) Object.extend(Element.Methods, methods || { });
  else {
    if (Object.isArray(tagName)) tagName.each(extend);
    else extend(tagName);
  }

  function extend(tagName) {
    tagName = tagName.toUpperCase();
    if (!Element.Methods.ByTag[tagName])
      Element.Methods.ByTag[tagName] = { };
    Object.extend(Element.Methods.ByTag[tagName], methods);
  }

  function copy(methods, destination, onlyIfAbsent) {
    onlyIfAbsent = onlyIfAbsent || false;
    for (var property in methods) {
      var value = methods[property];
      if (!Object.isFunction(value)) continue;
      if (!onlyIfAbsent || !(property in destination))
        destination[property] = value.methodize();
    }
  }

  function findDOMClass(tagName) {
    var klass;
    var trans = {
      "OPTGROUP": "OptGroup", "TEXTAREA": "TextArea", "P": "Paragraph",
      "FIELDSET": "FieldSet", "UL": "UList", "OL": "OList", "DL": "DList",
      "DIR": "Directory", "H1": "Heading", "H2": "Heading", "H3": "Heading",
      "H4": "Heading", "H5": "Heading", "H6": "Heading", "Q": "Quote",
      "INS": "Mod", "DEL": "Mod", "A": "Anchor", "IMG": "Image", "CAPTION":
      "TableCaption", "COL": "TableCol", "COLGROUP": "TableCol", "THEAD":
      "TableSection", "TFOOT": "TableSection", "TBODY": "TableSection", "TR":
      "TableRow", "TH": "TableCell", "TD": "TableCell", "FRAMESET":
      "FrameSet", "IFRAME": "IFrame"
    };
    if (trans[tagName]) klass = 'HTML' + trans[tagName] + 'Element';
    if (window[klass]) return window[klass];
    klass = 'HTML' + tagName + 'Element';
    if (window[klass]) return window[klass];
    klass = 'HTML' + tagName.capitalize() + 'Element';
    if (window[klass]) return window[klass];

    var element = document.createElement(tagName);
    var proto = element['__proto__'] || element.constructor.prototype;
    element = null;
    return proto;
  }

  var elementPrototype = window.HTMLElement ? HTMLElement.prototype :
   Element.prototype;

  if (F.ElementExtensions) {
    copy(Element.Methods, elementPrototype);
    copy(Element.Methods.Simulated, elementPrototype, true);
  }

  if (F.SpecificElementExtensions) {
    for (var tag in Element.Methods.ByTag) {
      var klass = findDOMClass(tag);
      if (Object.isUndefined(klass)) continue;
      copy(T[tag], klass.prototype);
    }
  }

  Object.extend(Element, Element.Methods);
  delete Element.ByTag;

  if (Element.extend.refresh) Element.extend.refresh();
  Element.cache = { };
};


document.viewport = {

  getDimensions: function() {
    return { width: this.getWidth(), height: this.getHeight() };
  },

  getScrollOffsets: function() {
    return Element._returnOffset(
      window.pageXOffset || document.documentElement.scrollLeft || document.body.scrollLeft,
      window.pageYOffset || document.documentElement.scrollTop  || document.body.scrollTop);
  }
};

(function(viewport) {
  var B = Prototype.Browser, doc = document, element, property = {};

  function getRootElement() {
    if (B.WebKit && !doc.evaluate)
      return document;

    if (B.Opera && window.parseFloat(window.opera.version()) < 9.5)
      return document.body;

    return document.documentElement;
  }

  function define(D) {
    if (!element) element = getRootElement();

    property[D] = 'client' + D;

    viewport['get' + D] = function() { return element[property[D]] };
    return viewport['get' + D]();
  }

  viewport.getWidth  = define.curry('Width');

  viewport.getHeight = define.curry('Height');
})(document.viewport);


Element.Storage = {
  UID: 1
};

Element.addMethods({
  getStorage: function(element) {
    if (!(element = $(element))) return;

    var uid;
    if (element === window) {
      uid = 0;
    } else {
      if (typeof element._prototypeUID === "undefined")
        element._prototypeUID = [Element.Storage.UID++];
      uid = element._prototypeUID[0];
    }

    if (!Element.Storage[uid])
      Element.Storage[uid] = $H();

    return Element.Storage[uid];
  },

  store: function(element, key, value) {
    if (!(element = $(element))) return;

    if (arguments.length === 2) {
      Element.getStorage(element).update(key);
    } else {
      Element.getStorage(element).set(key, value);
    }

    return element;
  },

  retrieve: function(element, key, defaultValue) {
    if (!(element = $(element))) return;
    var hash = Element.getStorage(element), value = hash.get(key);

    if (Object.isUndefined(value)) {
      hash.set(key, defaultValue);
      value = defaultValue;
    }

    return value;
  },

  clone: function(element, deep) {
    if (!(element = $(element))) return;
    var clone = element.cloneNode(deep);
    clone._prototypeUID = void 0;
    if (deep) {
      var descendants = Element.select(clone, '*'),
          i = descendants.length;
      while (i--) {
        descendants[i]._prototypeUID = void 0;
      }
    }
    return Element.extend(clone);
  }
});
/* Portions of the Selector class are derived from Jack Slocum's DomQuery,
 * part of YUI-Ext version 0.40, distributed under the terms of an MIT-style
 * license.  Please see http://www.yui-ext.com/ for more information. */

var Selector = Class.create({
  initialize: function(expression) {
    this.expression = expression.strip();

    if (this.shouldUseSelectorsAPI()) {
      this.mode = 'selectorsAPI';
    } else if (this.shouldUseXPath()) {
      this.mode = 'xpath';
      this.compileXPathMatcher();
    } else {
      this.mode = "normal";
      this.compileMatcher();
    }

  },

  shouldUseXPath: (function() {

    var IS_DESCENDANT_SELECTOR_BUGGY = (function(){
      var isBuggy = false;
      if (document.evaluate && window.XPathResult) {
        var el = document.createElement('div');
        el.innerHTML = '<ul><li></li></ul><div><ul><li></li></ul></div>';

        var xpath = ".//*[local-name()='ul' or local-name()='UL']" +
          "//*[local-name()='li' or local-name()='LI']";

        var result = document.evaluate(xpath, el, null,
          XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);

        isBuggy = (result.snapshotLength !== 2);
        el = null;
      }
      return isBuggy;
    })();

    return function() {
      if (!Prototype.BrowserFeatures.XPath) return false;

      var e = this.expression;

      if (Prototype.Browser.WebKit &&
       (e.include("-of-type") || e.include(":empty")))
        return false;

      if ((/(\[[\w-]*?:|:checked)/).test(e))
        return false;

      if (IS_DESCENDANT_SELECTOR_BUGGY) return false;

      return true;
    }

  })(),

  shouldUseSelectorsAPI: function() {
    if (!Prototype.BrowserFeatures.SelectorsAPI) return false;

    if (Selector.CASE_INSENSITIVE_CLASS_NAMES) return false;

    if (!Selector._div) Selector._div = new Element('div');

    try {
      Selector._div.querySelector(this.expression);
    } catch(e) {
      return false;
    }

    return true;
  },

  compileMatcher: function() {
    var e = this.expression, ps = Selector.patterns, h = Selector.handlers,
        c = Selector.criteria, le, p, m, len = ps.length, name;

    if (Selector._cache[e]) {
      this.matcher = Selector._cache[e];
      return;
    }

    this.matcher = ["this.matcher = function(root) {",
                    "var r = root, h = Selector.handlers, c = false, n;"];

    while (e && le != e && (/\S/).test(e)) {
      le = e;
      for (var i = 0; i<len; i++) {
        p = ps[i].re;
        name = ps[i].name;
        if (m = e.match(p)) {
          this.matcher.push(Object.isFunction(c[name]) ? c[name](m) :
            new Template(c[name]).evaluate(m));
          e = e.replace(m[0], '');
          break;
        }
      }
    }

    this.matcher.push("return h.unique(n);\n}");
    eval(this.matcher.join('\n'));
    Selector._cache[this.expression] = this.matcher;
  },

  compileXPathMatcher: function() {
    var e = this.expression, ps = Selector.patterns,
        x = Selector.xpath, le, m, len = ps.length, name;

    if (Selector._cache[e]) {
      this.xpath = Selector._cache[e]; return;
    }

    this.matcher = ['.//*'];
    while (e && le != e && (/\S/).test(e)) {
      le = e;
      for (var i = 0; i<len; i++) {
        name = ps[i].name;
        if (m = e.match(ps[i].re)) {
          this.matcher.push(Object.isFunction(x[name]) ? x[name](m) :
            new Template(x[name]).evaluate(m));
          e = e.replace(m[0], '');
          break;
        }
      }
    }

    this.xpath = this.matcher.join('');
    Selector._cache[this.expression] = this.xpath;
  },

  findElements: function(root) {
    root = root || document;
    var e = this.expression, results;

    switch (this.mode) {
      case 'selectorsAPI':
        if (root !== document) {
          var oldId = root.id, id = $(root).identify();
          id = id.replace(/([\.:])/g, "\\$1");
          e = "#" + id + " " + e;
        }

        results = $A(root.querySelectorAll(e)).map(Element.extend);
        root.id = oldId;

        return results;
      case 'xpath':
        return document._getElementsByXPath(this.xpath, root);
      default:
       return this.matcher(root);
    }
  },

  match: function(element) {
    this.tokens = [];

    var e = this.expression, ps = Selector.patterns, as = Selector.assertions;
    var le, p, m, len = ps.length, name;

    while (e && le !== e && (/\S/).test(e)) {
      le = e;
      for (var i = 0; i<len; i++) {
        p = ps[i].re;
        name = ps[i].name;
        if (m = e.match(p)) {
          if (as[name]) {
            this.tokens.push([name, Object.clone(m)]);
            e = e.replace(m[0], '');
          } else {
            return this.findElements(document).include(element);
          }
        }
      }
    }

    var match = true, name, matches;
    for (var i = 0, token; token = this.tokens[i]; i++) {
      name = token[0], matches = token[1];
      if (!Selector.assertions[name](element, matches)) {
        match = false; break;
      }
    }

    return match;
  },

  toString: function() {
    return this.expression;
  },

  inspect: function() {
    return "#<Selector:" + this.expression.inspect() + ">";
  }
});

if (Prototype.BrowserFeatures.SelectorsAPI &&
 document.compatMode === 'BackCompat') {
  Selector.CASE_INSENSITIVE_CLASS_NAMES = (function(){
    var div = document.createElement('div'),
     span = document.createElement('span');

    div.id = "prototype_test_id";
    span.className = 'Test';
    div.appendChild(span);
    var isIgnored = (div.querySelector('#prototype_test_id .test') !== null);
    div = span = null;
    return isIgnored;
  })();
}

Object.extend(Selector, {
  _cache: { },

  xpath: {
    descendant:   "//*",
    child:        "/*",
    adjacent:     "/following-sibling::*[1]",
    laterSibling: '/following-sibling::*',
    tagName:      function(m) {
      if (m[1] == '*') return '';
      return "[local-name()='" + m[1].toLowerCase() +
             "' or local-name()='" + m[1].toUpperCase() + "']";
    },
    className:    "[contains(concat(' ', @class, ' '), ' #{1} ')]",
    id:           "[@id='#{1}']",
    attrPresence: function(m) {
      m[1] = m[1].toLowerCase();
      return new Template("[@#{1}]").evaluate(m);
    },
    attr: function(m) {
      m[1] = m[1].toLowerCase();
      m[3] = m[5] || m[6];
      return new Template(Selector.xpath.operators[m[2]]).evaluate(m);
    },
    pseudo: function(m) {
      var h = Selector.xpath.pseudos[m[1]];
      if (!h) return '';
      if (Object.isFunction(h)) return h(m);
      return new Template(Selector.xpath.pseudos[m[1]]).evaluate(m);
    },
    operators: {
      '=':  "[@#{1}='#{3}']",
      '!=': "[@#{1}!='#{3}']",
      '^=': "[starts-with(@#{1}, '#{3}')]",
      '$=': "[substring(@#{1}, (string-length(@#{1}) - string-length('#{3}') + 1))='#{3}']",
      '*=': "[contains(@#{1}, '#{3}')]",
      '~=': "[contains(concat(' ', @#{1}, ' '), ' #{3} ')]",
      '|=': "[contains(concat('-', @#{1}, '-'), '-#{3}-')]"
    },
    pseudos: {
      'first-child': '[not(preceding-sibling::*)]',
      'last-child':  '[not(following-sibling::*)]',
      'only-child':  '[not(preceding-sibling::* or following-sibling::*)]',
      'empty':       "[count(*) = 0 and (count(text()) = 0)]",
      'checked':     "[@checked]",
      'disabled':    "[(@disabled) and (@type!='hidden')]",
      'enabled':     "[not(@disabled) and (@type!='hidden')]",
      'not': function(m) {
        var e = m[6], p = Selector.patterns,
            x = Selector.xpath, le, v, len = p.length, name;

        var exclusion = [];
        while (e && le != e && (/\S/).test(e)) {
          le = e;
          for (var i = 0; i<len; i++) {
            name = p[i].name
            if (m = e.match(p[i].re)) {
              v = Object.isFunction(x[name]) ? x[name](m) : new Template(x[name]).evaluate(m);
              exclusion.push("(" + v.substring(1, v.length - 1) + ")");
              e = e.replace(m[0], '');
              break;
            }
          }
        }
        return "[not(" + exclusion.join(" and ") + ")]";
      },
      'nth-child':      function(m) {
        return Selector.xpath.pseudos.nth("(count(./preceding-sibling::*) + 1) ", m);
      },
      'nth-last-child': function(m) {
        return Selector.xpath.pseudos.nth("(count(./following-sibling::*) + 1) ", m);
      },
      'nth-of-type':    function(m) {
        return Selector.xpath.pseudos.nth("position() ", m);
      },
      'nth-last-of-type': function(m) {
        return Selector.xpath.pseudos.nth("(last() + 1 - position()) ", m);
      },
      'first-of-type':  function(m) {
        m[6] = "1"; return Selector.xpath.pseudos['nth-of-type'](m);
      },
      'last-of-type':   function(m) {
        m[6] = "1"; return Selector.xpath.pseudos['nth-last-of-type'](m);
      },
      'only-of-type':   function(m) {
        var p = Selector.xpath.pseudos; return p['first-of-type'](m) + p['last-of-type'](m);
      },
      nth: function(fragment, m) {
        var mm, formula = m[6], predicate;
        if (formula == 'even') formula = '2n+0';
        if (formula == 'odd')  formula = '2n+1';
        if (mm = formula.match(/^(\d+)$/)) // digit only
          return '[' + fragment + "= " + mm[1] + ']';
        if (mm = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b
          if (mm[1] == "-") mm[1] = -1;
          var a = mm[1] ? Number(mm[1]) : 1;
          var b = mm[2] ? Number(mm[2]) : 0;
          predicate = "[((#{fragment} - #{b}) mod #{a} = 0) and " +
          "((#{fragment} - #{b}) div #{a} >= 0)]";
          return new Template(predicate).evaluate({
            fragment: fragment, a: a, b: b });
        }
      }
    }
  },

  criteria: {
    tagName:      'n = h.tagName(n, r, "#{1}", c);      c = false;',
    className:    'n = h.className(n, r, "#{1}", c);    c = false;',
    id:           'n = h.id(n, r, "#{1}", c);           c = false;',
    attrPresence: 'n = h.attrPresence(n, r, "#{1}", c); c = false;',
    attr: function(m) {
      m[3] = (m[5] || m[6]);
      return new Template('n = h.attr(n, r, "#{1}", "#{3}", "#{2}", c); c = false;').evaluate(m);
    },
    pseudo: function(m) {
      if (m[6]) m[6] = m[6].replace(/"/g, '\\"');
      return new Template('n = h.pseudo(n, "#{1}", "#{6}", r, c); c = false;').evaluate(m);
    },
    descendant:   'c = "descendant";',
    child:        'c = "child";',
    adjacent:     'c = "adjacent";',
    laterSibling: 'c = "laterSibling";'
  },

  patterns: [
    { name: 'laterSibling', re: /^\s*~\s*/ },
    { name: 'child',        re: /^\s*>\s*/ },
    { name: 'adjacent',     re: /^\s*\+\s*/ },
    { name: 'descendant',   re: /^\s/ },

    { name: 'tagName',      re: /^\s*(\*|[\w\-]+)(\b|$)?/ },
    { name: 'id',           re: /^#([\w\-\*]+)(\b|$)/ },
    { name: 'className',    re: /^\.([\w\-\*]+)(\b|$)/ },
    { name: 'pseudo',       re: /^:((first|last|nth|nth-last|only)(-child|-of-type)|empty|checked|(en|dis)abled|not)(\((.*?)\))?(\b|$|(?=\s|[:+~>]))/ },
    { name: 'attrPresence', re: /^\[((?:[\w-]+:)?[\w-]+)\]/ },
    { name: 'attr',         re: /\[((?:[\w-]*:)?[\w-]+)\s*(?:([!^$*~|]?=)\s*((['"])([^\4]*?)\4|([^'"][^\]]*?)))?\]/ }
  ],

  assertions: {
    tagName: function(element, matches) {
      return matches[1].toUpperCase() == element.tagName.toUpperCase();
    },

    className: function(element, matches) {
      return Element.hasClassName(element, matches[1]);
    },

    id: function(element, matches) {
      return element.id === matches[1];
    },

    attrPresence: function(element, matches) {
      return Element.hasAttribute(element, matches[1]);
    },

    attr: function(element, matches) {
      var nodeValue = Element.readAttribute(element, matches[1]);
      return nodeValue && Selector.operators[matches[2]](nodeValue, matches[5] || matches[6]);
    }
  },

  handlers: {
    concat: function(a, b) {
      for (var i = 0, node; node = b[i]; i++)
        a.push(node);
      return a;
    },

    mark: function(nodes) {
      var _true = Prototype.emptyFunction;
      for (var i = 0, node; node = nodes[i]; i++)
        node._countedByPrototype = _true;
      return nodes;
    },

    unmark: (function(){

      var PROPERTIES_ATTRIBUTES_MAP = (function(){
        var el = document.createElement('div'),
            isBuggy = false,
            propName = '_countedByPrototype',
            value = 'x'
        el[propName] = value;
        isBuggy = (el.getAttribute(propName) === value);
        el = null;
        return isBuggy;
      })();

      return PROPERTIES_ATTRIBUTES_MAP ?
        function(nodes) {
          for (var i = 0, node; node = nodes[i]; i++)
            node.removeAttribute('_countedByPrototype');
          return nodes;
        } :
        function(nodes) {
          for (var i = 0, node; node = nodes[i]; i++)
            node._countedByPrototype = void 0;
          return nodes;
        }
    })(),

    index: function(parentNode, reverse, ofType) {
      parentNode._countedByPrototype = Prototype.emptyFunction;
      if (reverse) {
        for (var nodes = parentNode.childNodes, i = nodes.length - 1, j = 1; i >= 0; i--) {
          var node = nodes[i];
          if (node.nodeType == 1 && (!ofType || node._countedByPrototype)) node.nodeIndex = j++;
        }
      } else {
        for (var i = 0, j = 1, nodes = parentNode.childNodes; node = nodes[i]; i++)
          if (node.nodeType == 1 && (!ofType || node._countedByPrototype)) node.nodeIndex = j++;
      }
    },

    unique: function(nodes) {
      if (nodes.length == 0) return nodes;
      var results = [], n;
      for (var i = 0, l = nodes.length; i < l; i++)
        if (typeof (n = nodes[i])._countedByPrototype == 'undefined') {
          n._countedByPrototype = Prototype.emptyFunction;
          results.push(Element.extend(n));
        }
      return Selector.handlers.unmark(results);
    },

    descendant: function(nodes) {
      var h = Selector.handlers;
      for (var i = 0, results = [], node; node = nodes[i]; i++)
        h.concat(results, node.getElementsByTagName('*'));
      return results;
    },

    child: function(nodes) {
      var h = Selector.handlers;
      for (var i = 0, results = [], node; node = nodes[i]; i++) {
        for (var j = 0, child; child = node.childNodes[j]; j++)
          if (child.nodeType == 1 && child.tagName != '!') results.push(child);
      }
      return results;
    },

    adjacent: function(nodes) {
      for (var i = 0, results = [], node; node = nodes[i]; i++) {
        var next = this.nextElementSibling(node);
        if (next) results.push(next);
      }
      return results;
    },

    laterSibling: function(nodes) {
      var h = Selector.handlers;
      for (var i = 0, results = [], node; node = nodes[i]; i++)
        h.concat(results, Element.nextSiblings(node));
      return results;
    },

    nextElementSibling: function(node) {
      while (node = node.nextSibling)
        if (node.nodeType == 1) return node;
      return null;
    },

    previousElementSibling: function(node) {
      while (node = node.previousSibling)
        if (node.nodeType == 1) return node;
      return null;
    },

    tagName: function(nodes, root, tagName, combinator) {
      var uTagName = tagName.toUpperCase();
      var results = [], h = Selector.handlers;
      if (nodes) {
        if (combinator) {
          if (combinator == "descendant") {
            for (var i = 0, node; node = nodes[i]; i++)
              h.concat(results, node.getElementsByTagName(tagName));
            return results;
          } else nodes = this[combinator](nodes);
          if (tagName == "*") return nodes;
        }
        for (var i = 0, node; node = nodes[i]; i++)
          if (node.tagName.toUpperCase() === uTagName) results.push(node);
        return results;
      } else return root.getElementsByTagName(tagName);
    },

    id: function(nodes, root, id, combinator) {
      var targetNode = $(id), h = Selector.handlers;

      if (root == document) {
        if (!targetNode) return [];
        if (!nodes) return [targetNode];
      } else {
        if (!root.sourceIndex || root.sourceIndex < 1) {
          var nodes = root.getElementsByTagName('*');
          for (var j = 0, node; node = nodes[j]; j++) {
            if (node.id === id) return [node];
          }
        }
      }

      if (nodes) {
        if (combinator) {
          if (combinator == 'child') {
            for (var i = 0, node; node = nodes[i]; i++)
              if (targetNode.parentNode == node) return [targetNode];
          } else if (combinator == 'descendant') {
            for (var i = 0, node; node = nodes[i]; i++)
              if (Element.descendantOf(targetNode, node)) return [targetNode];
          } else if (combinator == 'adjacent') {
            for (var i = 0, node; node = nodes[i]; i++)
              if (Selector.handlers.previousElementSibling(targetNode) == node)
                return [targetNode];
          } else nodes = h[combinator](nodes);
        }
        for (var i = 0, node; node = nodes[i]; i++)
          if (node == targetNode) return [targetNode];
        return [];
      }
      return (targetNode && Element.descendantOf(targetNode, root)) ? [targetNode] : [];
    },

    className: function(nodes, root, className, combinator) {
      if (nodes && combinator) nodes = this[combinator](nodes);
      return Selector.handlers.byClassName(nodes, root, className);
    },

    byClassName: function(nodes, root, className) {
      if (!nodes) nodes = Selector.handlers.descendant([root]);
      var needle = ' ' + className + ' ';
      for (var i = 0, results = [], node, nodeClassName; node = nodes[i]; i++) {
        nodeClassName = node.className;
        if (nodeClassName.length == 0) continue;
        if (nodeClassName == className || (' ' + nodeClassName + ' ').include(needle))
          results.push(node);
      }
      return results;
    },

    attrPresence: function(nodes, root, attr, combinator) {
      if (!nodes) nodes = root.getElementsByTagName("*");
      if (nodes && combinator) nodes = this[combinator](nodes);
      var results = [];
      for (var i = 0, node; node = nodes[i]; i++)
        if (Element.hasAttribute(node, attr)) results.push(node);
      return results;
    },

    attr: function(nodes, root, attr, value, operator, combinator) {
      if (!nodes) nodes = root.getElementsByTagName("*");
      if (nodes && combinator) nodes = this[combinator](nodes);
      var handler = Selector.operators[operator], results = [];
      for (var i = 0, node; node = nodes[i]; i++) {
        var nodeValue = Element.readAttribute(node, attr);
        if (nodeValue === null) continue;
        if (handler(nodeValue, value)) results.push(node);
      }
      return results;
    },

    pseudo: function(nodes, name, value, root, combinator) {
      if (nodes && combinator) nodes = this[combinator](nodes);
      if (!nodes) nodes = root.getElementsByTagName("*");
      return Selector.pseudos[name](nodes, value, root);
    }
  },

  pseudos: {
    'first-child': function(nodes, value, root) {
      for (var i = 0, results = [], node; node = nodes[i]; i++) {
        if (Selector.handlers.previousElementSibling(node)) continue;
          results.push(node);
      }
      return results;
    },
    'last-child': function(nodes, value, root) {
      for (var i = 0, results = [], node; node = nodes[i]; i++) {
        if (Selector.handlers.nextElementSibling(node)) continue;
          results.push(node);
      }
      return results;
    },
    'only-child': function(nodes, value, root) {
      var h = Selector.handlers;
      for (var i = 0, results = [], node; node = nodes[i]; i++)
        if (!h.previousElementSibling(node) && !h.nextElementSibling(node))
          results.push(node);
      return results;
    },
    'nth-child':        function(nodes, formula, root) {
      return Selector.pseudos.nth(nodes, formula, root);
    },
    'nth-last-child':   function(nodes, formula, root) {
      return Selector.pseudos.nth(nodes, formula, root, true);
    },
    'nth-of-type':      function(nodes, formula, root) {
      return Selector.pseudos.nth(nodes, formula, root, false, true);
    },
    'nth-last-of-type': function(nodes, formula, root) {
      return Selector.pseudos.nth(nodes, formula, root, true, true);
    },
    'first-of-type':    function(nodes, formula, root) {
      return Selector.pseudos.nth(nodes, "1", root, false, true);
    },
    'last-of-type':     function(nodes, formula, root) {
      return Selector.pseudos.nth(nodes, "1", root, true, true);
    },
    'only-of-type':     function(nodes, formula, root) {
      var p = Selector.pseudos;
      return p['last-of-type'](p['first-of-type'](nodes, formula, root), formula, root);
    },

    getIndices: function(a, b, total) {
      if (a == 0) return b > 0 ? [b] : [];
      return $R(1, total).inject([], function(memo, i) {
        if (0 == (i - b) % a && (i - b) / a >= 0) memo.push(i);
        return memo;
      });
    },

    nth: function(nodes, formula, root, reverse, ofType) {
      if (nodes.length == 0) return [];
      if (formula == 'even') formula = '2n+0';
      if (formula == 'odd')  formula = '2n+1';
      var h = Selector.handlers, results = [], indexed = [], m;
      h.mark(nodes);
      for (var i = 0, node; node = nodes[i]; i++) {
        if (!node.parentNode._countedByPrototype) {
          h.index(node.parentNode, reverse, ofType);
          indexed.push(node.parentNode);
        }
      }
      if (formula.match(/^\d+$/)) { // just a number
        formula = Number(formula);
        for (var i = 0, node; node = nodes[i]; i++)
          if (node.nodeIndex == formula) results.push(node);
      } else if (m = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b
        if (m[1] == "-") m[1] = -1;
        var a = m[1] ? Number(m[1]) : 1;
        var b = m[2] ? Number(m[2]) : 0;
        var indices = Selector.pseudos.getIndices(a, b, nodes.length);
        for (var i = 0, node, l = indices.length; node = nodes[i]; i++) {
          for (var j = 0; j < l; j++)
            if (node.nodeIndex == indices[j]) results.push(node);
        }
      }
      h.unmark(nodes);
      h.unmark(indexed);
      return results;
    },

    'empty': function(nodes, value, root) {
      for (var i = 0, results = [], node; node = nodes[i]; i++) {
        if (node.tagName == '!' || node.firstChild) continue;
        results.push(node);
      }
      return results;
    },

    'not': function(nodes, selector, root) {
      var h = Selector.handlers, selectorType, m;
      var exclusions = new Selector(selector).findElements(root);
      h.mark(exclusions);
      for (var i = 0, results = [], node; node = nodes[i]; i++)
        if (!node._countedByPrototype) results.push(node);
      h.unmark(exclusions);
      return results;
    },

    'enabled': function(nodes, value, root) {
      for (var i = 0, results = [], node; node = nodes[i]; i++)
        if (!node.disabled && (!node.type || node.type !== 'hidden'))
          results.push(node);
      return results;
    },

    'disabled': function(nodes, value, root) {
      for (var i = 0, results = [], node; node = nodes[i]; i++)
        if (node.disabled) results.push(node);
      return results;
    },

    'checked': function(nodes, value, root) {
      for (var i = 0, results = [], node; node = nodes[i]; i++)
        if (node.checked) results.push(node);
      return results;
    }
  },

  operators: {
    '=':  function(nv, v) { return nv == v; },
    '!=': function(nv, v) { return nv != v; },
    '^=': function(nv, v) { return nv == v || nv && nv.startsWith(v); },
    '$=': function(nv, v) { return nv == v || nv && nv.endsWith(v); },
    '*=': function(nv, v) { return nv == v || nv && nv.include(v); },
    '~=': function(nv, v) { return (' ' + nv + ' ').include(' ' + v + ' '); },
    '|=': function(nv, v) { return ('-' + (nv || "").toUpperCase() +
     '-').include('-' + (v || "").toUpperCase() + '-'); }
  },

  split: function(expression) {
    var expressions = [];
    expression.scan(/(([\w#:.~>+()\s-]+|\*|\[.*?\])+)\s*(,|$)/, function(m) {
      expressions.push(m[1].strip());
    });
    return expressions;
  },

  matchElements: function(elements, expression) {
    var matches = $$(expression), h = Selector.handlers;
    h.mark(matches);
    for (var i = 0, results = [], element; element = elements[i]; i++)
      if (element._countedByPrototype) results.push(element);
    h.unmark(matches);
    return results;
  },

  findElement: function(elements, expression, index) {
    if (Object.isNumber(expression)) {
      index = expression; expression = false;
    }
    return Selector.matchElements(elements, expression || '*')[index || 0];
  },

  findChildElements: function(element, expressions) {
    expressions = Selector.split(expressions.join(','));
    var results = [], h = Selector.handlers;
    for (var i = 0, l = expressions.length, selector; i < l; i++) {
      selector = new Selector(expressions[i].strip());
      h.concat(results, selector.findElements(element));
    }
    return (l > 1) ? h.unique(results) : results;
  }
});

if (Prototype.Browser.IE) {
  Object.extend(Selector.handlers, {
    concat: function(a, b) {
      for (var i = 0, node; node = b[i]; i++)
        if (node.tagName !== "!") a.push(node);
      return a;
    }
  });
}

function $$() {
  return Selector.findChildElements(document, $A(arguments));
}

var Form = {
  reset: function(form) {
    form = $(form);
    form.reset();
    return form;
  },

  serializeElements: function(elements, options) {
    if (typeof options != 'object') options = { hash: !!options };
    else if (Object.isUndefined(options.hash)) options.hash = true;
    var key, value, submitted = false, submit = options.submit;

    var data = elements.inject({ }, function(result, element) {
      if (!element.disabled && element.name) {
        key = element.name; value = $(element).getValue();
        if (value != null && element.type != 'file' && (element.type != 'submit' || (!submitted &&
            submit !== false && (!submit || key == submit) && (submitted = true)))) {
          if (key in result) {
            if (!Object.isArray(result[key])) result[key] = [result[key]];
            result[key].push(value);
          }
          else result[key] = value;
        }
      }
      return result;
    });

    return options.hash ? data : Object.toQueryString(data);
  }
};

Form.Methods = {
  serialize: function(form, options) {
    return Form.serializeElements(Form.getElements(form), options);
  },

  getElements: function(form) {
    var elements = $(form).getElementsByTagName('*'),
        element,
        arr = [ ],
        serializers = Form.Element.Serializers;
    for (var i = 0; element = elements[i]; i++) {
      arr.push(element);
    }
    return arr.inject([], function(elements, child) {
      if (serializers[child.tagName.toLowerCase()])
        elements.push(Element.extend(child));
      return elements;
    })
  },

  getInputs: function(form, typeName, name) {
    form = $(form);
    var inputs = form.getElementsByTagName('input');

    if (!typeName && !name) return $A(inputs).map(Element.extend);

    for (var i = 0, matchingInputs = [], length = inputs.length; i < length; i++) {
      var input = inputs[i];
      if ((typeName && input.type != typeName) || (name && input.name != name))
        continue;
      matchingInputs.push(Element.extend(input));
    }

    return matchingInputs;
  },

  disable: function(form) {
    form = $(form);
    Form.getElements(form).invoke('disable');
    return form;
  },

  enable: function(form) {
    form = $(form);
    Form.getElements(form).invoke('enable');
    return form;
  },

  findFirstElement: function(form) {
    var elements = $(form).getElements().findAll(function(element) {
      return 'hidden' != element.type && !element.disabled;
    });
    var firstByIndex = elements.findAll(function(element) {
      return element.hasAttribute('tabIndex') && element.tabIndex >= 0;
    }).sortBy(function(element) { return element.tabIndex }).first();

    return firstByIndex ? firstByIndex : elements.find(function(element) {
      return /^(?:input|select|textarea)$/i.test(element.tagName);
    });
  },

  focusFirstElement: function(form) {
    form = $(form);
    form.findFirstElement().activate();
    return form;
  },

  request: function(form, options) {
    form = $(form), options = Object.clone(options || { });

    var params = options.parameters, action = form.readAttribute('action') || '';
    if (action.blank()) action = window.location.href;
    options.parameters = form.serialize(true);

    if (params) {
      if (Object.isString(params)) params = params.toQueryParams();
      Object.extend(options.parameters, params);
    }

    if (form.hasAttribute('method') && !options.method)
      options.method = form.method;

    return new Ajax.Request(action, options);
  }
};

/*--------------------------------------------------------------------------*/


Form.Element = {
  focus: function(element) {
    $(element).focus();
    return element;
  },

  select: function(element) {
    $(element).select();
    return element;
  }
};

Form.Element.Methods = {

  serialize: function(element) {
    element = $(element);
    if (!element.disabled && element.name) {
      var value = element.getValue();
      if (value != undefined) {
        var pair = { };
        pair[element.name] = value;
        return Object.toQueryString(pair);
      }
    }
    return '';
  },

  getValue: function(element) {
    element = $(element);
    var method = element.tagName.toLowerCase();
    return Form.Element.Serializers[method](element);
  },

  setValue: function(element, value) {
    element = $(element);
    var method = element.tagName.toLowerCase();
    Form.Element.Serializers[method](element, value);
    return element;
  },

  clear: function(element) {
    $(element).value = '';
    return element;
  },

  present: function(element) {
    return $(element).value != '';
  },

  activate: function(element) {
    element = $(element);
    try {
      element.focus();
      if (element.select && (element.tagName.toLowerCase() != 'input' ||
          !(/^(?:button|reset|submit)$/i.test(element.type))))
        element.select();
    } catch (e) { }
    return element;
  },

  disable: function(element) {
    element = $(element);
    element.disabled = true;
    return element;
  },

  enable: function(element) {
    element = $(element);
    element.disabled = false;
    return element;
  }
};

/*--------------------------------------------------------------------------*/

var Field = Form.Element;

var $F = Form.Element.Methods.getValue;

/*--------------------------------------------------------------------------*/

Form.Element.Serializers = {
  input: function(element, value) {
    switch (element.type.toLowerCase()) {
      case 'checkbox':
      case 'radio':
        return Form.Element.Serializers.inputSelector(element, value);
      default:
        return Form.Element.Serializers.textarea(element, value);
    }
  },

  inputSelector: function(element, value) {
    if (Object.isUndefined(value)) return element.checked ? element.value : null;
    else element.checked = !!value;
  },

  textarea: function(element, value) {
    if (Object.isUndefined(value)) return element.value;
    else element.value = value;
  },

  select: function(element, value) {
    if (Object.isUndefined(value))
      return this[element.type == 'select-one' ?
        'selectOne' : 'selectMany'](element);
    else {
      var opt, currentValue, single = !Object.isArray(value);
      for (var i = 0, length = element.length; i < length; i++) {
        opt = element.options[i];
        currentValue = this.optionValue(opt);
        if (single) {
          if (currentValue == value) {
            opt.selected = true;
            return;
          }
        }
        else opt.selected = value.include(currentValue);
      }
    }
  },

  selectOne: function(element) {
    var index = element.selectedIndex;
    return index >= 0 ? this.optionValue(element.options[index]) : null;
  },

  selectMany: function(element) {
    var values, length = element.length;
    if (!length) return null;

    for (var i = 0, values = []; i < length; i++) {
      var opt = element.options[i];
      if (opt.selected) values.push(this.optionValue(opt));
    }
    return values;
  },

  optionValue: function(opt) {
    return Element.extend(opt).hasAttribute('value') ? opt.value : opt.text;
  }
};

/*--------------------------------------------------------------------------*/


Abstract.TimedObserver = Class.create(PeriodicalExecuter, {
  initialize: function($super, element, frequency, callback) {
    $super(callback, frequency);
    this.element   = $(element);
    this.lastValue = this.getValue();
  },

  execute: function() {
    var value = this.getValue();
    if (Object.isString(this.lastValue) && Object.isString(value) ?
        this.lastValue != value : String(this.lastValue) != String(value)) {
      this.callback(this.element, value);
      this.lastValue = value;
    }
  }
});

Form.Element.Observer = Class.create(Abstract.TimedObserver, {
  getValue: function() {
    return Form.Element.getValue(this.element);
  }
});

Form.Observer = Class.create(Abstract.TimedObserver, {
  getValue: function() {
    return Form.serialize(this.element);
  }
});

/*--------------------------------------------------------------------------*/

Abstract.EventObserver = Class.create({
  initialize: function(element, callback) {
    this.element  = $(element);
    this.callback = callback;

    this.lastValue = this.getValue();
    if (this.element.tagName.toLowerCase() == 'form')
      this.registerFormCallbacks();
    else
      this.registerCallback(this.element);
  },

  onElementEvent: function() {
    var value = this.getValue();
    if (this.lastValue != value) {
      this.callback(this.element, value);
      this.lastValue = value;
    }
  },

  registerFormCallbacks: function() {
    Form.getElements(this.element).each(this.registerCallback, this);
  },

  registerCallback: function(element) {
    if (element.type) {
      switch (element.type.toLowerCase()) {
        case 'checkbox':
        case 'radio':
          Event.observe(element, 'click', this.onElementEvent.bind(this));
          break;
        default:
          Event.observe(element, 'change', this.onElementEvent.bind(this));
          break;
      }
    }
  }
});

Form.Element.EventObserver = Class.create(Abstract.EventObserver, {
  getValue: function() {
    return Form.Element.getValue(this.element);
  }
});

Form.EventObserver = Class.create(Abstract.EventObserver, {
  getValue: function() {
    return Form.serialize(this.element);
  }
});
(function() {

  var Event = {
    KEY_BACKSPACE: 8,
    KEY_TAB:       9,
    KEY_RETURN:   13,
    KEY_ESC:      27,
    KEY_LEFT:     37,
    KEY_UP:       38,
    KEY_RIGHT:    39,
    KEY_DOWN:     40,
    KEY_DELETE:   46,
    KEY_HOME:     36,
    KEY_END:      35,
    KEY_PAGEUP:   33,
    KEY_PAGEDOWN: 34,
    KEY_INSERT:   45,

    cache: {}
  };

  var docEl = document.documentElement;
  var MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED = 'onmouseenter' in docEl
    && 'onmouseleave' in docEl;

  var _isButton;
  if (Prototype.Browser.IE) {
    var buttonMap = { 0: 1, 1: 4, 2: 2 };
    _isButton = function(event, code) {
      return event.button === buttonMap[code];
    };
  } else if (Prototype.Browser.WebKit) {
    _isButton = function(event, code) {
      switch (code) {
        case 0: return event.which == 1 && !event.metaKey;
        case 1: return event.which == 1 && event.metaKey;
        default: return false;
      }
    };
  } else {
    _isButton = function(event, code) {
      return event.which ? (event.which === code + 1) : (event.button === code);
    };
  }

  function isLeftClick(event)   { return _isButton(event, 0) }

  function isMiddleClick(event) { return _isButton(event, 1) }

  function isRightClick(event)  { return _isButton(event, 2) }

  function element(event) {
    event = Event.extend(event);

    var node = event.target, type = event.type,
     currentTarget = event.currentTarget;

    if (currentTarget && currentTarget.tagName) {
      if (type === 'load' || type === 'error' ||
        (type === 'click' && currentTarget.tagName.toLowerCase() === 'input'
          && currentTarget.type === 'radio'))
            node = currentTarget;
    }

    if (node.nodeType == Node.TEXT_NODE)
      node = node.parentNode;

    return Element.extend(node);
  }

  function findElement(event, expression) {
    var element = Event.element(event);
    if (!expression) return element;
    var elements = [element].concat(element.ancestors());
    return Selector.findElement(elements, expression, 0);
  }

  function pointer(event) {
    return { x: pointerX(event), y: pointerY(event) };
  }

  function pointerX(event) {
    var docElement = document.documentElement,
     body = document.body || { scrollLeft: 0 };

    return event.pageX || (event.clientX +
      (docElement.scrollLeft || body.scrollLeft) -
      (docElement.clientLeft || 0));
  }

  function pointerY(event) {
    var docElement = document.documentElement,
     body = document.body || { scrollTop: 0 };

    return  event.pageY || (event.clientY +
       (docElement.scrollTop || body.scrollTop) -
       (docElement.clientTop || 0));
  }


  function stop(event) {
    Event.extend(event);
    event.preventDefault();
    event.stopPropagation();

    event.stopped = true;
  }

  Event.Methods = {
    isLeftClick: isLeftClick,
    isMiddleClick: isMiddleClick,
    isRightClick: isRightClick,

    element: element,
    findElement: findElement,

    pointer: pointer,
    pointerX: pointerX,
    pointerY: pointerY,

    stop: stop
  };


  var methods = Object.keys(Event.Methods).inject({ }, function(m, name) {
    m[name] = Event.Methods[name].methodize();
    return m;
  });

  if (Prototype.Browser.IE) {
    function _relatedTarget(event) {
      var element;
      switch (event.type) {
        case 'mouseover': element = event.fromElement; break;
        case 'mouseout':  element = event.toElement;   break;
        default: return null;
      }
      return Element.extend(element);
    }

    Object.extend(methods, {
      stopPropagation: function() { this.cancelBubble = true },
      preventDefault:  function() { this.returnValue = false },
      inspect: function() { return '[object Event]' }
    });

    Event.extend = function(event, element) {
      if (!event) return false;
      if (event._extendedByPrototype) return event;

      event._extendedByPrototype = Prototype.emptyFunction;
      var pointer = Event.pointer(event);

      Object.extend(event, {
        target: event.srcElement || element,
        relatedTarget: _relatedTarget(event),
        pageX:  pointer.x,
        pageY:  pointer.y
      });

      return Object.extend(event, methods);
    };
  } else {
    Event.prototype = window.Event.prototype || document.createEvent('HTMLEvents').__proto__;
    Object.extend(Event.prototype, methods);
    Event.extend = Prototype.K;
  }

  function _createResponder(element, eventName, handler) {
    var registry = Element.retrieve(element, 'prototype_event_registry');

    if (Object.isUndefined(registry)) {
      CACHE.push(element);
      registry = Element.retrieve(element, 'prototype_event_registry', $H());
    }

    var respondersForEvent = registry.get(eventName);
    if (Object.isUndefined(respondersForEvent)) {
      respondersForEvent = [];
      registry.set(eventName, respondersForEvent);
    }

    if (respondersForEvent.pluck('handler').include(handler)) return false;

    var responder;
    if (eventName.include(":")) {
      responder = function(event) {
        if (Object.isUndefined(event.eventName))
          return false;

        if (event.eventName !== eventName)
          return false;

        Event.extend(event, element);
        handler.call(element, event);
      };
    } else {
      if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED &&
       (eventName === "mouseenter" || eventName === "mouseleave")) {
        if (eventName === "mouseenter" || eventName === "mouseleave") {
          responder = function(event) {
            Event.extend(event, element);

            var parent = event.relatedTarget;
            while (parent && parent !== element) {
              try { parent = parent.parentNode; }
              catch(e) { parent = element; }
            }

            if (parent === element) return;

            handler.call(element, event);
          };
        }
      } else {
        responder = function(event) {
          Event.extend(event, element);
          handler.call(element, event);
        };
      }
    }

    responder.handler = handler;
    respondersForEvent.push(responder);
    return responder;
  }

  function _destroyCache() {
    for (var i = 0, length = CACHE.length; i < length; i++) {
      Event.stopObserving(CACHE[i]);
      CACHE[i] = null;
    }
  }

  var CACHE = [];

  if (Prototype.Browser.IE)
    window.attachEvent('onunload', _destroyCache);

  if (Prototype.Browser.WebKit)
    window.addEventListener('unload', Prototype.emptyFunction, false);


  var _getDOMEventName = Prototype.K;

  if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED) {
    _getDOMEventName = function(eventName) {
      var translations = { mouseenter: "mouseover", mouseleave: "mouseout" };
      return eventName in translations ? translations[eventName] : eventName;
    };
  }

  function observe(element, eventName, handler) {
    element = $(element);

    var responder = _createResponder(element, eventName, handler);

    if (!responder) return element;

    if (eventName.include(':')) {
      if (element.addEventListener)
        element.addEventListener("dataavailable", responder, false);
      else {
        element.attachEvent("ondataavailable", responder);
        element.attachEvent("onfilterchange", responder);
      }
    } else {
      var actualEventName = _getDOMEventName(eventName);

      if (element.addEventListener)
        element.addEventListener(actualEventName, responder, false);
      else
        element.attachEvent("on" + actualEventName, responder);
    }

    return element;
  }

  function stopObserving(element, eventName, handler) {
    element = $(element);

    var registry = Element.retrieve(element, 'prototype_event_registry');

    if (Object.isUndefined(registry)) return element;

    if (eventName && !handler) {
      var responders = registry.get(eventName);

      if (Object.isUndefined(responders)) return element;

      responders.each( function(r) {
        Element.stopObserving(element, eventName, r.handler);
      });
      return element;
    } else if (!eventName) {
      registry.each( function(pair) {
        var eventName = pair.key, responders = pair.value;

        responders.each( function(r) {
          Element.stopObserving(element, eventName, r.handler);
        });
      });
      return element;
    }

    var responders = registry.get(eventName);

    if (!responders) return;

    var responder = responders.find( function(r) { return r.handler === handler; });
    if (!responder) return element;

    var actualEventName = _getDOMEventName(eventName);

    if (eventName.include(':')) {
      if (element.removeEventListener)
        element.removeEventListener("dataavailable", responder, false);
      else {
        element.detachEvent("ondataavailable", responder);
        element.detachEvent("onfilterchange",  responder);
      }
    } else {
      if (element.removeEventListener)
        element.removeEventListener(actualEventName, responder, false);
      else
        element.detachEvent('on' + actualEventName, responder);
    }

    registry.set(eventName, responders.without(responder));

    return element;
  }

  function fire(element, eventName, memo, bubble) {
    element = $(element);

    if (Object.isUndefined(bubble))
      bubble = true;

    if (element == document && document.createEvent && !element.dispatchEvent)
      element = document.documentElement;

    var event;
    if (document.createEvent) {
      event = document.createEvent('HTMLEvents');
      event.initEvent('dataavailable', true, true);
    } else {
      event = document.createEventObject();
      event.eventType = bubble ? 'ondataavailable' : 'onfilterchange';
    }

    event.eventName = eventName;
    event.memo = memo || { };

    if (document.createEvent)
      element.dispatchEvent(event);
    else
      element.fireEvent(event.eventType, event);

    return Event.extend(event);
  }


  Object.extend(Event, Event.Methods);

  Object.extend(Event, {
    fire:          fire,
    observe:       observe,
    stopObserving: stopObserving
  });

  Element.addMethods({
    fire:          fire,

    observe:       observe,

    stopObserving: stopObserving
  });

  Object.extend(document, {
    fire:          fire.methodize(),

    observe:       observe.methodize(),

    stopObserving: stopObserving.methodize(),

    loaded:        false
  });

  if (window.Event) Object.extend(window.Event, Event);
  else window.Event = Event;
})();

(function() {
  /* Support for the DOMContentLoaded event is based on work by Dan Webb,
     Matthias Miller, Dean Edwards, John Resig, and Diego Perini. */

  var timer;

  function fireContentLoadedEvent() {
    if (document.loaded) return;
    if (timer) window.clearTimeout(timer);
    document.loaded = true;
    document.fire('dom:loaded');
  }

  function checkReadyState() {
    if (document.readyState === 'complete') {
      document.stopObserving('readystatechange', checkReadyState);
      fireContentLoadedEvent();
    }
  }

  function pollDoScroll() {
    try { document.documentElement.doScroll('left'); }
    catch(e) {
      timer = pollDoScroll.defer();
      return;
    }
    fireContentLoadedEvent();
  }

  if (document.addEventListener) {
    document.addEventListener('DOMContentLoaded', fireContentLoadedEvent, false);
  } else {
    document.observe('readystatechange', checkReadyState);
    if (window == top)
      timer = pollDoScroll.defer();
  }

  Event.observe(window, 'load', fireContentLoadedEvent);
})();

Element.addMethods();

/*------------------------------- DEPRECATED -------------------------------*/

Hash.toQueryString = Object.toQueryString;

var Toggle = { display: Element.toggle };

Element.Methods.childOf = Element.Methods.descendantOf;

var Insertion = {
  Before: function(element, content) {
    return Element.insert(element, {before:content});
  },

  Top: function(element, content) {
    return Element.insert(element, {top:content});
  },

  Bottom: function(element, content) {
    return Element.insert(element, {bottom:content});
  },

  After: function(element, content) {
    return Element.insert(element, {after:content});
  }
};

var $continue = new Error('"throw $continue" is deprecated, use "return" instead');

var Position = {
  includeScrollOffsets: false,

  prepare: function() {
    this.deltaX =  window.pageXOffset
                || document.documentElement.scrollLeft
                || document.body.scrollLeft
                || 0;
    this.deltaY =  window.pageYOffset
                || document.documentElement.scrollTop
                || document.body.scrollTop
                || 0;
  },

  within: function(element, x, y) {
    if (this.includeScrollOffsets)
      return this.withinIncludingScrolloffsets(element, x, y);
    this.xcomp = x;
    this.ycomp = y;
    this.offset = Element.cumulativeOffset(element);

    return (y >= this.offset[1] &&
            y <  this.offset[1] + element.offsetHeight &&
            x >= this.offset[0] &&
            x <  this.offset[0] + element.offsetWidth);
  },

  withinIncludingScrolloffsets: function(element, x, y) {
    var offsetcache = Element.cumulativeScrollOffset(element);

    this.xcomp = x + offsetcache[0] - this.deltaX;
    this.ycomp = y + offsetcache[1] - this.deltaY;
    this.offset = Element.cumulativeOffset(element);

    return (this.ycomp >= this.offset[1] &&
            this.ycomp <  this.offset[1] + element.offsetHeight &&
            this.xcomp >= this.offset[0] &&
            this.xcomp <  this.offset[0] + element.offsetWidth);
  },

  overlap: function(mode, element) {
    if (!mode) return 0;
    if (mode == 'vertical')
      return ((this.offset[1] + element.offsetHeight) - this.ycomp) /
        element.offsetHeight;
    if (mode == 'horizontal')
      return ((this.offset[0] + element.offsetWidth) - this.xcomp) /
        element.offsetWidth;
  },


  cumulativeOffset: Element.Methods.cumulativeOffset,

  positionedOffset: Element.Methods.positionedOffset,

  absolutize: function(element) {
    Position.prepare();
    return Element.absolutize(element);
  },

  relativize: function(element) {
    Position.prepare();
    return Element.relativize(element);
  },

  realOffset: Element.Methods.cumulativeScrollOffset,

  offsetParent: Element.Methods.getOffsetParent,

  page: Element.Methods.viewportOffset,

  clone: function(source, target, options) {
    options = options || { };
    return Element.clonePosition(target, source, options);
  }
};

/*--------------------------------------------------------------------------*/

if (!document.getElementsByClassName) document.getElementsByClassName = function(instanceMethods){
  function iter(name) {
    return name.blank() ? null : "[contains(concat(' ', @class, ' '), ' " + name + " ')]";
  }

  instanceMethods.getElementsByClassName = Prototype.BrowserFeatures.XPath ?
  function(element, className) {
    className = className.toString().strip();
    var cond = /\s/.test(className) ? $w(className).map(iter).join('') : iter(className);
    return cond ? document._getElementsByXPath('.//*' + cond, element) : [];
  } : function(element, className) {
    className = className.toString().strip();
    var elements = [], classNames = (/\s/.test(className) ? $w(className) : null);
    if (!classNames && !className) return elements;

    var nodes = $(element).getElementsByTagName('*');
    className = ' ' + className + ' ';

    for (var i = 0, child, cn; child = nodes[i]; i++) {
      if (child.className && (cn = ' ' + child.className + ' ') && (cn.include(className) ||
          (classNames && classNames.all(function(name) {
            return !name.toString().blank() && cn.include(' ' + name + ' ');
          }))))
        elements.push(Element.extend(child));
    }
    return elements;
  };

  return function(className, parentElement) {
    return $(parentElement || document.body).getElementsByClassName(className);
  };
}(Element.Methods);

/*--------------------------------------------------------------------------*/

Element.ClassNames = Class.create();
Element.ClassNames.prototype = {
  initialize: function(element) {
    this.element = $(element);
  },

  _each: function(iterator) {
    this.element.className.split(/\s+/).select(function(name) {
      return name.length > 0;
    })._each(iterator);
  },

  set: function(className) {
    this.element.className = className;
  },

  add: function(classNameToAdd) {
    if (this.include(classNameToAdd)) return;
    this.set($A(this).concat(classNameToAdd).join(' '));
  },

  remove: function(classNameToRemove) {
    if (!this.include(classNameToRemove)) return;
    this.set($A(this).without(classNameToRemove).join(' '));
  },

  toString: function() {
    return $A(this).join(' ');
  }
};

Object.extend(Element.ClassNames.prototype, Enumerable);

/*--------------------------------------------------------------------------*/
// script.aculo.us effects.js v1.8.2, Tue Nov 18 18:30:58 +0100 2008

// Copyright (c) 2005-2008 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
// Contributors:
//  Justin Palmer (http://encytemedia.com/)
//  Mark Pilgrim (http://diveintomark.org/)
//  Martin Bialasinki
//
// script.aculo.us is freely distributable under the terms of an MIT-style license.
// For details, see the script.aculo.us web site: http://script.aculo.us/

// converts rgb() and #xxx to #xxxxxx format,
// returns self (or first argument) if not convertable
String.prototype.parseColor = function() {
  var color = '#';
  if (this.slice(0,4) == 'rgb(') {
    var cols = this.slice(4,this.length-1).split(',');
    var i=0; do { color += parseInt(cols[i]).toColorPart() } while (++i<3);
  } else {
    if (this.slice(0,1) == '#') {
      if (this.length==4) for(var i=1;i<4;i++) color += (this.charAt(i) + this.charAt(i)).toLowerCase();
      if (this.length==7) color = this.toLowerCase();
    }
  }
  return (color.length==7 ? color : (arguments[0] || this));
};

/*--------------------------------------------------------------------------*/

Element.collectTextNodes = function(element) {
  return $A($(element).childNodes).collect( function(node) {
    return (node.nodeType==3 ? node.nodeValue :
      (node.hasChildNodes() ? Element.collectTextNodes(node) : ''));
  }).flatten().join('');
};

Element.collectTextNodesIgnoreClass = function(element, className) {
  return $A($(element).childNodes).collect( function(node) {
    return (node.nodeType==3 ? node.nodeValue :
      ((node.hasChildNodes() && !Element.hasClassName(node,className)) ?
        Element.collectTextNodesIgnoreClass(node, className) : ''));
  }).flatten().join('');
};

Element.setContentZoom = function(element, percent) {
  element = $(element);
  element.setStyle({fontSize: (percent/100) + 'em'});
  if (Prototype.Browser.WebKit) window.scrollBy(0,0);
  return element;
};

Element.getInlineOpacity = function(element){
  return $(element).style.opacity || '';
};

Element.forceRerendering = function(element) {
  try {
    element = $(element);
    var n = document.createTextNode(' ');
    element.appendChild(n);
    element.removeChild(n);
  } catch(e) { }
};

/*--------------------------------------------------------------------------*/

var Effect = {
  _elementDoesNotExistError: {
    name: 'ElementDoesNotExistError',
    message: 'The specified DOM element does not exist, but is required for this effect to operate'
  },
  Transitions: {
    linear: Prototype.K,
    sinoidal: function(pos) {
      return (-Math.cos(pos*Math.PI)/2) + .5;
    },
    reverse: function(pos) {
      return 1-pos;
    },
    flicker: function(pos) {
      var pos = ((-Math.cos(pos*Math.PI)/4) + .75) + Math.random()/4;
      return pos > 1 ? 1 : pos;
    },
    wobble: function(pos) {
      return (-Math.cos(pos*Math.PI*(9*pos))/2) + .5;
    },
    pulse: function(pos, pulses) {
      return (-Math.cos((pos*((pulses||5)-.5)*2)*Math.PI)/2) + .5;
    },
    spring: function(pos) {
      return 1 - (Math.cos(pos * 4.5 * Math.PI) * Math.exp(-pos * 6));
    },
    none: function(pos) {
      return 0;
    },
    full: function(pos) {
      return 1;
    }
  },
  DefaultOptions: {
    duration:   1.0,   // seconds
    fps:        100,   // 100= assume 66fps max.
    sync:       false, // true for combining
    from:       0.0,
    to:         1.0,
    delay:      0.0,
    queue:      'parallel'
  },
  tagifyText: function(element) {
    var tagifyStyle = 'position:relative';
    if (Prototype.Browser.IE) tagifyStyle += ';zoom:1';

    element = $(element);
    $A(element.childNodes).each( function(child) {
      if (child.nodeType==3) {
        child.nodeValue.toArray().each( function(character) {
          element.insertBefore(
            new Element('span', {style: tagifyStyle}).update(
              character == ' ' ? String.fromCharCode(160) : character),
              child);
        });
        Element.remove(child);
      }
    });
  },
  multiple: function(element, effect) {
    var elements;
    if (((typeof element == 'object') ||
        Object.isFunction(element)) &&
       (element.length))
      elements = element;
    else
      elements = $(element).childNodes;

    var options = Object.extend({
      speed: 0.1,
      delay: 0.0
    }, arguments[2] || { });
    var masterDelay = options.delay;

    $A(elements).each( function(element, index) {
      new effect(element, Object.extend(options, { delay: index * options.speed + masterDelay }));
    });
  },
  PAIRS: {
    'slide':  ['SlideDown','SlideUp'],
    'blind':  ['BlindDown','BlindUp'],
    'appear': ['Appear','Fade']
  },
  toggle: function(element, effect) {
    element = $(element);
    effect = (effect || 'appear').toLowerCase();
    var options = Object.extend({
      queue: { position:'end', scope:(element.id || 'global'), limit: 1 }
    }, arguments[2] || { });
    Effect[element.visible() ?
      Effect.PAIRS[effect][1] : Effect.PAIRS[effect][0]](element, options);
  }
};

Effect.DefaultOptions.transition = Effect.Transitions.sinoidal;

/* ------------- core effects ------------- */

Effect.ScopedQueue = Class.create(Enumerable, {
  initialize: function() {
    this.effects  = [];
    this.interval = null;
  },
  _each: function(iterator) {
    this.effects._each(iterator);
  },
  add: function(effect) {
    var timestamp = new Date().getTime();

    var position = Object.isString(effect.options.queue) ?
      effect.options.queue : effect.options.queue.position;

    switch(position) {
      case 'front':
        // move unstarted effects after this effect
        this.effects.findAll(function(e){ return e.state=='idle' }).each( function(e) {
            e.startOn  += effect.finishOn;
            e.finishOn += effect.finishOn;
          });
        break;
      case 'with-last':
        timestamp = this.effects.pluck('startOn').max() || timestamp;
        break;
      case 'end':
        // start effect after last queued effect has finished
        timestamp = this.effects.pluck('finishOn').max() || timestamp;
        break;
    }

    effect.startOn  += timestamp;
    effect.finishOn += timestamp;

    if (!effect.options.queue.limit || (this.effects.length < effect.options.queue.limit))
      this.effects.push(effect);

    if (!this.interval)
      this.interval = setInterval(this.loop.bind(this), 15);
  },
  remove: function(effect) {
    this.effects = this.effects.reject(function(e) { return e==effect });
    if (this.effects.length == 0) {
      clearInterval(this.interval);
      this.interval = null;
    }
  },
  loop: function() {
    var timePos = new Date().getTime();
    for(var i=0, len=this.effects.length;i<len;i++)
      this.effects[i] && this.effects[i].loop(timePos);
  }
});

Effect.Queues = {
  instances: $H(),
  get: function(queueName) {
    if (!Object.isString(queueName)) return queueName;

    return this.instances.get(queueName) ||
      this.instances.set(queueName, new Effect.ScopedQueue());
  }
};
Effect.Queue = Effect.Queues.get('global');

Effect.Base = Class.create({
  position: null,
  start: function(options) {
    function codeForEvent(options,eventName){
      return (
        (options[eventName+'Internal'] ? 'this.options.'+eventName+'Internal(this);' : '') +
        (options[eventName] ? 'this.options.'+eventName+'(this);' : '')
      );
    }
    if (options && options.transition === false) options.transition = Effect.Transitions.linear;
    this.options      = Object.extend(Object.extend({ },Effect.DefaultOptions), options || { });
    this.currentFrame = 0;
    this.state        = 'idle';
    this.startOn      = this.options.delay*1000;
    this.finishOn     = this.startOn+(this.options.duration*1000);
    this.fromToDelta  = this.options.to-this.options.from;
    this.totalTime    = this.finishOn-this.startOn;
    this.totalFrames  = this.options.fps*this.options.duration;

    this.render = (function() {
      function dispatch(effect, eventName) {
        if (effect.options[eventName + 'Internal'])
          effect.options[eventName + 'Internal'](effect);
        if (effect.options[eventName])
          effect.options[eventName](effect);
      }

      return function(pos) {
        if (this.state === "idle") {
          this.state = "running";
          dispatch(this, 'beforeSetup');
          if (this.setup) this.setup();
          dispatch(this, 'afterSetup');
        }
        if (this.state === "running") {
          pos = (this.options.transition(pos) * this.fromToDelta) + this.options.from;
          this.position = pos;
          dispatch(this, 'beforeUpdate');
          if (this.update) this.update(pos);
          dispatch(this, 'afterUpdate');
        }
      };
    })();

    this.event('beforeStart');
    if (!this.options.sync)
      Effect.Queues.get(Object.isString(this.options.queue) ?
        'global' : this.options.queue.scope).add(this);
  },
  loop: function(timePos) {
    if (timePos >= this.startOn) {
      if (timePos >= this.finishOn) {
        this.render(1.0);
        this.cancel();
        this.event('beforeFinish');
        if (this.finish) this.finish();
        this.event('afterFinish');
        return;
      }
      var pos   = (timePos - this.startOn) / this.totalTime,
          frame = (pos * this.totalFrames).round();
      if (frame > this.currentFrame) {
        this.render(pos);
        this.currentFrame = frame;
      }
    }
  },
  cancel: function() {
    if (!this.options.sync)
      Effect.Queues.get(Object.isString(this.options.queue) ?
        'global' : this.options.queue.scope).remove(this);
    this.state = 'finished';
  },
  event: function(eventName) {
    if (this.options[eventName + 'Internal']) this.options[eventName + 'Internal'](this);
    if (this.options[eventName]) this.options[eventName](this);
  },
  inspect: function() {
    var data = $H();
    for(property in this)
      if (!Object.isFunction(this[property])) data.set(property, this[property]);
    return '#<Effect:' + data.inspect() + ',options:' + $H(this.options).inspect() + '>';
  }
});

Effect.Parallel = Class.create(Effect.Base, {
  initialize: function(effects) {
    this.effects = effects || [];
    this.start(arguments[1]);
  },
  update: function(position) {
    this.effects.invoke('render', position);
  },
  finish: function(position) {
    this.effects.each( function(effect) {
      effect.render(1.0);
      effect.cancel();
      effect.event('beforeFinish');
      if (effect.finish) effect.finish(position);
      effect.event('afterFinish');
    });
  }
});

Effect.Tween = Class.create(Effect.Base, {
  initialize: function(object, from, to) {
    object = Object.isString(object) ? $(object) : object;
    var args = $A(arguments), method = args.last(),
      options = args.length == 5 ? args[3] : null;
    this.method = Object.isFunction(method) ? method.bind(object) :
      Object.isFunction(object[method]) ? object[method].bind(object) :
      function(value) { object[method] = value };
    this.start(Object.extend({ from: from, to: to }, options || { }));
  },
  update: function(position) {
    this.method(position);
  }
});

Effect.Event = Class.create(Effect.Base, {
  initialize: function() {
    this.start(Object.extend({ duration: 0 }, arguments[0] || { }));
  },
  update: Prototype.emptyFunction
});

Effect.Opacity = Class.create(Effect.Base, {
  initialize: function(element) {
    this.element = $(element);
    if (!this.element) throw(Effect._elementDoesNotExistError);
    // make this work on IE on elements without 'layout'
    if (Prototype.Browser.IE && (!this.element.currentStyle.hasLayout))
      this.element.setStyle({zoom: 1});
    var options = Object.extend({
      from: this.element.getOpacity() || 0.0,
      to:   1.0
    }, arguments[1] || { });
    this.start(options);
  },
  update: function(position) {
    this.element.setOpacity(position);
  }
});

Effect.Move = Class.create(Effect.Base, {
  initialize: function(element) {
    this.element = $(element);
    if (!this.element) throw(Effect._elementDoesNotExistError);
    var options = Object.extend({
      x:    0,
      y:    0,
      mode: 'relative'
    }, arguments[1] || { });
    this.start(options);
  },
  setup: function() {
    this.element.makePositioned();
    this.originalLeft = parseFloat(this.element.getStyle('left') || '0');
    this.originalTop  = parseFloat(this.element.getStyle('top')  || '0');
    if (this.options.mode == 'absolute') {
      this.options.x = this.options.x - this.originalLeft;
      this.options.y = this.options.y - this.originalTop;
    }
  },
  update: function(position) {
    this.element.setStyle({
      left: (this.options.x  * position + this.originalLeft).round() + 'px',
      top:  (this.options.y  * position + this.originalTop).round()  + 'px'
    });
  }
});

// for backwards compatibility
Effect.MoveBy = function(element, toTop, toLeft) {
  return new Effect.Move(element,
    Object.extend({ x: toLeft, y: toTop }, arguments[3] || { }));
};

Effect.Scale = Class.create(Effect.Base, {
  initialize: function(element, percent) {
    this.element = $(element);
    if (!this.element) throw(Effect._elementDoesNotExistError);
    var options = Object.extend({
      scaleX: true,
      scaleY: true,
      scaleContent: true,
      scaleFromCenter: false,
      scaleMode: 'box',        // 'box' or 'contents' or { } with provided values
      scaleFrom: 100.0,
      scaleTo:   percent
    }, arguments[2] || { });
    this.start(options);
  },
  setup: function() {
    this.restoreAfterFinish = this.options.restoreAfterFinish || false;
    this.elementPositioning = this.element.getStyle('position');

    this.originalStyle = { };
    ['top','left','width','height','fontSize'].each( function(k) {
      this.originalStyle[k] = this.element.style[k];
    }.bind(this));

    this.originalTop  = this.element.offsetTop;
    this.originalLeft = this.element.offsetLeft;

    var fontSize = this.element.getStyle('font-size') || '100%';
    ['em','px','%','pt'].each( function(fontSizeType) {
      if (fontSize.indexOf(fontSizeType)>0) {
        this.fontSize     = parseFloat(fontSize);
        this.fontSizeType = fontSizeType;
      }
    }.bind(this));

    this.factor = (this.options.scaleTo - this.options.scaleFrom)/100;

    this.dims = null;
    if (this.options.scaleMode=='box')
      this.dims = [this.element.offsetHeight, this.element.offsetWidth];
    if (/^content/.test(this.options.scaleMode))
      this.dims = [this.element.scrollHeight, this.element.scrollWidth];
    if (!this.dims)
      this.dims = [this.options.scaleMode.originalHeight,
                   this.options.scaleMode.originalWidth];
  },
  update: function(position) {
    var currentScale = (this.options.scaleFrom/100.0) + (this.factor * position);
    if (this.options.scaleContent && this.fontSize)
      this.element.setStyle({fontSize: this.fontSize * currentScale + this.fontSizeType });
    this.setDimensions(this.dims[0] * currentScale, this.dims[1] * currentScale);
  },
  finish: function(position) {
    if (this.restoreAfterFinish) this.element.setStyle(this.originalStyle);
  },
  setDimensions: function(height, width) {
    var d = { };
    if (this.options.scaleX) d.width = width.round() + 'px';
    if (this.options.scaleY) d.height = height.round() + 'px';
    if (this.options.scaleFromCenter) {
      var topd  = (height - this.dims[0])/2;
      var leftd = (width  - this.dims[1])/2;
      if (this.elementPositioning == 'absolute') {
        if (this.options.scaleY) d.top = this.originalTop-topd + 'px';
        if (this.options.scaleX) d.left = this.originalLeft-leftd + 'px';
      } else {
        if (this.options.scaleY) d.top = -topd + 'px';
        if (this.options.scaleX) d.left = -leftd + 'px';
      }
    }
    this.element.setStyle(d);
  }
});

Effect.Highlight = Class.create(Effect.Base, {
  initialize: function(element) {
    this.element = $(element);
    if (!this.element) throw(Effect._elementDoesNotExistError);
    var options = Object.extend({ startcolor: '#ffff99' }, arguments[1] || { });
    this.start(options);
  },
  setup: function() {
    // Prevent executing on elements not in the layout flow
    if (this.element.getStyle('display')=='none') { this.cancel(); return; }
    // Disable background image during the effect
    this.oldStyle = { };
    if (!this.options.keepBackgroundImage) {
      this.oldStyle.backgroundImage = this.element.getStyle('background-image');
      this.element.setStyle({backgroundImage: 'none'});
    }
    if (!this.options.endcolor)
      this.options.endcolor = this.element.getStyle('background-color').parseColor('#ffffff');
    if (!this.options.restorecolor)
      this.options.restorecolor = this.element.getStyle('background-color');
    // init color calculations
    this._base  = $R(0,2).map(function(i){ return parseInt(this.options.startcolor.slice(i*2+1,i*2+3),16) }.bind(this));
    this._delta = $R(0,2).map(function(i){ return parseInt(this.options.endcolor.slice(i*2+1,i*2+3),16)-this._base[i] }.bind(this));
  },
  update: function(position) {
    this.element.setStyle({backgroundColor: $R(0,2).inject('#',function(m,v,i){
      return m+((this._base[i]+(this._delta[i]*position)).round().toColorPart()); }.bind(this)) });
  },
  finish: function() {
    this.element.setStyle(Object.extend(this.oldStyle, {
      backgroundColor: this.options.restorecolor
    }));
  }
});

Effect.ScrollTo = function(element) {
  var options = arguments[1] || { },
  scrollOffsets = document.viewport.getScrollOffsets(),
  elementOffsets = $(element).cumulativeOffset();

  if (options.offset) elementOffsets[1] += options.offset;

  return new Effect.Tween(null,
    scrollOffsets.top,
    elementOffsets[1],
    options,
    function(p){ scrollTo(scrollOffsets.left, p.round()); }
  );
};

/* ------------- combination effects ------------- */

Effect.Fade = function(element) {
  element = $(element);
  var oldOpacity = element.getInlineOpacity();
  var options = Object.extend({
    from: element.getOpacity() || 1.0,
    to:   0.0,
    afterFinishInternal: function(effect) {
      if (effect.options.to!=0) return;
      effect.element.hide().setStyle({opacity: oldOpacity});
    }
  }, arguments[1] || { });
  return new Effect.Opacity(element,options);
};

Effect.Appear = function(element) {
  element = $(element);
  var options = Object.extend({
  from: (element.getStyle('display') == 'none' ? 0.0 : element.getOpacity() || 0.0),
  to:   1.0,
  // force Safari to render floated elements properly
  afterFinishInternal: function(effect) {
    effect.element.forceRerendering();
  },
  beforeSetup: function(effect) {
    effect.element.setOpacity(effect.options.from).show();
  }}, arguments[1] || { });
  return new Effect.Opacity(element,options);
};

Effect.Puff = function(element) {
  element = $(element);
  var oldStyle = {
    opacity: element.getInlineOpacity(),
    position: element.getStyle('position'),
    top:  element.style.top,
    left: element.style.left,
    width: element.style.width,
    height: element.style.height
  };
  return new Effect.Parallel(
   [ new Effect.Scale(element, 200,
      { sync: true, scaleFromCenter: true, scaleContent: true, restoreAfterFinish: true }),
     new Effect.Opacity(element, { sync: true, to: 0.0 } ) ],
     Object.extend({ duration: 1.0,
      beforeSetupInternal: function(effect) {
        Position.absolutize(effect.effects[0].element);
      },
      afterFinishInternal: function(effect) {
         effect.effects[0].element.hide().setStyle(oldStyle); }
     }, arguments[1] || { })
   );
};

Effect.BlindUp = function(element) {
  element = $(element);
  element.makeClipping();
  return new Effect.Scale(element, 0,
    Object.extend({ scaleContent: false,
      scaleX: false,
      restoreAfterFinish: true,
      afterFinishInternal: function(effect) {
        effect.element.hide().undoClipping();
      }
    }, arguments[1] || { })
  );
};

Effect.BlindDown = function(element) {
  element = $(element);
  var elementDimensions = element.getDimensions();
  return new Effect.Scale(element, 100, Object.extend({
    scaleContent: false,
    scaleX: false,
    scaleFrom: 0,
    scaleMode: {originalHeight: elementDimensions.height, originalWidth: elementDimensions.width},
    restoreAfterFinish: true,
    afterSetup: function(effect) {
      effect.element.makeClipping().setStyle({height: '0px'}).show();
    },
    afterFinishInternal: function(effect) {
      effect.element.undoClipping();
    }
  }, arguments[1] || { }));
};

Effect.SwitchOff = function(element) {
  element = $(element);
  var oldOpacity = element.getInlineOpacity();
  return new Effect.Appear(element, Object.extend({
    duration: 0.4,
    from: 0,
    transition: Effect.Transitions.flicker,
    afterFinishInternal: function(effect) {
      new Effect.Scale(effect.element, 1, {
        duration: 0.3, scaleFromCenter: true,
        scaleX: false, scaleContent: false, restoreAfterFinish: true,
        beforeSetup: function(effect) {
          effect.element.makePositioned().makeClipping();
        },
        afterFinishInternal: function(effect) {
          effect.element.hide().undoClipping().undoPositioned().setStyle({opacity: oldOpacity});
        }
      });
    }
  }, arguments[1] || { }));
};

Effect.DropOut = function(element) {
  element = $(element);
  var oldStyle = {
    top: element.getStyle('top'),
    left: element.getStyle('left'),
    opacity: element.getInlineOpacity() };
  return new Effect.Parallel(
    [ new Effect.Move(element, {x: 0, y: 100, sync: true }),
      new Effect.Opacity(element, { sync: true, to: 0.0 }) ],
    Object.extend(
      { duration: 0.5,
        beforeSetup: function(effect) {
          effect.effects[0].element.makePositioned();
        },
        afterFinishInternal: function(effect) {
          effect.effects[0].element.hide().undoPositioned().setStyle(oldStyle);
        }
      }, arguments[1] || { }));
};

Effect.Shake = function(element) {
  element = $(element);
  var options = Object.extend({
    distance: 20,
    duration: 0.5
  }, arguments[1] || {});
  var distance = parseFloat(options.distance);
  var split = parseFloat(options.duration) / 10.0;
  var oldStyle = {
    top: element.getStyle('top'),
    left: element.getStyle('left') };
    return new Effect.Move(element,
      { x:  distance, y: 0, duration: split, afterFinishInternal: function(effect) {
    new Effect.Move(effect.element,
      { x: -distance*2, y: 0, duration: split*2,  afterFinishInternal: function(effect) {
    new Effect.Move(effect.element,
      { x:  distance*2, y: 0, duration: split*2,  afterFinishInternal: function(effect) {
    new Effect.Move(effect.element,
      { x: -distance*2, y: 0, duration: split*2,  afterFinishInternal: function(effect) {
    new Effect.Move(effect.element,
      { x:  distance*2, y: 0, duration: split*2,  afterFinishInternal: function(effect) {
    new Effect.Move(effect.element,
      { x: -distance, y: 0, duration: split, afterFinishInternal: function(effect) {
        effect.element.undoPositioned().setStyle(oldStyle);
  }}); }}); }}); }}); }}); }});
};

Effect.SlideDown = function(element) {
  element = $(element).cleanWhitespace();
  // SlideDown need to have the content of the element wrapped in a container element with fixed height!
  var oldInnerBottom = element.down().getStyle('bottom');
  var elementDimensions = element.getDimensions();
  return new Effect.Scale(element, 100, Object.extend({
    scaleContent: false,
    scaleX: false,
    scaleFrom: window.opera ? 0 : 1,
    scaleMode: {originalHeight: elementDimensions.height, originalWidth: elementDimensions.width},
    restoreAfterFinish: true,
    afterSetup: function(effect) {
      effect.element.makePositioned();
      effect.element.down().makePositioned();
      if (window.opera) effect.element.setStyle({top: ''});
      effect.element.makeClipping().setStyle({height: '0px'}).show();
    },
    afterUpdateInternal: function(effect) {
      effect.element.down().setStyle({bottom:
        (effect.dims[0] - effect.element.clientHeight) + 'px' });
    },
    afterFinishInternal: function(effect) {
      effect.element.undoClipping().undoPositioned();
      effect.element.down().undoPositioned().setStyle({bottom: oldInnerBottom}); }
    }, arguments[1] || { })
  );
};

Effect.SlideUp = function(element) {
  element = $(element).cleanWhitespace();
  var oldInnerBottom = element.down().getStyle('bottom');
  var elementDimensions = element.getDimensions();
  return new Effect.Scale(element, window.opera ? 0 : 1,
   Object.extend({ scaleContent: false,
    scaleX: false,
    scaleMode: 'box',
    scaleFrom: 100,
    scaleMode: {originalHeight: elementDimensions.height, originalWidth: elementDimensions.width},
    restoreAfterFinish: true,
    afterSetup: function(effect) {
      effect.element.makePositioned();
      effect.element.down().makePositioned();
      if (window.opera) effect.element.setStyle({top: ''});
      effect.element.makeClipping().show();
    },
    afterUpdateInternal: function(effect) {
      effect.element.down().setStyle({bottom:
        (effect.dims[0] - effect.element.clientHeight) + 'px' });
    },
    afterFinishInternal: function(effect) {
      effect.element.hide().undoClipping().undoPositioned();
      effect.element.down().undoPositioned().setStyle({bottom: oldInnerBottom});
    }
   }, arguments[1] || { })
  );
};

// Bug in opera makes the TD containing this element expand for a instance after finish
Effect.Squish = function(element) {
  return new Effect.Scale(element, window.opera ? 1 : 0, {
    restoreAfterFinish: true,
    beforeSetup: function(effect) {
      effect.element.makeClipping();
    },
    afterFinishInternal: function(effect) {
      effect.element.hide().undoClipping();
    }
  });
};

Effect.Grow = function(element) {
  element = $(element);
  var options = Object.extend({
    direction: 'center',
    moveTransition: Effect.Transitions.sinoidal,
    scaleTransition: Effect.Transitions.sinoidal,
    opacityTransition: Effect.Transitions.full
  }, arguments[1] || { });
  var oldStyle = {
    top: element.style.top,
    left: element.style.left,
    height: element.style.height,
    width: element.style.width,
    opacity: element.getInlineOpacity() };

  var dims = element.getDimensions();
  var initialMoveX, initialMoveY;
  var moveX, moveY;

  switch (options.direction) {
    case 'top-left':
      initialMoveX = initialMoveY = moveX = moveY = 0;
      break;
    case 'top-right':
      initialMoveX = dims.width;
      initialMoveY = moveY = 0;
      moveX = -dims.width;
      break;
    case 'bottom-left':
      initialMoveX = moveX = 0;
      initialMoveY = dims.height;
      moveY = -dims.height;
      break;
    case 'bottom-right':
      initialMoveX = dims.width;
      initialMoveY = dims.height;
      moveX = -dims.width;
      moveY = -dims.height;
      break;
    case 'center':
      initialMoveX = dims.width / 2;
      initialMoveY = dims.height / 2;
      moveX = -dims.width / 2;
      moveY = -dims.height / 2;
      break;
  }

  return new Effect.Move(element, {
    x: initialMoveX,
    y: initialMoveY,
    duration: 0.01,
    beforeSetup: function(effect) {
      effect.element.hide().makeClipping().makePositioned();
    },
    afterFinishInternal: function(effect) {
      new Effect.Parallel(
        [ new Effect.Opacity(effect.element, { sync: true, to: 1.0, from: 0.0, transition: options.opacityTransition }),
          new Effect.Move(effect.element, { x: moveX, y: moveY, sync: true, transition: options.moveTransition }),
          new Effect.Scale(effect.element, 100, {
            scaleMode: { originalHeight: dims.height, originalWidth: dims.width },
            sync: true, scaleFrom: window.opera ? 1 : 0, transition: options.scaleTransition, restoreAfterFinish: true})
        ], Object.extend({
             beforeSetup: function(effect) {
               effect.effects[0].element.setStyle({height: '0px'}).show();
             },
             afterFinishInternal: function(effect) {
               effect.effects[0].element.undoClipping().undoPositioned().setStyle(oldStyle);
             }
           }, options)
      );
    }
  });
};

Effect.Shrink = function(element) {
  element = $(element);
  var options = Object.extend({
    direction: 'center',
    moveTransition: Effect.Transitions.sinoidal,
    scaleTransition: Effect.Transitions.sinoidal,
    opacityTransition: Effect.Transitions.none
  }, arguments[1] || { });
  var oldStyle = {
    top: element.style.top,
    left: element.style.left,
    height: element.style.height,
    width: element.style.width,
    opacity: element.getInlineOpacity() };

  var dims = element.getDimensions();
  var moveX, moveY;

  switch (options.direction) {
    case 'top-left':
      moveX = moveY = 0;
      break;
    case 'top-right':
      moveX = dims.width;
      moveY = 0;
      break;
    case 'bottom-left':
      moveX = 0;
      moveY = dims.height;
      break;
    case 'bottom-right':
      moveX = dims.width;
      moveY = dims.height;
      break;
    case 'center':
      moveX = dims.width / 2;
      moveY = dims.height / 2;
      break;
  }

  return new Effect.Parallel(
    [ new Effect.Opacity(element, { sync: true, to: 0.0, from: 1.0, transition: options.opacityTransition }),
      new Effect.Scale(element, window.opera ? 1 : 0, { sync: true, transition: options.scaleTransition, restoreAfterFinish: true}),
      new Effect.Move(element, { x: moveX, y: moveY, sync: true, transition: options.moveTransition })
    ], Object.extend({
         beforeStartInternal: function(effect) {
           effect.effects[0].element.makePositioned().makeClipping();
         },
         afterFinishInternal: function(effect) {
           effect.effects[0].element.hide().undoClipping().undoPositioned().setStyle(oldStyle); }
       }, options)
  );
};

Effect.Pulsate = function(element) {
  element = $(element);
  var options    = arguments[1] || { },
    oldOpacity = element.getInlineOpacity(),
    transition = options.transition || Effect.Transitions.linear,
    reverser   = function(pos){
      return 1 - transition((-Math.cos((pos*(options.pulses||5)*2)*Math.PI)/2) + .5);
    };

  return new Effect.Opacity(element,
    Object.extend(Object.extend({  duration: 2.0, from: 0,
      afterFinishInternal: function(effect) { effect.element.setStyle({opacity: oldOpacity}); }
    }, options), {transition: reverser}));
};

Effect.Fold = function(element) {
  element = $(element);
  var oldStyle = {
    top: element.style.top,
    left: element.style.left,
    width: element.style.width,
    height: element.style.height };
  element.makeClipping();
  return new Effect.Scale(element, 5, Object.extend({
    scaleContent: false,
    scaleX: false,
    afterFinishInternal: function(effect) {
    new Effect.Scale(element, 1, {
      scaleContent: false,
      scaleY: false,
      afterFinishInternal: function(effect) {
        effect.element.hide().undoClipping().setStyle(oldStyle);
      } });
  }}, arguments[1] || { }));
};

Effect.Morph = Class.create(Effect.Base, {
  initialize: function(element) {
    this.element = $(element);
    if (!this.element) throw(Effect._elementDoesNotExistError);
    var options = Object.extend({
      style: { }
    }, arguments[1] || { });

    if (!Object.isString(options.style)) this.style = $H(options.style);
    else {
      if (options.style.include(':'))
        this.style = options.style.parseStyle();
      else {
        this.element.addClassName(options.style);
        this.style = $H(this.element.getStyles());
        this.element.removeClassName(options.style);
        var css = this.element.getStyles();
        this.style = this.style.reject(function(style) {
          return style.value == css[style.key];
        });
        options.afterFinishInternal = function(effect) {
          effect.element.addClassName(effect.options.style);
          effect.transforms.each(function(transform) {
            effect.element.style[transform.style] = '';
          });
        };
      }
    }
    this.start(options);
  },

  setup: function(){
    function parseColor(color){
      if (!color || ['rgba(0, 0, 0, 0)','transparent'].include(color)) color = '#ffffff';
      color = color.parseColor();
      return $R(0,2).map(function(i){
        return parseInt( color.slice(i*2+1,i*2+3), 16 );
      });
    }
    this.transforms = this.style.map(function(pair){
      var property = pair[0], value = pair[1], unit = null;

      if (value.parseColor('#zzzzzz') != '#zzzzzz') {
        value = value.parseColor();
        unit  = 'color';
      } else if (property == 'opacity') {
        value = parseFloat(value);
        if (Prototype.Browser.IE && (!this.element.currentStyle.hasLayout))
          this.element.setStyle({zoom: 1});
      } else if (Element.CSS_LENGTH.test(value)) {
          var components = value.match(/^([\+\-]?[0-9\.]+)(.*)$/);
          value = parseFloat(components[1]);
          unit = (components.length == 3) ? components[2] : null;
      }

      var originalValue = this.element.getStyle(property);
      return {
        style: property.camelize(),
        originalValue: unit=='color' ? parseColor(originalValue) : parseFloat(originalValue || 0),
        targetValue: unit=='color' ? parseColor(value) : value,
        unit: unit
      };
    }.bind(this)).reject(function(transform){
      return (
        (transform.originalValue == transform.targetValue) ||
        (
          transform.unit != 'color' &&
          (isNaN(transform.originalValue) || isNaN(transform.targetValue))
        )
      );
    });
  },
  update: function(position) {
    var style = { }, transform, i = this.transforms.length;
    while(i--)
      style[(transform = this.transforms[i]).style] =
        transform.unit=='color' ? '#'+
          (Math.round(transform.originalValue[0]+
            (transform.targetValue[0]-transform.originalValue[0])*position)).toColorPart() +
          (Math.round(transform.originalValue[1]+
            (transform.targetValue[1]-transform.originalValue[1])*position)).toColorPart() +
          (Math.round(transform.originalValue[2]+
            (transform.targetValue[2]-transform.originalValue[2])*position)).toColorPart() :
        (transform.originalValue +
          (transform.targetValue - transform.originalValue) * position).toFixed(3) +
            (transform.unit === null ? '' : transform.unit);
    this.element.setStyle(style, true);
  }
});

Effect.Transform = Class.create({
  initialize: function(tracks){
    this.tracks  = [];
    this.options = arguments[1] || { };
    this.addTracks(tracks);
  },
  addTracks: function(tracks){
    tracks.each(function(track){
      track = $H(track);
      var data = track.values().first();
      this.tracks.push($H({
        ids:     track.keys().first(),
        effect:  Effect.Morph,
        options: { style: data }
      }));
    }.bind(this));
    return this;
  },
  play: function(){
    return new Effect.Parallel(
      this.tracks.map(function(track){
        var ids = track.get('ids'), effect = track.get('effect'), options = track.get('options');
        var elements = [$(ids) || $$(ids)].flatten();
        return elements.map(function(e){ return new effect(e, Object.extend({ sync:true }, options)) });
      }).flatten(),
      this.options
    );
  }
});

Element.CSS_PROPERTIES = $w(
  'backgroundColor backgroundPosition borderBottomColor borderBottomStyle ' +
  'borderBottomWidth borderLeftColor borderLeftStyle borderLeftWidth ' +
  'borderRightColor borderRightStyle borderRightWidth borderSpacing ' +
  'borderTopColor borderTopStyle borderTopWidth bottom clip color ' +
  'fontSize fontWeight height left letterSpacing lineHeight ' +
  'marginBottom marginLeft marginRight marginTop markerOffset maxHeight '+
  'maxWidth minHeight minWidth opacity outlineColor outlineOffset ' +
  'outlineWidth paddingBottom paddingLeft paddingRight paddingTop ' +
  'right textIndent top width wordSpacing zIndex');

Element.CSS_LENGTH = /^(([\+\-]?[0-9\.]+)(em|ex|px|in|cm|mm|pt|pc|\%))|0$/;

String.__parseStyleElement = document.createElement('div');
String.prototype.parseStyle = function(){
  var style, styleRules = $H();
  if (Prototype.Browser.WebKit)
    style = new Element('div',{style:this}).style;
  else {
    String.__parseStyleElement.innerHTML = '<div style="' + this + '"></div>';
    style = String.__parseStyleElement.childNodes[0].style;
  }

  Element.CSS_PROPERTIES.each(function(property){
    if (style[property]) styleRules.set(property, style[property]);
  });

  if (Prototype.Browser.IE && this.include('opacity'))
    styleRules.set('opacity', this.match(/opacity:\s*((?:0|1)?(?:\.\d*)?)/)[1]);

  return styleRules;
};

if (document.defaultView && document.defaultView.getComputedStyle) {
  Element.getStyles = function(element) {
    var css = document.defaultView.getComputedStyle($(element), null);
    return Element.CSS_PROPERTIES.inject({ }, function(styles, property) {
      styles[property] = css[property];
      return styles;
    });
  };
} else {
  Element.getStyles = function(element) {
    element = $(element);
    var css = element.currentStyle, styles;
    styles = Element.CSS_PROPERTIES.inject({ }, function(results, property) {
      results[property] = css[property];
      return results;
    });
    if (!styles.opacity) styles.opacity = element.getOpacity();
    return styles;
  };
}

Effect.Methods = {
  morph: function(element, style) {
    element = $(element);
    new Effect.Morph(element, Object.extend({ style: style }, arguments[2] || { }));
    return element;
  },
  visualEffect: function(element, effect, options) {
    element = $(element);
    var s = effect.dasherize().camelize(), klass = s.charAt(0).toUpperCase() + s.substring(1);
    new Effect[klass](element, options);
    return element;
  },
  highlight: function(element, options) {
    element = $(element);
    new Effect.Highlight(element, options);
    return element;
  }
};

$w('fade appear grow shrink fold blindUp blindDown slideUp slideDown '+
  'pulsate shake puff squish switchOff dropOut').each(
  function(effect) {
    Effect.Methods[effect] = function(element, options){
      element = $(element);
      Effect[effect.charAt(0).toUpperCase() + effect.substring(1)](element, options);
      return element;
    };
  }
);

$w('getInlineOpacity forceRerendering setContentZoom collectTextNodes collectTextNodesIgnoreClass getStyles').each(
  function(f) { Effect.Methods[f] = Element[f]; }
);

Element.addMethods(Effect.Methods);// script.aculo.us builder.js v1.8.2, Tue Nov 18 18:30:58 +0100 2008

// Copyright (c) 2005-2008 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
//
// script.aculo.us is freely distributable under the terms of an MIT-style license.
// For details, see the script.aculo.us web site: http://script.aculo.us/

var Builder = {
  NODEMAP: {
    AREA: 'map',
    CAPTION: 'table',
    COL: 'table',
    COLGROUP: 'table',
    LEGEND: 'fieldset',
    OPTGROUP: 'select',
    OPTION: 'select',
    PARAM: 'object',
    TBODY: 'table',
    TD: 'table',
    TFOOT: 'table',
    TH: 'table',
    THEAD: 'table',
    TR: 'table'
  },
  // note: For Firefox < 1.5, OPTION and OPTGROUP tags are currently broken,
  //       due to a Firefox bug
  node: function(elementName) {
    elementName = elementName.toUpperCase();

    // try innerHTML approach
    var parentTag = this.NODEMAP[elementName] || 'div';
    var parentElement = document.createElement(parentTag);
    try { // prevent IE "feature": http://dev.rubyonrails.org/ticket/2707
      parentElement.innerHTML = "<" + elementName + "></" + elementName + ">";
    } catch(e) {}
    var element = parentElement.firstChild || null;

    // see if browser added wrapping tags
    if(element && (element.tagName.toUpperCase() != elementName))
      element = element.getElementsByTagName(elementName)[0];

    // fallback to createElement approach
    if(!element) element = document.createElement(elementName);

    // abort if nothing could be created
    if(!element) return;

    // attributes (or text)
    if(arguments[1])
      if(this._isStringOrNumber(arguments[1]) ||
        (arguments[1] instanceof Array) ||
        arguments[1].tagName) {
          this._children(element, arguments[1]);
        } else {
          var attrs = this._attributes(arguments[1]);
          if(attrs.length) {
            try { // prevent IE "feature": http://dev.rubyonrails.org/ticket/2707
              parentElement.innerHTML = "<" +elementName + " " +
                attrs + "></" + elementName + ">";
            } catch(e) {}
            element = parentElement.firstChild || null;
            // workaround firefox 1.0.X bug
            if(!element) {
              element = document.createElement(elementName);
              for(attr in arguments[1])
                element[attr == 'class' ? 'className' : attr] = arguments[1][attr];
            }
            if(element.tagName.toUpperCase() != elementName)
              element = parentElement.getElementsByTagName(elementName)[0];
          }
        }

    // text, or array of children
    if(arguments[2])
      this._children(element, arguments[2]);

     return $(element);
  },
  _text: function(text) {
     return document.createTextNode(text);
  },

  ATTR_MAP: {
    'className': 'class',
    'htmlFor': 'for'
  },

  _attributes: function(attributes) {
    var attrs = [];
    for(attribute in attributes)
      attrs.push((attribute in this.ATTR_MAP ? this.ATTR_MAP[attribute] : attribute) +
          '="' + attributes[attribute].toString().escapeHTML().gsub(/"/,'&quot;') + '"');
    return attrs.join(" ");
  },
  _children: function(element, children) {
    if(children.tagName) {
      element.appendChild(children);
      return;
    }
    if(typeof children=='object') { // array can hold nodes and text
      children.flatten().each( function(e) {
        if(typeof e=='object')
          element.appendChild(e);
        else
          if(Builder._isStringOrNumber(e))
            element.appendChild(Builder._text(e));
      });
    } else
      if(Builder._isStringOrNumber(children))
        element.appendChild(Builder._text(children));
  },
  _isStringOrNumber: function(param) {
    return(typeof param=='string' || typeof param=='number');
  },
  build: function(html) {
    var element = this.node('div');
    $(element).update(html.strip());
    return element.down();
  },
  dump: function(scope) {
    if(typeof scope != 'object' && typeof scope != 'function') scope = window; //global scope

    var tags = ("A ABBR ACRONYM ADDRESS APPLET AREA B BASE BASEFONT BDO BIG BLOCKQUOTE BODY " +
      "BR BUTTON CAPTION CENTER CITE CODE COL COLGROUP DD DEL DFN DIR DIV DL DT EM FIELDSET " +
      "FONT FORM FRAME FRAMESET H1 H2 H3 H4 H5 H6 HEAD HR HTML I IFRAME IMG INPUT INS ISINDEX "+
      "KBD LABEL LEGEND LI LINK MAP MENU META NOFRAMES NOSCRIPT OBJECT OL OPTGROUP OPTION P "+
      "PARAM PRE Q S SAMP SCRIPT SELECT SMALL SPAN STRIKE STRONG STYLE SUB SUP TABLE TBODY TD "+
      "TEXTAREA TFOOT TH THEAD TITLE TR TT U UL VAR").split(/\s+/);

    tags.each( function(tag){
      scope[tag] = function() {
        return Builder.node.apply(Builder, [tag].concat($A(arguments)));
      };
    });
  }
};/*  
	Animator.js 1.1.9
	
	This library is released under the BSD license:

	Copyright (c) 2006, Bernard Sumption. All rights reserved.
	
	Redistribution and use in source and binary forms, with or without
	modification, are permitted provided that the following conditions are met:
	
	Redistributions of source code must retain the above copyright notice, this
	list of conditions and the following disclaimer. Redistributions in binary
	form must reproduce the above copyright notice, this list of conditions and
	the following disclaimer in the documentation and/or other materials
	provided with the distribution. Neither the name BernieCode nor
	the names of its contributors may be used to endorse or promote products
	derived from this software without specific prior written permission. 
	
	THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
	AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
	IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
	ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE FOR
	ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
	DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
	SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
	CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
	LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
	OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH
	DAMAGE.

*/


// Applies a sequence of numbers between 0 and 1 to a number of subjects
// construct - see setOptions for parameters
function Animator(options) {
	this.setOptions(options);
	var _this = this;
    this.timerDelegate = function(){_this.onTimerEvent();};
	this.subjects = [];
	this.target = 0;
	this.state = 0;
	this.lastTime = null;
}
Animator.prototype = {
	// apply defaults
	setOptions: function(options) {
		this.options = Animator.applyDefaults({
			interval: 20,  // time between animation frames
			duration: 400, // length of animation
			onComplete: function(){},
			onStep: function(){},
			transition: Animator.tx.easeInOut
		}, options);
	},
	// animate from the current state to provided value
	seekTo: function(to) {
		this.seekFromTo(this.state, to);
	},
	// animate from the current state to provided value
	seekFromTo: function(from, to) {
		this.target = Math.max(0, Math.min(1, to));
		this.state = Math.max(0, Math.min(1, from));
		this.lastTime = new Date().getTime();
		if (!this.intervalId) {
			this.intervalId = window.setInterval(this.timerDelegate, this.options.interval);
		}
	},
	// animate from the current state to provided value
	jumpTo: function(to) {
		this.target = this.state = Math.max(0, Math.min(1, to));
		this.propagate();
	},
	// seek to the opposite of the current target
	toggle: function() {
		this.seekTo(1 - this.target);
	},
	// add a function or an object with a method setState(state) that will be called with a number
	// between 0 and 1 on each frame of the animation
	addSubject: function(subject) {
		this.subjects[this.subjects.length] = subject;
		return this;
	},
	// remove all subjects
	clearSubjects: function() {
		this.subjects = [];
	},
	// forward the current state to the animation subjects
	propagate: function() {
		var value = this.options.transition(this.state);
		for (var i=0; i<this.subjects.length; i++) {
			if (this.subjects[i].setState) {
				this.subjects[i].setState(value);
			} else {
				this.subjects[i](value);
			}
		}
	},
	// called once per frame to update the current state
	onTimerEvent: function() {
		var now = new Date().getTime();
		var timePassed = now - this.lastTime;
		this.lastTime = now;
		var movement = (timePassed / this.options.duration) * (this.state < this.target ? 1 : -1);
		if (Math.abs(movement) >= Math.abs(this.state - this.target)) {
			this.state = this.target;
		} else {
			this.state += movement;
		}
		
		try {
			this.propagate();
		} finally {
			this.options.onStep.call(this);
			if (this.target == this.state) {
				window.clearInterval(this.intervalId);
				this.intervalId = null;
				this.options.onComplete.call(this);
			}
		}
	},
	// shortcuts
    play: function() {this.seekFromTo(0, 1);},
    reverse: function() {this.seekFromTo(1, 0);},
	// return a string describing this Animator, for debugging
	inspect: function() {
		var str = "#<Animator:\n";
		for (var i=0; i<this.subjects.length; i++) {
			str += this.subjects[i].inspect();
		}
		str += ">";
		return str;
	}
};
// merge the properties of two objects
Animator.applyDefaults = function(defaults, prefs) {
	prefs = prefs || {};
	var prop, result = {};
	for (prop in defaults) result[prop] = prefs[prop] !== undefined ? prefs[prop] : defaults[prop];
	return result;
};
// make an array from any object
Animator.makeArray = function(o) {
	if (o == null) return [];
	if (!o.length) return [o];
	var result = [];
	for (var i=0; i<o.length; i++) result[i] = o[i];
	return result;
};
// convert a dash-delimited-property to a camelCaseProperty (c/o Prototype, thanks Sam!)
Animator.camelize = function(string) {
	var oStringList = string.split('-');
	if (oStringList.length == 1) return oStringList[0];
	
	var camelizedString = string.indexOf('-') == 0
		? oStringList[0].charAt(0).toUpperCase() + oStringList[0].substring(1)
		: oStringList[0];
	
	for (var i = 1, len = oStringList.length; i < len; i++) {
		var s = oStringList[i];
		camelizedString += s.charAt(0).toUpperCase() + s.substring(1);
	}
	return camelizedString;
};
// syntactic sugar for creating CSSStyleSubjects
Animator.apply = function(el, style, options) {
	if (style instanceof Array) {
		return new Animator(options).addSubject(new CSSStyleSubject(el, style[0], style[1]));
	}
	return new Animator(options).addSubject(new CSSStyleSubject(el, style));
};
// make a transition function that gradually accelerates. pass a=1 for smooth
// gravitational acceleration, higher values for an exaggerated effect
Animator.makeEaseIn = function(a) {
	return function(state) {
		return Math.pow(state, a*2); 
	};
};
// as makeEaseIn but for deceleration
Animator.makeEaseOut = function(a) {
	return function(state) {
		return 1 - Math.pow(1 - state, a*2); 
	};
};
// make a transition function that, like an object with momentum being attracted to a point,
// goes past the target then returns
Animator.makeElastic = function(bounces) {
	return function(state) {
		state = Animator.tx.easeInOut(state);
		return ((1-Math.cos(state * Math.PI * bounces)) * (1 - state)) + state; 
	};
};
// make an Attack Decay Sustain Release envelope that starts and finishes on the same level
// 
Animator.makeADSR = function(attackEnd, decayEnd, sustainEnd, sustainLevel) {
	if (sustainLevel == null) sustainLevel = 0.5;
	return function(state) {
		if (state < attackEnd) {
			return state / attackEnd;
		}
		if (state < decayEnd) {
			return 1 - ((state - attackEnd) / (decayEnd - attackEnd) * (1 - sustainLevel));
		}
		if (state < sustainEnd) {
			return sustainLevel;
		}
		return sustainLevel * (1 - ((state - sustainEnd) / (1 - sustainEnd)));
	};
};
// make a transition function that, like a ball falling to floor, reaches the target and/
// bounces back again
Animator.makeBounce = function(bounces) {
	var fn = Animator.makeElastic(bounces);
	return function(state) {
		state = fn(state); 
		return state <= 1 ? state : 2-state;
	};
};
 
// pre-made transition functions to use with the 'transition' option
Animator.tx = {
	easeInOut: function(pos){
		return ((-Math.cos(pos*Math.PI)/2) + 0.5);
	},
	linear: function(x) {
		return x;
	},
	easeIn: Animator.makeEaseIn(1.5),
	easeOut: Animator.makeEaseOut(1.5),
	strongEaseIn: Animator.makeEaseIn(2.5),
	strongEaseOut: Animator.makeEaseOut(2.5),
	elastic: Animator.makeElastic(1),
	veryElastic: Animator.makeElastic(3),
	bouncy: Animator.makeBounce(1),
	veryBouncy: Animator.makeBounce(3)
};

// animates a pixel-based style property between two integer values
function NumericalStyleSubject(els, property, from, to, units) {
	this.els = Animator.makeArray(els);
	if (property == 'opacity' && window.ActiveXObject) {
		this.property = 'filter';
	} else {
		this.property = Animator.camelize(property);
	}
	this.from = parseFloat(from);
	this.to = parseFloat(to);
	this.units = units != null ? units : 'px';
}
NumericalStyleSubject.prototype = {
	setState: function(state) {
		var style = this.getStyle(state);
		var visibility = (this.property == 'opacity' && state == 0) ? 'hidden' : '';
		var j=0;
		for (var i=0; i<this.els.length; i++) {
			try {
				this.els[i].style[this.property] = style;
			} catch (e) {
				// ignore fontWeight - intermediate numerical values cause exeptions in firefox
				if (this.property != 'fontWeight') throw e;
			}
			if (j++ > 20) return;
		}
	},
	getStyle: function(state) {
		state = this.from + ((this.to - this.from) * state);
		if (this.property == 'filter') return "alpha(opacity=" + Math.round(state*100) + ")";
		if (this.property == 'opacity') return state;
		return Math.round(state) + this.units;
	},
	inspect: function() {
		return "\t" + this.property + "(" + this.from + this.units + " to " + this.to + this.units + ")\n";
	}
};

// animates a colour based style property between two hex values
function ColorStyleSubject(els, property, from, to) {
	this.els = Animator.makeArray(els);
	this.property = Animator.camelize(property);
	this.to = this.expandColor(to);
	this.from = this.expandColor(from);
	this.origFrom = from;
	this.origTo = to;
}

ColorStyleSubject.prototype = {
	// parse "#FFFF00" to [256, 256, 0]
	expandColor: function(color) {
		var hexColor, red, green, blue;
		hexColor = ColorStyleSubject.parseColor(color);
		if (hexColor) {
		    red = parseInt(hexColor.slice(1, 3), 16);
		    green = parseInt(hexColor.slice(3, 5), 16);
		    blue = parseInt(hexColor.slice(5, 7), 16);
		    return [red,green,blue];
		}
		if (window.DEBUG) {
			alert("Invalid colour: '" + color + "'");
		}
	},
	getValueForState: function(color, state) {
		return Math.round(this.from[color] + ((this.to[color] - this.from[color]) * state));
	},
	setState: function(state) {
		var color = '#'	+ ColorStyleSubject.toColorPart(this.getValueForState(0, state))
				+ ColorStyleSubject.toColorPart(this.getValueForState(1, state))
				+ ColorStyleSubject.toColorPart(this.getValueForState(2, state));
		for (var i=0; i<this.els.length; i++) {
			this.els[i].style[this.property] = color;
		}
	},
	inspect: function() {
		return "\t" + this.property + "(" + this.origFrom + " to " + this.origTo + ")\n";
	}
};

// return a properly formatted 6-digit hex colour spec, or false
ColorStyleSubject.parseColor = function(string) {
    var color = '#', match, i;
	if(match = ColorStyleSubject.parseColor.rgbRe.exec(string)) {
		var part;
		for (i=1; i<=3; i++) {
		    part = Math.max(0, Math.min(255, parseInt(match[i],10)));
			color += ColorStyleSubject.toColorPart(part);
		}
		return color;
	}
	if (match = ColorStyleSubject.parseColor.hexRe.exec(string)) {
		if(match[1].length == 3) {
			for (i=0; i<3; i++) {
				color += match[1].charAt(i) + match[1].charAt(i);
			}
			return color;
		}
		return '#' + match[1];
	}
	return false;
};
// convert a number to a 2 digit hex string
ColorStyleSubject.toColorPart = function(number) {
	if (number > 255) number = 255;
	var digits = number.toString(16);
	if (number < 16) return '0' + digits;
	return digits;
};
ColorStyleSubject.parseColor.rgbRe = /^rgb\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)\s*\)$/i;
ColorStyleSubject.parseColor.hexRe = /^\#([0-9a-fA-F]{3}|[0-9a-fA-F]{6})$/;

// Animates discrete styles, i.e. ones that do not scale but have discrete values
// that can't be interpolated
function DiscreteStyleSubject(els, property, from, to, threshold) {
	this.els = Animator.makeArray(els);
	this.property = Animator.camelize(property);
	this.from = from;
	this.to = to;
	this.threshold = threshold || 0.5;
}

DiscreteStyleSubject.prototype = {
	setState: function(state) {
		var j=0;
		for (var i=0; i<this.els.length; i++) {
			this.els[i].style[this.property] = state <= this.threshold ? this.from : this.to; 
		}
	},
	inspect: function() {
		return "\t" + this.property + "(" + this.from + " to " + this.to + " @ " + this.threshold + ")\n";
	}
};

// animates between two styles defined using CSS.
// if style1 and style2 are present, animate between them, if only style1
// is present, animate between the element's current style and style1
function CSSStyleSubject(els, style1, style2) {
	els = Animator.makeArray(els);
	this.subjects = [];
	if (els.length == 0) return;
	var prop, toStyle, fromStyle;
	if (style2) {
		fromStyle = this.parseStyle(style1, els[0]);
		toStyle = this.parseStyle(style2, els[0]);
	} else {
		toStyle = this.parseStyle(style1, els[0]);
		fromStyle = {};
		for (prop in toStyle) {
			fromStyle[prop] = CSSStyleSubject.getStyle(els[0], prop);
		}
	}
	// remove unchanging properties
	var prop;
	for (prop in fromStyle) {
		if (fromStyle[prop] == toStyle[prop]) {
			delete fromStyle[prop];
			delete toStyle[prop];
		}
	}
	// discover the type (numerical or colour) of each style
	var prop, units, match, type, from, to;
	for (prop in fromStyle) {
		var fromProp = String(fromStyle[prop]);
		var toProp = String(toStyle[prop]);
		if (toStyle[prop] == null) {
			if (window.DEBUG) alert("No to style provided for '" + prop + '"');
			continue;
		}
		
		if (from = ColorStyleSubject.parseColor(fromProp)) {
			to = ColorStyleSubject.parseColor(toProp);
			type = ColorStyleSubject;
		} else if (fromProp.match(CSSStyleSubject.numericalRe)
				&& toProp.match(CSSStyleSubject.numericalRe)) {
			from = parseFloat(fromProp);
			to = parseFloat(toProp);
			type = NumericalStyleSubject;
			match = CSSStyleSubject.numericalRe.exec(fromProp);
			var reResult = CSSStyleSubject.numericalRe.exec(toProp);
			if (match[1] != null) {
				units = match[1];
			} else if (reResult[1] != null) {
				units = reResult[1];
			} else {
				units = reResult;
			}
		} else if (fromProp.match(CSSStyleSubject.discreteRe)
				&& toProp.match(CSSStyleSubject.discreteRe)) {
			from = fromProp;
			to = toProp;
			type = DiscreteStyleSubject;
			units = 0;   // hack - how to get an animator option down to here
		} else {
			if (window.DEBUG) {
				alert("Unrecognised format for value of "
					+ prop + ": '" + fromStyle[prop] + "'");
			}
			continue;
		}
		this.subjects[this.subjects.length] = new type(els, prop, from, to, units);
	}
}

CSSStyleSubject.prototype = {
	// parses "width: 400px; color: #FFBB2E" to {width: "400px", color: "#FFBB2E"}
	parseStyle: function(style, el) {
		var rtn = {};
            var i;
		// if style is a rule set
		if (style.indexOf(":") != -1) {
			var styles = style.split(";");
			for (i=0; i<styles.length; i++) {
				var parts = CSSStyleSubject.ruleRe.exec(styles[i]);
				if (parts) {
					rtn[parts[1]] = parts[2];
				}
			}
		}
		// else assume style is a class name
		else {
			var prop, value, oldClass;
			oldClass = el.className;
			el.className = style;
			for (i=0; i<CSSStyleSubject.cssProperties.length; i++) {
				prop = CSSStyleSubject.cssProperties[i];
				value = CSSStyleSubject.getStyle(el, prop);
				if (value != null) {
					rtn[prop] = value;
				}
			}
			el.className = oldClass;
		}
		return rtn;
		
	},
	setState: function(state) {
		for (var i=0; i<this.subjects.length; i++) {
			this.subjects[i].setState(state);
		}
	},
	inspect: function() {
		var str = "";
		for (var i=0; i<this.subjects.length; i++) {
			str += this.subjects[i].inspect();
		}
		return str;
	}
};
// get the current value of a css property, 
CSSStyleSubject.getStyle = function(el, property){
	var style;
	if(document.defaultView && document.defaultView.getComputedStyle){
		style = document.defaultView.getComputedStyle(el, "").getPropertyValue(property);
		if (style) {
			return style;
		}
	}
	property = Animator.camelize(property);
	if(el.currentStyle){
		style = el.currentStyle[property];
	}
    return style || el.style[property];
};


CSSStyleSubject.ruleRe = /^\s*([a-zA-Z\-]+)\s*:\s*(\S(.+\S)?)\s*$/;
CSSStyleSubject.numericalRe = /^-?\d+(?:\.\d+)?(%|[a-zA-Z]{2})?$/;
CSSStyleSubject.discreteRe = /^\w+$/;

// required because the style object of elements isn't enumerable in Safari
/*
CSSStyleSubject.cssProperties = ['background-color','border','border-color','border-spacing',
'border-style','border-top','border-right','border-bottom','border-left','border-top-color',
'border-right-color','border-bottom-color','border-left-color','border-top-width','border-right-width',
'border-bottom-width','border-left-width','border-width','bottom','color','font-size','font-size-adjust',
'font-stretch','font-style','height','left','letter-spacing','line-height','margin','margin-top',
'margin-right','margin-bottom','margin-left','marker-offset','max-height','max-width','min-height',
'min-width','orphans','outline','outline-color','outline-style','outline-width','overflow','padding',
'padding-top','padding-right','padding-bottom','padding-left','quotes','right','size','text-indent',
'top','width','word-spacing','z-index','opacity','outline-offset'];*/


CSSStyleSubject.cssProperties = ['azimuth','background','background-attachment','background-color','background-image','background-position','background-repeat','border-collapse','border-color','border-spacing','border-style','border-top','border-top-color','border-right-color','border-bottom-color','border-left-color','border-top-style','border-right-style','border-bottom-style','border-left-style','border-top-width','border-right-width','border-bottom-width','border-left-width','border-width','bottom','clear','clip','color','content','cursor','direction','display','elevation','empty-cells','css-float','font','font-family','font-size','font-size-adjust','font-stretch','font-style','font-variant','font-weight','height','left','letter-spacing','line-height','list-style','list-style-image','list-style-position','list-style-type','margin','margin-top','margin-right','margin-bottom','margin-left','max-height','max-width','min-height','min-width','orphans','outline','outline-color','outline-style','outline-width','overflow','padding','padding-top','padding-right','padding-bottom','padding-left','pause','position','right','size','table-layout','text-align','text-decoration','text-indent','text-shadow','text-transform','top','vertical-align','visibility','white-space','width','word-spacing','z-index','opacity','outline-offset','overflow-x','overflow-y'];


// chains several Animator objects together
function AnimatorChain(animators, options) {
	this.animators = animators;
	this.setOptions(options);
	for (var i=0; i<this.animators.length; i++) {
		this.listenTo(this.animators[i]);
	}
	this.forwards = false;
	this.current = 0;
}

AnimatorChain.prototype = {
	// apply defaults
	setOptions: function(options) {
		this.options = Animator.applyDefaults({
			// by default, each call to AnimatorChain.play() calls jumpTo(0) of each animator
			// before playing, which can cause flickering if you have multiple animators all
			// targeting the same element. Set this to false to avoid this.
			resetOnPlay: true
		}, options);
	},
	// play each animator in turn
	play: function() {
		this.forwards = true;
		this.current = -1;
		if (this.options.resetOnPlay) {
			for (var i=0; i<this.animators.length; i++) {
				this.animators[i].jumpTo(0);
			}
		}
		this.advance();
	},
	// play all animators backwards
	reverse: function() {
		this.forwards = false;
		this.current = this.animators.length;
		if (this.options.resetOnPlay) {
			for (var i=0; i<this.animators.length; i++) {
				this.animators[i].jumpTo(1);
			}
		}
		this.advance();
	},
	// if we have just play()'d, then call reverse(), and vice versa
	toggle: function() {
		if (this.forwards) {
			this.seekTo(0);
		} else {
			this.seekTo(1);
		}
	},
	// internal: install an event listener on an animator's onComplete option
	// to trigger the next animator
	listenTo: function(animator) {
		var oldOnComplete = animator.options.onComplete;
		var _this = this;
		animator.options.onComplete = function() {
			if (oldOnComplete) oldOnComplete.call(animator);
			_this.advance();
		};
	},
	// play the next animator
	advance: function() {
		if (this.forwards) {
			if (this.animators[this.current + 1] == null) return;
			this.current++;
			this.animators[this.current].play();
		} else {
			if (this.animators[this.current - 1] == null) return;
			this.current--;
			this.animators[this.current].reverse();
		}
	},
	// this function is provided for drop-in compatibility with Animator objects,
	// but only accepts 0 and 1 as target values
	seekTo: function(target) {
		if (target <= 0) {
			this.forwards = false;
			this.animators[this.current].seekTo(0);
		} else {
			this.forwards = true;
			this.animators[this.current].seekTo(1);
		}
	}
};

// an Accordion is a class that creates and controls a number of Animators. An array of elements is passed in,
// and for each element an Animator and a activator button is created. When an Animator's activator button is
// clicked, the Animator and all before it seek to 0, and all Animators after it seek to 1. This can be used to
// create the classic Accordion effect, hence the name.
// see setOptions for arguments
function Accordion(options) {
	this.setOptions(options);
	var selected = this.options.initialSection, current;
	if (this.options.rememberance) {
		current = document.location.hash.substring(1);
	}
	this.rememberanceTexts = [];
	this.ans = [];
	var _this = this;
	for (var i=0; i<this.options.sections.length; i++) {
		var el = this.options.sections[i];
		var an = new Animator(this.options.animatorOptions);
		var from = this.options.from + (this.options.shift * i);
		var to = this.options.to + (this.options.shift * i);
		an.addSubject(new NumericalStyleSubject(el, this.options.property, from, to, this.options.units));
		an.jumpTo(0);
		var activator = this.options.getActivator(el);
		activator.index = i;
	    activator.onclick = function(){_this.show(this.index);};
		this.ans[this.ans.length] = an;
		this.rememberanceTexts[i] = activator.innerHTML.replace(/\s/g, "");
		if (this.rememberanceTexts[i] === current) {
			selected = i;
		}
	}
	this.show(selected);
}

Accordion.prototype = {
	// apply defaults
	setOptions: function(options) {
		this.options = Object.extend({
			// REQUIRED: an array of elements to use as the accordion sections
			sections: null,
			// a function that locates an activator button element given a section element.
			// by default it takes a button id from the section's "activator" attibute
		    getActivator: function(el) {return document.getElementById(el.getAttribute("activator"));},
			// shifts each animator's range, for example with options {from:0,to:100,shift:20}
			// the animators' ranges will be 0-100, 20-120, 40-140 etc.
			shift: 0,
			// the first page to show
			initialSection: 0,
			// if set to true, document.location.hash will be used to preserve the open section across page reloads 
			rememberance: true,
			// constructor arguments to the Animator objects
			animatorOptions: {}
		}, options || {});
	},
	show: function(section) {
		for (var i=0; i<this.ans.length; i++) {
			this.ans[i].seekTo(i > section ? 1 : 0);
		}
		if (this.options.rememberance) {
			document.location.hash = this.rememberanceTexts[section];
		}
	}
};
Popup = Class.create({    
    initialize: function(items) {
        this.items = items;
        this.options = {
            id: 'popup',
            popupclass: 'popup',
            popupshadowid: 'popupshadow',
            popupshadowclass: 'popupshadow',
            popupimage: '/music/local/images/popbkb.gif',
            onbeginexpand: null,
            onendexpand: null,
            onbeginshrink: null,
            openduration:0.2,
            closeduration:0.3,
            onendshrink: null
        };

        Object.extend( this.options, arguments[1] || {});

        this.createPopup();
        this.setDefaults();
        this.setItemAttributes();
    },

    createPopup: function() {
        this.closeBtn = Builder.node('a', {href:'#close', 'class':'close'}, 'Close');
        Event.observe(this.closeBtn, 'click', this.close.bindAsEventListener(this), false);

        this.popup = Builder.node('div', {'id':this.options.id, 'class':this.options.popupclass}, [ this.closeBtn ]);
        this.popupshadow = Builder.node('div', {id:this.options.popupshadowid, 'class':this.options.popupshadowclass}, [
            Builder.node('img', {src: this.options.popupimage, alt:'', border:0})
        ]);

        document.body.appendChild(this.popupshadow);
        document.body.appendChild(this.popup);
    },

    setDefaults: function() {
        this.defaultWidth = this.popup.offsetWidth;
        this.padleft = parseInt(Element.getStyle(this.popup, 'marginLeft').replace(/px/i,''));
        this.padright = parseInt(Element.getStyle(this.popup, 'marginRight').replace(/px/i,''));

        this.defaultHeight = this.popup.offsetHeight;
        this.padtop = parseInt(Element.getStyle(this.popup, 'marginTop').replace(/px/,''));
        this.padbottom = parseInt(Element.getStyle(this.popup, 'marginBottom').replace(/px/,''));
    },

    setItemAttributes: function() {
        for (var i=0; i<this.items.length; i++) {
            var item = this.items[i];
            this.setEvent(item, i);
        }
    },

    setEvent: function(item, i) {
        Event.observe(item, 'click', this.onClick.bindAsEventListener(this, item, i), false);
    },

    popupAt: function(evt) {
        // store the small size and position for later
        this.left = evt.x;
        this.left -= this.width/2;
        // this.left = this.left || document.body.getDimensions().width / 2;
        this.left = this.left || document.body.offsetWidth / 2;
        this.width = 80;
        this.height = 120;
        this.top = evt.y;
        this.top -= this.height/2;
        
        if (Adito.Detector.isiPhone()) {
            this.left = 3;
            this.top = 200;
        }
        
        // do the image
        this.prepPop(evt, null, 0);
    },

    onClick: function(evt, item, i) {
        // store the small size and position for later
        this.left = evt.pageX || evt.clientX + (document.documentElement.scrollLeft ? document.documentElement.scrollLeft : document.body.scrollLeft);
        this.left -= this.width/2;
        // this.left = this.left || document.body.getDimensions().width / 2;
        this.left = this.left || document.body.offsetWidth / 2;
        this.width = (item.offsetWidth>80) ? 80 : item.offsetWidth;
        this.height = item.offsetHeight;
        this.top = evt.pageY || evt.clientY + (document.documentElement.scrollTop ? document.documentElement.scrollTop : document.body.scrollTop);
        this.top -= this.height/2;
        
        if (Adito.Detector.isiPhone()) {
            this.left = 3;
            this.top = 200;
        }
        
        // stop the default event
        Event.stop(evt);

        // do the image
        this.prepPop(evt, item, i);
    },

    windowSize: function() {
        var width = window.innerWidth || (window.document.documentElement.clientWidth || window.document.body.clientWidth);
        var height = window.innerHeight || (window.document.documentElement.clientHeight || window.document.body.clientHeight);
        var x = window.pageXOffset || (window.document.documentElement.scrollLeft || window.document.body.scrollLeft);
        var y = window.pageYOffset || (window.document.documentElement.scrollTop || window.document.body.scrollTop);
        
        if(Adito.Detector.isiPhone()) {
            width = 980;
            height = 1212;
        }

        return {'width':width, 'height':height, 'x':x, 'y':y}
    },

    setPopPosition: function() {
        // set the position/offset of the image
        var left, top = null;

        var ws = this.windowSize();
        left = ws.x+(ws.width-this.defaultWidth-this.padleft-this.padright)/2;
        if (ws.width<this.defaultWidth+this.padleft+this.padright) left = ws.x-(this.padtop-this.closeBtn.offsetWidth);

        top = ws.y+(ws.height-this.defaultHeight-this.padtop-this.padbottom)/2;
        if (ws.height<this.defaultHeight+this.padtop+this.padbottom) top = ws.y-(this.padtop-this.closeBtn.offsetHeight);
        
        return { left:left, top:top };
    },

    prepPop: function(evt, item, i) {
        // call the effect
        var popp = this.setPopPosition();
        this.pop(this.defaultWidth, popp.top, this.defaultHeight, popp.left, item, i);
    },

    beforePop: function() {
        if ( this.options.onbeginexpand )
        {
            this.options.onbeginexpand(this.popup);
        }
        Element.addClassName(this.popup, 'isanim');
        Element.addClassName(this.popupshadow, 'isanim');
    },

    pop: function(width, top, height, left, item, i) {
        // prep the popup/shadow for the effect
        this.popup.style.width = this.width+'px';
        this.popupshadow.style.width = this.width+'px';

        this.popup.style.height = this.height+'px';
        this.popupshadow.style.height = this.height+'px';

        this.popup.style.left = this.left+'px';
        this.popupshadow.style.left = this.left+'px';

        this.popup.style.top = this.top+'px';
        this.popupshadow.style.top = this.top+'px';

        Element.setOpacity(this.popup, 0);
        Element.setOpacity(this.popupshadow, 0);

        if (!Adito.Detector.isiPhone()) {
            // do the craziness
            new Effect.Parallel([
                new Effect.MoveBy(this.popup, top-this.top, left-this.left, { sync:true }), 
                new Effect.MoveBy(this.popupshadow, top-this.top, left-this.left, { sync:true }), 
                new Effect.Scale(this.popup, (width/this.width)*100, { sync:true, scaleY:false, scaleContent:false }),
                new Effect.Scale(this.popupshadow, ((width+this.padleft+this.padleft)/this.width)*100, { sync:true, scaleY:false, scaleContent:false }),
                new Effect.Scale(this.popup, (height/this.height)*100, { sync:true, scaleX:false, scaleContent:false }),
                new Effect.Scale(this.popupshadow, ((height+this.padtop+this.padbottom)/this.height)*100, { sync:true, scaleX:false, scaleContent:false }),
                new Effect.Appear(this.popupshadow, { sync:true }),
                new Effect.Appear(this.popup, { sync:true })
            ],
                                { duration:this.options.openduration, beforeStart:this.beforePop.bind(this), afterFinish:this.afterPop.bind(this, item, i) }
                               );
        } else {
            this.beforePop();
            this.afterPop();
            this.popup.scrollTo();
        }
    },

    afterPop: function(item, i) {
        Element.removeClassName(this.popup, 'isanim');
        Element.removeClassName(this.popupshadow, 'isanim');
        Element.addClassName(this.popup, 'popped');
        Element.addClassName(this.popupshadow, 'popped');

        this.popup.style.width = '';
        this.popupshadow.style.width = '';

        this.popup.style.height = '';
        this.popupshadow.style.height = '';

        Element.setOpacity(this.popup, '');
        Element.setOpacity(this.popupshadow, '');
        if ( this.options.onendexpand )
        {
            this.options.onendexpand(this.popup);
        }
    },

    beforeClose: function() {
        if ( this.options.onbeginshrink )
        {
            this.options.onbeginshrink(this.popup);
        }
        Element.addClassName(this.popup, 'isanim');
        Element.addClassName(this.popupshadow, 'isanim');
        Element.removeClassName(this.popup, 'popped');
        Element.removeClassName(this.popupshadow, 'popped');
    },

    close: function(evt) {
        if (evt) Event.stop(evt);

        var width = this.defaultWidth;
        var left = this.popup.offsetLeft;
        var height = this.defaultHeight;
        var top = this.popup.offsetTop;

        if (!Adito.Detector.isiPhone()) {
            // do the craziness
            new Effect.Parallel([
                new Effect.MoveBy(this.popup, this.top-top, this.left-left, { sync:true }), 
                new Effect.MoveBy(this.popupshadow, this.top-top, this.left-left, { sync:true }), 
                new Effect.Scale(this.popup, (this.width/width)*100, { sync:true, scaleY:false, scaleContent:false }),
                new Effect.Scale(this.popupshadow, (this.width/(width+this.padleft+this.padleft))*100, { sync:true, scaleY:false, scaleContent:false }),
                new Effect.Scale(this.popup, (this.height/height)*100, { sync:true,scaleX:false, scaleContent:false }),
                new Effect.Scale(this.popupshadow, (this.height/(height+this.padtop+this.padbottom))*100, { sync:true, scaleX:false, scaleContent:false }),
                new Effect.Fade(this.popup, { sync:true }),
                new Effect.Fade(this.popupshadow, { sync:true })
            ],
                                { duration:this.options.closeduration, beforeStart:this.beforeClose.bind(this), afterFinish:this.afterClose.bind(this) });
        } else {
            this.beforeClose();
            this.afterClose();
            this.popup.scrollTo();
        }
    },

    afterClose: function() {
        Element.removeClassName(this.popup, 'isanim');
        Element.removeClassName(this.popupshadow, 'isanim');

        // reset everything
        this.popup.style.width = '';
        this.popupshadow.style.width = '';

        this.popup.style.height = '';
        this.popupshadow.style.height = '';

        this.popup.style.left = '';
        this.popupshadow.style.left = '';

        this.popup.style.top = '';
        this.popupshadow.style.top = '';

        this.popup.style.display = '';
        this.popupshadow.style.display = '';

        if (Adito.Detector.isWebKit()) this.fixSafarisScrollBars();
        if ( this.options.onendshrink )
        {
            this.options.onendshrink(this.popup);
        }
    },

    fixSafarisScrollBars: function() {
        scrollTo = 1;
        window.scroll(this.windowSize().x+scrollTo, this.windowSize().y+scrollTo);
        scrollTo = -scrollTo;
        window.scroll(this.windowSize().x+scrollTo, this.windowSize().y+scrollTo);
    }

});
var Transbase = Class.create({

  initialize: function() {

    // Get all children
    // ~~~~~~~~~~~~~~~~
    this.options = {
      opacity : 0.5,
        zindex: 998,
        backcolour: "#666666"
      };

      Object.extend( this.options, arguments[0] || {});
    
    this.lastzindex = this.options.zindex;
    
    var el = $("transbase");
    if ( el == null )
    {
      el = document.createElement("div");
      el.id = "transbase";
      document.body.appendChild( el );
      el = $("transbase");
    }

      var ww = getwinwidth();
      var wh = getwinheight();
      var po = offsets();
      var bgX=po.x;
      var bgY=po.y;

      var props = {"position": "absolute", "backgroundColor": this.options.backcolour, "zIndex": this.options.zindex, "opacity": this.options.opacity, "display": "none", "left": bgX+"px", "top": bgY+"px", "width": ww+"px", "height": wh+"px" };
      el.setStyle(props);

    this.callback = this.winchange.bindAsEventListener(this);
  },

  // Append a section to the translucent base
  // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
  add: function( id, html, options )
  {
    var el = $(id);
    if ( el == null )
    {
      var opts = Object.extend({
        opacity : 1,
        zindex: ++this.lastzindex,
          backcolour: "#666666",
          left: 0,
          top: 0,
          width: '100px',
          height: '100px',
          position:'absolute',
          visibility: 'visible',
          display: 'block'
      }, options || {});

      el = document.createElement("div");
      el.id = id;
      document.body.appendChild(el);

      var props = {"visibility":opts.visibility, "position":opts.position, "backgroundColor": opts.backcolour, "zIndex":opts.zindex, "opacity":opts.opacity, "display":opts.display, "left":opts.left+"px","top":opts.top+"px","width":opts.width, "height":opts.height };      

      el.innerHTML = html;
      $(id).setStyle(props);
    }
  },
  remove: function( id )
  {
    var todel = $(id);
    if ( todel != null )
    {
      Element.remove( todel );
    }
  },
  
  show: function()
  {
    var el = $("transbase");
    
    if ( el.style.display == "none" )
    {
      el.style.display = "block";
      Event.observe(window, "resize", this.callback, false);
      Event.observe(window, "scroll", this.callback, false);
    }
  },
  hide: function()
  {
    var el = $("transbase");
    if ( el.style.display == "block" )
    {
      el.style.display = "none";
      Event.stopObserving(window, "resize", this.callback );
      Event.stopObserving(window, "scroll", this.callback );
    }
  },
  
  winchange: function()
  {
    var el = $("transbase");
    var po = offsets();

    el.style.left=po.x+"px";
    el.style.top=po.y+"px";
    el.style.width=getwinwidth()+"px";
    el.style.height=getwinheight()+"px";
  }
});

/*
Script: Scroller.js
	Contains the <Scroller>.

License:
	MIT-style license.
*/

/*
Class: Scroller
	The Scroller is a class to scroll any element with an overflow (including the window) when the mouse cursor reaches certain buondaries of that element.
	You must call its start method to start listening to mouse movements.

Arguments:
	element - required, the element to scroll.
	options - optional, see options below, and <Fx.Base> options.

Options:
	area - integer, the necessary boundaries to make the element scroll.
	velocity - integer, velocity ratio, the modifier for the window scrolling speed.
	onChange - optionally, when the mouse reaches some boundaries, you can choose to alter some other values, instead of the scrolling offsets.
		Automatically passes as parameters x and y values.
*/

var Scroller = Class.create({

  initialize: function(element){

    this.element = $(element);

    this.options = {
        autostart: false,
        area: 20,
        velocity: 1,
        horz: true,
        vert: true,
        _this: this.element,
        onChange: function(x, y){ this._this.scrollLeft = x;this._this.scrollTop = y; }
      };

    Object.extend( this.options, arguments[1] || {});

    this.timer = null;
    this.started = false;
    this.mousemover = ([window, document].indexOf(element)) >= 0? $(document.body) : this.element;
    if ( this.options.autostart == true )
    {
      this.start();
    }
  },

  /*      
	Property: start
        The scroller starts listening to mouse movements.
  */
  start: function()
  {
    if ( !this.started )
    {
      this.coord = this.getCoords.bindAsEventListener(this);
      Event.observe( this.mousemover, "mousemove", this.coord );
      this.started = true;
    }
  },

  /*
    Property: stop
    The scroller stops listening to mouse movements.
  */

  stop: function()
  {
    if ( this.started )
    {
      Event.stopObserving(this.mousemover, "mousemove", this.coord );
      if (this.timer)
      {
        clearTimeout(this.timer);
        this.timer = null;
      }
      
      this.started = false;
    }
  },

  convertEvent: function( event )
  {
    var converted = {};
    
    converted.page = {
      'x': event.pageX || event.clientX + document.documentElement.scrollLeft,
      'y': event.pageY || event.clientY + document.documentElement.scrollTop
    };
    converted.client = {
      'x': event.pageX ? event.pageX - window.pageXOffset : event.clientX,
      'y': event.pageY ? event.pageY - window.pageYOffset : event.clientY
      };
    
    return converted;
  },

  getCoords: function(event)
  {
    var converted = this.convertEvent(event);
    this.page = (this.element == window) ? converted.client : converted.page;
    if (!this.timer)
    {
      this.timer = setInterval( this.scroll.bind(this), 50 );
    }
  },
  
  getSize: function( element )
  {
    return {
      "scroll": {"x": element.scrollLeft, "y": element.scrollTop},
      "size": {"x": element.offsetWidth, "y": element.offsetHeight},
      "scrollSize": {"x": element.scrollWidth, "y": element.scrollHeight}
    };
  },

  getPosition: function(el, overflown){
    overflown = overflown || [];
    var left = 0, top = 0;
    do {
      left += el.offsetLeft || 0;
      top += el.offsetTop || 0;
      el = el.offsetParent;
    } while (el);
    overflown.each(function(element){
      left -= element.scrollLeft || 0;
      top -= element.scrollTop || 0;
    });
    return {"x": left, "y": top};
  },

  scroll: function()
  {
    var el = this.getSize(this.element);
    var pos = this.getPosition(this.element);
    
    // alert("el = "+el.size.x+ ", "+el.size.y+" pos = "+pos.x+", "+pos.y);
    var change = {"x": 0, "y": 0};
    for (var z in this.page)
    {
      if (this.page[z] < (this.options.area + pos[z]) && el.scroll[z] != 0)
        change[z] = (this.page[z] - this.options.area - pos[z]) * this.options.velocity;
      else if (this.page[z] + this.options.area > (el.size[z] + pos[z]) && el.scroll[z] + el.size[z] != el.scrollSize[z])
        change[z] = (this.page[z] - el.size[z] + this.options.area - pos[z]) * this.options.velocity;
    }
    if ( this.options.horz == false ) change.x = 0;
    if ( this.options.vert == false ) change.y = 0;
    if (change.y || change.x)
    {
      this.options.onChange( el.scroll.x + change.x, el.scroll.y + change.y );
    }
  }
});

ImageReplacer = Class.create({
    initialize: function(options) {

        this.options = {
            baseurl: "/music/local/custtext.php",
            doNotPrintImages: false,
            printerCSS:"/music/local/replacement-print.css",
            hideFlicker: false,
            hideFlickerCSS:"/music/local/replacement-screen.css",
            hideFlickerTimeout:1000
        };

        Object.extend( this.options, options || {});

        this.items = [];
        this.imageLoaded = false;
        // if( this.options.hideFlicker )
        // {
        //     document.write('<link id="hide-flicker" rel="stylesheet" media="screen" href="' + this.options.hideFlickerCSS + '" />');
        //    window.flickerCheck = function() {
        //         if( !imageLoaded )
        //             setStyleSheetState('hide-flicker',false);
        //     };
        // 
        //     setTimeout('window.flickerCheck();', this.options.hideFlickerTimeout );
        // }

        if( this.options.doNotPrintImages )
        {
            document.write('<link id="print-text" rel="stylesheet" media="print" href="' + this.options.printerCSS + '" />');
        }
    },
    replaceSelector: function( options )
    {
        var opts = {
            selector: "h1",
            fontname: "vrb_____.pfb",
            url: this.options.baseurl,
            wordwrap: false,
            fontsize: 24,
            colour:"#dedede",
            backcolour:"#ff0000",
            alias: 1,
            transparent: 1,
            override: 1
        };

        Object.extend( opts, options || {});

        this.items[this.items.length] = opts; // { selector: selector, fontname: fontname, url: url, wordwrap: wordwrap };
    },
    replacetext: function()
    {
        for(var i=0, p=this.items.length;i<p;i++)
        {
            var opts = this.items[i];
            var elements = $(document.body).select(opts.selector);

            if(elements.length > 0)
            {
                for(var j=0,ll = elements.length;j<ll;j++)
                {
                    if( !elements[j] )
                        continue;

                    var el = elements[j];
                    var text = this.extractText(el);
                    var elwidth = parseInt(el.style.width)|| el.innerWidth || el.offsetWidth;
                    var elheight = parseInt(el.style.height)|| el.innerHeight || el.offsetHeight;
                    var elcolour = el.style.color|| opts.colour;
                    var elbackcolour = el.style.backgroundColor|| opts.backcolour;
                    var elsz = parseInt(el.style.fontSize) || opts.fontsize;

                    // Do we override height
                    // ~~~~~~~~~~~~~~~~~~~~~
                    if ( opts.override != 0 && elsz > elheight )
                    {
                      elsz = elheight;
                    }
                    while(el.hasChildNodes())
                    {
                        el.removeChild( el.firstChild );
                    }
          
                    var tokens = opts.wordwrap ? text.split(' ') : [text];

                    for(var k = 0, h=tokens.length; k < h; k++)
                    {
                        // Need colour, width and height
                        // ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
                        var url = opts.url + "?transparent="+opts.transparent+"&alias="+opts.alias+"&backcolour="+encodeURIComponent(elbackcolour)+"&colour="+encodeURIComponent(elcolour)+"&fontname="+encodeURIComponent(opts.fontname)+"&sz="+elsz+"&w="+elwidth+"&h="+elheight+"&text="+encodeURIComponent(tokens[k])+"&selector="+escape(opts.selector);
                        var image = document.createElement("img");
                        Element.addClassName( image, "replacement" );
                        image.alt = tokens[k];
                        image.src = url;
                        // image.height = elsz;
                        el.appendChild(image);
                    }
          
                    Element.addClassName( el, "replaced" );

                    if( this.options.doNotPrintImages )
                    {
                        var span = document.createElement("span");
                        span.style.display = "none";
                        span.className = "print-text";
                        span.appendChild(document.createTextNode(text));
                        el.appendChild(span);
                    }
                }
            }
        }

        if ( this.options.hideFlicker )
        {
            setStyleSheetState('hide-flicker',false);
        }
    },

    setStyleSheetState: function( id, enabled ) 
    {
        var sheet = document.getElementById(id);
        if( sheet )
        {
            sheet.disabled = (!enabled);
        }
    },
    extractText: function( element )
    {
        if(typeof element == "string")
            return element;
        else if(typeof element == "undefined")
            return element;
        else if(element.innerText)
            return element.innerText;

        var text = "";
        var kids = element.childNodes;
        for(var i=0;i<kids.length;i++)
        {
            if(kids[i].nodeType == 1)
                text += extractText(kids[i]);
            else if(kids[i].nodeType == 3)
                text += kids[i].nodeValue;
        }
        
        return text;
    }
});
var CanvasBase=Class.create({start:function(){
if(Prototype.Browser.IE){
try{
if(document.namespaces["v"]==null){
var _1=document.createStyleSheet();
_1.addRule("v\\:*","behavior: url(#default#VML);");
document.namespaces.add("v","urn:schemas-microsoft-com:vml");
}else{
var _2=false;
var _3=document.styleSheets;
for(var i=0,j=_3.length;i<j;++i){
var _6=_3[i];
if(_6.cssText.length>=3){
var _7=_6.cssText.substring(0,3);
if(_7=="v:\\*"){
_2=true;
break;
}
}
}
if(!_2){
var _1=document.createStyleSheet();
_1.addRule("v\\:*","behavior: url(#default#VML);");
}
}
}
catch(e){
}
}
},getImages:function(_8){
var _9=document.getElementsByTagName("img");
var _a=new Array();
var i=0;
var _c;
var _d;
var j=0;
for(i=0;i<_9.length;i++){
_c=_9[i];
_d=_c.className.split(" ");
for(var j=0;j<_d.length;j++){
if(_d[j]==_8){
_a.push(_c);
break;
}
}
}
return _a;
},getClasses:function(_f,_10){
var _11="";
for(var j=0;j<_f.length;j++){
if(_f[j]!=_10){
if(_11){
_11+=" ";
}
_11+=_f[j];
}
}
return _11;
},getClassValue:function(_13,_14){
var _15=0;
var pos=_14.length;
for(var j=0;j<_13.length;j++){
if(_13[j].indexOf(_14)==0){
_15=Math.min(_13[j].substring(pos),100);
break;
}
}
return Math.max(0,_15);
},getClassColor:function(_18,_19){
var _1a=0;
var str="";
var pos=_19.length;
for(var j=0;j<_18.length;j++){
if(_18[j].indexOf(_19)==0){
_1a=_18[j].substring(pos);
str="#"+_1a.toLowerCase();
break;
}
}
if(str.match(/^#[0-9a-f][0-9a-f][0-9a-f][0-9a-f][0-9a-f][0-9a-f]$/i)){
return str;
}else{
return 0;
}
},getClassAttribute:function(_1e,_1f){
var _20=0;
var pos=_1f.length;
for(var j=0;j<_1e.length;j++){
if(_1e[j].indexOf(_1f)==0){
_20=1;
break;
}
}
return _20;
},clipPolyRight:function(ctx,x,y,w,h,t,d,s){
var z=(h-t-t)/h;
ctx.beginPath();
ctx.moveTo(x,y);
ctx.lineTo(w,y+t);
ctx.lineTo(w,y+h-t);
ctx.lineTo(x,y+h);
if(d>0){
ctx.lineTo(x,y+h-s);
ctx.lineTo(w,y+h-t-(z*s));
ctx.lineTo(w,y+h-t-(z*(s+d)));
ctx.lineTo(x,y+h-s-d);
}
ctx.closePath();
},clipPolyLeft:function(ctx,x,y,w,h,t,d,s){
var z=(h-t-t)/h;
ctx.beginPath();
ctx.moveTo(x,y+t);
ctx.lineTo(w,y);
ctx.lineTo(w,y+h);
ctx.lineTo(x,y+h-t);
if(d>0){
ctx.lineTo(x,y+h-t-(z*s));
ctx.lineTo(w,y+h-s);
ctx.lineTo(w,y+h-s-d);
ctx.lineTo(x,y+h-t-(z*(s+d)));
}
ctx.closePath();
},strokePolyRight:function(ctx,x,y,w,h,t,d,s,b){
var z=(h-t-t)/h;
var n=(b>=1?1:0);
ctx.beginPath();
ctx.moveTo(x+b,y+b);
ctx.lineTo(w-b,y+t+b-n);
ctx.lineTo(w-b,y+h-t-(z*(s+d))-b);
ctx.lineTo(x+b,y+h-s-d-b);
ctx.closePath();
},strokePolyLeft:function(ctx,x,y,w,h,t,d,s,b){
var z=(h-t-t)/h;
var n=(b>=1?1:0);
ctx.beginPath();
ctx.moveTo(x+b,y+t+b-n);
ctx.lineTo(w-b,y+b);
ctx.lineTo(w-b,y+h-s-d-b);
ctx.lineTo(x+b,y+h-t-(z*(s+d))-b);
ctx.closePath();
},clipReflex:function(ctx,x,y,w,h,t,d,s,o){
var z=(h-t-t)/h;
ctx.beginPath();
if(o=="r"){
ctx.moveTo(x,y+h-s);
ctx.lineTo(w,y+h-t-(z*s));
ctx.lineTo(w,y+h-t+2);
ctx.lineTo(x,y+h+2);
}else{
ctx.moveTo(w,y+h+2);
ctx.lineTo(w,y+h-s);
ctx.lineTo(x,y+h-t-(z*s));
ctx.lineTo(x,y+h-t+2);
}
ctx.closePath();
},clearReflex:function(ctx,x,y,w,h,t,d,s,o){
var z=(h-t-t)/h;
ctx.beginPath();
if(o=="r"){
ctx.moveTo(x,y+h-1);
ctx.lineTo(w,y+h-t-1);
ctx.lineTo(w,y+h-t+1);
ctx.lineTo(x,y+h+1);
}else{
ctx.moveTo(w,y+h-1);
ctx.lineTo(x,y+h-t-1);
ctx.lineTo(x,y+h-t+1);
ctx.lineTo(w,y+h+1);
}
ctx.closePath();
},roundedRect:function(ctx,x,y,_62,_63,_64,_65){
if(!_65){
ctx.beginPath();
}
ctx.moveTo(x,y+_64);
ctx.lineTo(x,y+_63-_64);
ctx.quadraticCurveTo(x,y+_63,x+_64,y+_63);
ctx.lineTo(x+_62-_64,y+_63);
ctx.quadraticCurveTo(x+_62,y+_63,x+_62,y+_63-_64);
ctx.lineTo(x+_62,y+_64);
ctx.quadraticCurveTo(x+_62,y,x+_62-_64,y);
ctx.lineTo(x+_64,y);
ctx.quadraticCurveTo(x,y,x,y+_64);
if(!_65){
ctx.closePath();
}
},addRadialStyle:function(ctx,x1,y1,r1,x2,y2,r2,_6d){
var tmp=ctx.createRadialGradient(x1,y1,r1,x2,y2,r2);
var opt=Math.min(parseFloat(_6d+0.1),1);
tmp.addColorStop(0,"rgba(0,0,0,"+opt+")");
tmp.addColorStop(0.25,"rgba(0,0,0,"+_6d+")");
tmp.addColorStop(1,"rgba(0,0,0,0)");
return tmp;
},addLinearStyle:function(ctx,x,y,w,h,_75){
var tmp=ctx.createLinearGradient(x,y,w,h);
var opt=Math.min(parseFloat(_75+0.1),1);
tmp.addColorStop(0,"rgba(0,0,0,"+opt+")");
tmp.addColorStop(0.25,"rgba(0,0,0,"+_75+")");
tmp.addColorStop(1,"rgba(0,0,0,0)");
return tmp;
},addBright:function(ctx,x,y,_7b,_7c,_7d,_7e){
var _7f=ctx.createLinearGradient(0,y,0,y+_7c);
_7f.addColorStop(0,"rgba(254,254,254,"+_7e+")");
_7f.addColorStop(1,"rgba(254,254,254,0.1)");
ctx.beginPath();
ctx.moveTo(x,y+_7d);
ctx.lineTo(x,y+_7c-_7d);
ctx.quadraticCurveTo(x,y+_7c,x+_7d,y+_7c);
ctx.lineTo(x+_7b-_7d,y+_7c);
ctx.quadraticCurveTo(x+_7b,y+_7c,x+_7b,y+_7c-_7d);
ctx.lineTo(x+_7b,y+_7d);
ctx.quadraticCurveTo(x+_7b,y,x+_7b-_7d,y);
ctx.lineTo(x+_7d,y);
ctx.quadraticCurveTo(x,y,x,y+_7d);
ctx.closePath();
ctx.fillStyle=_7f;
ctx.fill();
},addDark:function(ctx,x,y,_83,_84,_85,_86){
var _87=ctx.createLinearGradient(0,y,0,y+_84);
_87.addColorStop(0,"rgba(0,0,0,0)");
_87.addColorStop(1,"rgba(0,0,0,"+_86+")");
ctx.beginPath();
ctx.moveTo(x,y);
ctx.lineTo(x,y+_84-_85);
ctx.quadraticCurveTo(x,y+_84,x+_85,y+_84);
ctx.lineTo(x+_83-_85,y+_84);
ctx.quadraticCurveTo(x+_83,y+_84,x+_83,y+_84-_85);
ctx.lineTo(x+_83,y);
ctx.lineTo(x,y);
ctx.closePath();
ctx.fillStyle=_87;
ctx.fill();
},addFrame:function(ctx,x,y,_8b,_8c,_8d,_8e){
this.roundedRect(ctx,x,y,_8b,_8c,_8d);
var _8f=ctx.createLinearGradient(0,0,0,_8c);
_8f.addColorStop(0,"rgba(254,254,254,"+_8e+")");
_8f.addColorStop(1,"rgba(0,0,0,"+_8e+")");
ctx.lineWidth=(_8d+x)/2;
ctx.strokeStyle=_8f;
ctx.stroke();
},glossyShadow:function(ctx,x,y,_93,_94,_95,_96){
var _97;
var os=_95/2;
ctx.beginPath();
ctx.rect(x+_95,y,_93-(_95*2),y+os);
ctx.closePath();
_97=this.addLinearStyle(ctx,x+_95,y+os,x+_95,y,_96);
ctx.fillStyle=_97;
ctx.fill();
ctx.beginPath();
ctx.rect(x,y,_95,_95);
ctx.closePath();
_97=this.addRadialStyle(ctx,x+_95,y+_95,_95-os,x+_95,y+_95,_95,_96);
ctx.fillStyle=_97;
ctx.fill();
ctx.beginPath();
ctx.rect(x,y+_95,os,_94-(_95*2));
ctx.closePath();
_97=this.addLinearStyle(ctx,x+os,y+_95,x,y+_95,_96);
ctx.fillStyle=_97;
ctx.fill();
ctx.beginPath();
ctx.rect(x,y+_94-_95,_95,_95);
ctx.closePath();
_97=this.addRadialStyle(ctx,x+_95,y+_94-_95,_95-os,x+_95,y+_94-_95,_95,_96);
ctx.fillStyle=_97;
ctx.fill();
ctx.beginPath();
ctx.rect(x+_95,y+_94-os,_93-(_95*2),os);
ctx.closePath();
_97=this.addLinearStyle(ctx,x+_95,y+_94-os,x+_95,y+_94,_96);
ctx.fillStyle=_97;
ctx.fill();
ctx.beginPath();
ctx.rect(x+_93-_95,y+_94-_95,_95,_95);
ctx.closePath();
_97=this.addRadialStyle(ctx,x+_93-_95,y+_94-_95,_95-os,x+_93-_95,y+_94-_95,_95,_96);
ctx.fillStyle=_97;
ctx.fill();
ctx.beginPath();
ctx.rect(x+_93-os,y+_95,os,_94-(_95*2));
ctx.closePath();
_97=this.addLinearStyle(ctx,x+_93-os,y+_95,x+_93,y+_95,_96);
ctx.fillStyle=_97;
ctx.fill();
ctx.beginPath();
ctx.rect(x+_93-_95,y,_95,_95);
ctx.closePath();
_97=this.addRadialStyle(ctx,x+_93-_95,y+_95,_95-os,x+_93-_95,y+_95,_95,_96);
ctx.fillStyle=_97;
ctx.fill();
},addShading:function(ctx,x,y,_9c,_9d,_9e){
var _9f=ctx.createLinearGradient(0,y,0,y+_9d);
_9f.addColorStop(0,"rgba(0,0,0,"+(_9e/2)+")");
_9f.addColorStop(0.3,"rgba(0,0,0,0)");
_9f.addColorStop(0.7,"rgba(254,254,254,0)");
_9f.addColorStop(1,"rgba(254,254,254,"+(_9e)+")");
ctx.beginPath();
ctx.rect(x,y,_9c,_9d);
ctx.closePath();
ctx.fillStyle=_9f;
ctx.fill();
},addLining:function(ctx,x,y,_a3,_a4,_a5,_a6,_a7,_a8){
var _a9=ctx.createLinearGradient(x,y,_a3,_a4);
if(_a7==true){
_a9.addColorStop(0,"rgba(192,192,192,"+_a5+")");
_a9.addColorStop(0.7,"rgba(254,254,254,0.8)");
_a9.addColorStop(1,"rgba(254,254,254,0.9)");
}else{
if(_a8=="#f0f4ff"){
_a9.addColorStop(0,"rgba(254,254,254,0.9)");
_a9.addColorStop(0.3,"rgba(254,254,254,0.8)");
_a9.addColorStop(1,"rgba(192,192,192,0)");
}else{
_a9.addColorStop(0,"rgba(254,254,254,0)");
_a9.addColorStop(1,"rgba(192,192,192,0)");
}
}
ctx.strokeStyle=_a9;
ctx.lineWidth=_a6;
ctx.rect(x,y,_a3,_a4);
ctx.stroke();
},tiltShadow:function(ctx,x,y,_ad,_ae,_af,_b0){
var _b1;
ctx.beginPath();
ctx.rect(x,y+_ae-_af,_af,_af);
ctx.closePath();
_b1=this.addRadialStyle(ctx,x+_af,y+_ae-_af,_af-x,x+_af,y+_ae-_af,_af,_b0);
ctx.fillStyle=_b1;
ctx.fill();
ctx.beginPath();
ctx.rect(x+_af,y+_ae-y,_ad-(_af*2.25),y);
ctx.closePath();
_b1=this.addLinearStyle(ctx,x+_af,y+_ae-y,x+_af,y+_ae,_b0);
ctx.fillStyle=_b1;
ctx.fill();
ctx.beginPath();
ctx.rect(x+_ad-(_af*1.25),y+_ae-(_af*1.25),_af*1.25,_af*1.25);
ctx.closePath();
_b1=this.addRadialStyle(ctx,x+_ad-(_af*1.25),y+_ae-(_af*1.25),(_af*1.25)-1.5-x,x+_ad-(_af*1.25),y+_ae-(_af*1.25),_af*1.25,_b0);
ctx.fillStyle=_b1;
ctx.fill();
ctx.beginPath();
ctx.moveTo(x+_ad-x,y+_af);
ctx.lineTo(x+_ad,y+_af);
ctx.quadraticCurveTo(x+_ad-x,y+(_ae/2),x+_ad,y+_ae-(_af*1.25));
ctx.lineTo(x+_ad-x,y+_ae-(_af*1.25));
ctx.quadraticCurveTo(x+_ad-(x*2),y+(_ae/2),x+_ad-x,y+_af);
ctx.closePath();
_b1=this.addLinearStyle(ctx,x+_ad-x,y+_af,x+_ad,y+_af,_b0);
ctx.fillStyle=_b1;
ctx.fill();
ctx.beginPath();
ctx.rect(x+_ad-_af,y,_af,_af);
ctx.closePath();
_b1=this.addRadialStyle(ctx,x+_ad-_af,y+_af,_af-x,x+_ad-_af,y+_af,_af,_b0);
ctx.fillStyle=_b1;
ctx.fill();
}});

var Reflex=Class.create(CanvasBase,{initialize:function(_1){
this.setOptions(_1);
this.start();
this.begin();
},setOptions:function(_2){
this.options={onloaded:null,classname:"reflex",alt:true};
Object.extend(this.options,_2||{});
},begin:function(){
var _3=true;
var _4=this.getImages(this.options.classname);
for(var nr=0;nr<_4.length;++nr){
var _6=_4[nr];
if(!_6.complete||(typeof _6.naturalWidth!="undefined"&&_6.naturalWidth==0)){
window.setTimeout(function(){
this.begin();
}.bind(this),100);
_3=false;
break;
}
}
if(_3){
if(Prototype.Browser.IE){
this.addIEReflex();
}else{
this.addReflex();
}
if(this.options.onloaded){
this.options.onloaded();
}
}
},addIEReflex:function(){
var _7=this.getImages(this.options.classname);
var _8,_9,_a,_b,_c,_d,_e,_f,_10,_11,_12,_13;
var i,j,_16,stl,_18,_19,_1a,_1b,_1c,_1d,_1e,_1f,ih,iw,fb,xb;
var _24,_25,_26,_27,_28,_29,_2a,_2b;
var _2c=document.getElementsByTagName("img");
var _2d="r";
for(i=0;i<_7.length;i++){
_8=_7[i];
_9=_8.parentNode;
_f=0;
_24=0;
_25=0;
_26=0;
_27=33;
_28=33;
_29=0;
_2a=0;
_2b="#000000";
if(_8.width>=32&&_8.height>=32){
_d=_8.className.split(" ");
_2a=this.getClassValue(_d,"iborder");
_27=this.getClassValue(_d,"iheight");
_28=this.getClassValue(_d,"iopacity");
_29=this.getClassValue(_d,"idistance");
_2b=this.getClassColor(_d,"icolor");
_26=this.getClassAttribute(_d,"itiltleft");
_24=this.getClassAttribute(_d,"itiltright");
_25=this.getClassAttribute(_d,"itiltnone");
if(_24==true){
_2d="r";
}
if(_25==true){
_2d="n";
}
if(_26==true){
_2d="l";
}
_e=this.getClasses(_d,this.options.classname);
ih=_8.height;
iw=_8.width;
var _16=_29;
_1d=(_2b!=0?_2b:"#000000");
_1c=(_28>0?_28:33);
_1b=100/(_27>=10?_27:33);
_1a=Math.floor(ih/_1b);
_2a=Math.round(Math.min(Math.min(_2a,_1a/4),Math.max(iw,ih)/20));
_1f=12;
var _1e=(_2a>0?_2a/2:0);
_b=(_8.currentStyle.display.toLowerCase()=="block")?"block":"inline-block";
_a=document.createElement(["<var style=\"overflow:hidden;display:"+_b+";width:"+iw+"px;height:"+(ih+_1a+_16)+"px;padding:0;\">"].join(""));
_c=_8.currentStyle.styleFloat.toLowerCase();
_b=(_c=="left"||_c=="right")?"inline":_b;
_10="<v:group style=\"zoom:1; display:"+_b+"; margin:-1px 0 0 -1px; padding:0; position:relative; width:"+iw+"px;height:"+(ih+_1a+_16)+"px;\" coordsize=\""+iw+","+(ih+_1a+_16)+"\">";
if(_2d=="n"){
_11="<v:rect strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#ffffff\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:0px; left:0px; width:"+iw+"px;height:"+ih+"px;\"><v:fill src=\""+_8.src+"\" type=\"frame\" /></v:rect>";
fb="<v:rect strokeweight=\""+_2a+"\" strokecolor=\""+_1d+"\" filled=\"f\" stroked=\""+(_1e>0||_2a>0?"t":"f")+"\" fillcolor=\"#ffffff\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:"+_1e+"px; left:"+_1e+"px; width:"+(iw-_1e-_1e)+"px;height:"+(ih-_1e-_1e)+"px;\"></v:rect>";
xb="<v:rect strokeweight=\""+_2a+"\" strokecolor=\""+_1d+"\" filled=\"f\" stroked=\""+(_1e>0||_2a>0?"t":"f")+"\" fillcolor=\"#ffffff\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:"+(ih+_16+_1e)+"px; left:"+_1e+"px; width:"+(iw-_1e-_1e)+"px;height:"+(_1a-_1e-_1e)+"px; filter: progid:DXImageTransform.Microsoft.Alpha(opacity="+_1c+",style=1,finishOpacity=0,startx=0,starty=0,finishx=0,finishy="+parseInt(ih*0.66)+");\"></v:rect>";
_12="<v:rect strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#ffffff\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:"+(ih+_16)+"px; left:0px; width:"+iw+"px;height:"+_1a+"px; filter:flipv progid:DXImageTransform.Microsoft.Alpha(opacity="+_1c+",style=1,finishOpacity=0,startx=0,starty=0,finishx=0,finishy="+ih+");\"><v:fill origin=\"0,0\" position=\"0,-"+(_1b/2-0.5)+"\" size=\"1,"+(1*_1b)+"\" src=\""+_8.src+"\" type=\"frame\" /></v:rect>";
}else{
if(_2d=="r"){
_11="<v:rect strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#808080\" style=\"position:absolute; margin:-1px 0 0 -1px;padding:0 ;width:"+iw+"px;height:"+(ih+_1a+_16)+"px;\"><v:fill color=\"#808080\" opacity=\"0.0\" /></v:rect><v:shape strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#ffffff\" coordorigin=\"0,0\" coordsize=\""+iw+","+ih+"\" path=\"m "+ww+",0 l "+ww+","+ih+","+(iw-ww)+","+(ih-hh)+","+(iw-ww)+","+hh+" x e\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:0px; left:0px; width:"+iw+"px;height:"+ih+"px;\"><v:fill src=\""+_8.src+"\" type=\"frame\" /></v:shape>";
for(j=0;j<_18;j++){
if(j==(_18-1)){
q=(_19>0?1:0);
}
_11+="<v:shape strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#808080\" coordorigin=\"0,0\" coordsize=\""+iw+","+ih+"\" path=\"m "+(ww+(j*_1f))+","+j+" l "+(q+ww+((j+1)*_1f))+","+(j+1)+","+(q+ww+((j+1)*_1f))+","+(ih-1-j)+","+(ww+(j*_1f))+","+(ih-j)+" x e\" style=\"position:absolute; margin: -1px 0 0 -1px; padding:0px; top:0px; left:0px; width:"+iw+"px; height:"+ih+"px;\"><v:fill origin=\"0,0\" position=\""+(half-j)+",0\" size=\""+((iw-ww-ww)/_1f)+",1\" type=\"frame\" src=\""+_8.src+"\" /></v:shape>";
}
if(_19>0){
_11+="<v:shape strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#808080\" coordorigin=\"0,0\" coordsize=\""+iw+","+ih+"\" path=\"m "+(ww+(j*_1f))+","+j+" l "+(ww+((j+1)*_1f))+","+(j+1)+","+(ww+((j+1)*_1f))+","+(ih-1-j)+","+(ww+(j*_1f))+","+(ih-j)+" x e\" style=\"position:absolute; margin: -1px 0 0 -1px; padding:0px; top:0px; left:0px; width:"+iw+"px; height:"+ih+"px;\"><v:fill origin=\"0,0\" position=\""+(half-j)+",0\" size=\""+((iw-ww-ww)/_1f)+",1\" type=\"frame\" src=\""+_8.src+"\" /></v:shape>";
}
q=((_18*z)/(ih/100))/2;
if(_1e>0||_2a>0){
fb="<v:shape strokeweight=\""+_2a+"\" strokecolor=\""+_1d+"\" filled=\"f\" stroked=\""+(_1e>0||_2a>0?"t":"f")+"\" coordorigin=\"0,0\" coordsize=\""+iw+","+ih+"\" path=\"m "+(ww+_1e)+","+_1e+" l "+(ww+_1e)+","+(ih-_1e)+","+(iw-ww-_1e)+","+(ih-hh-_1e)+","+(iw-ww-_1e)+","+(hh+_1e)+" x e\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:0px; left:0px; width:"+iw+"px;height:"+ih+"px;\"></v:shape>";
xb="<v:shape strokeweight=\""+_2a+"\" strokecolor=\""+_1d+"\" stroked=\""+(_1e>0||_2a>0?"t":"f")+"\" filled=\"f\" coordorigin=\"0,0\" coordsize=\""+iw+","+(hh+_1a+_16)+"\" path=\"m "+(ww+_1e)+","+(hh+_16+_1e)+" l "+(ww+_1e)+","+(_1a+hh+_16-_1e)+","+(iw-ww-_1e)+","+(parseInt((_1a+_16)*z)-_1e)+","+(iw-ww-_1e)+","+(parseInt(_16*z)+_1e)+" x e\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:"+(ih-hh+_16)+"px; left:0px; width:"+iw+"px;height:"+(hh+_1a+_16)+"px; flip: y; filter:flipv progid:DXImageTransform.Microsoft.Alpha(opacity="+_1c+",style=1,finishOpacity=0,startx=0,starty=0,finishx="+q+",finishy=80);\"></v:shape>";
}else{
fb="";
xb="";
}
_12="<v:shape strokeweight=\"0\" stroked=\"f\" filled=\"t\" fillcolor=\"#808080\" coordorigin=\"0,0\" coordsize=\""+iw+","+(hh+_1a+_16)+"\" path=\"m "+ww+","+(hh+_16)+" l "+ww+","+(_1a+hh+_16)+","+(iw-ww)+","+parseInt((_1a+_16)*z)+","+(iw-ww)+","+parseInt(_16*z)+" x e\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:"+(ih-hh+_16)+"px; left:0px; width:"+iw+"px;height:"+(hh+_1a+_16)+"px; flip: y; filter:flipv progid:DXImageTransform.Microsoft.Alpha(opacity="+_1c+",style=1,finishOpacity=0,startx=0,starty=0,finishx="+q+",finishy=90);\"><v:fill origin=\"0,0\" position=\"0,-"+((_1b/2)-0.5)+"\" size=\"1,"+(_1b)+"\" src=\""+_8.src+"\" type=\"frame\" /></v:shape>";
}else{
if(_2d=="l"){
_11="<v:rect strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#808080\" style=\"position:absolute; margin:-1px 0 0 -1px;padding:0 ;width:"+iw+"px;height:"+(ih+_1a+_16)+"px;\"><v:fill color=\"#808080\" opacity=\"0.0\" /></v:rect><v:shape strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#ffffff\" coordorigin=\"0,0\" coordsize=\""+iw+","+ih+"\" path=\"m "+ww+","+hh+" l "+ww+","+(ih-hh)+","+(iw-ww)+","+ih+","+(iw-ww)+",0 x e\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:0px; left:0px; width:"+iw+"px;height:"+ih+"px;\"><v:fill src=\""+_8.src+"\" type=\"frame\" /></v:shape>";
for(j=0;j<_18;j++){
if(j==(_18-1)){
q=(_19>0?1:0);
}
_11+="<v:shape strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#808080\" coordorigin=\"0,0\" coordsize=\""+iw+","+ih+"\" path=\"m "+(ww+(j*_1f))+","+(_18-j)+" l "+(q+ww+((j+1)*_1f))+","+(_18-1-j)+","+(q+ww+((j+1)*_1f))+","+(ih-1-_18+j)+","+(ww+(j*_1f))+","+(ih-_18+j)+" x e\" style=\"position:absolute; margin: -1px 0 0 -1px; padding:0px; top:0px; left:0px; width:"+iw+"px; height:"+ih+"px;\"><v:fill origin=\"0,0\" position=\""+(half-j)+",0\" size=\""+((iw-ww-ww)/_1f)+",1\" type=\"frame\" src=\""+_8.src+"\" /></v:shape>";
}
if(_19>0){
_11+="<v:shape strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#808080\" coordorigin=\"0,0\" coordsize=\""+iw+","+ih+"\" path=\"m "+(ww+(j*_1f))+","+(_18-j)+" l "+(ww+((j+1)*_1f))+","+(_18-1-j)+","+(ww+((j+1)*_1f))+","+(ih-1-_18+j)+","+(ww+(j*_1f))+","+(ih-_18+j)+" x e\" style=\"position:absolute; margin: -1px 0 0 -1px; padding:0px; top:0px; left:0px; width:"+iw+"px; height:"+ih+"px;\"><v:fill origin=\"0,0\" position=\""+(half-j)+",0\" size=\""+((iw-ww-ww)/_1f)+",1\" type=\"frame\" src=\""+_8.src+"\" /></v:shape>";
}
q=100-(((_18*z)/(ih/100))/2);
if(_1e>0||_2a>0){
fb="<v:shape strokeweight=\""+_2a+"\" strokecolor=\""+_1d+"\" filled=\"f\" stroked=\""+(_1e>0||_2a>0?"t":"f")+"\" coordorigin=\"0,0\" coordsize=\""+iw+","+ih+"\" path=\"m "+(ww+_1e)+","+(hh+_1e)+" l "+(ww+_1e)+","+(ih-hh-_1e)+","+(iw-ww-_1e)+","+(ih-_1e)+","+(iw-ww-_1e)+","+_1e+" x e\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:0px; left:0px; width:"+iw+"px;height:"+ih+"px;\"></v:shape>";
xb="<v:shape strokeweight=\""+_2a+"\" strokecolor=\""+_1d+"\" stroked=\""+(_1e>0||_2a>0?"t":"f")+"\" filled=\"f\" coordorigin=\"0,0\" coordsize=\""+iw+","+(hh+_1a+_16)+"\" path=\"m "+(ww+_1e)+","+(parseInt(_16*z)+_1e)+" l "+(ww+_1e)+","+(parseInt((_1a+_16)*z)-_1e)+","+(iw-ww-_1e)+","+(_1a+hh+_16-_1e)+","+(iw-ww-_1e)+","+(hh+_16+_1e)+" x e\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:"+(ih-hh+_16)+"px; left:0px; width:"+iw+"px;height:"+(hh+_1a+_16)+"px; flip: y; filter:flipv progid:DXImageTransform.Microsoft.Alpha(opacity="+_1c+",style=1,finishOpacity=0,startx=100,starty=0,finishx="+q+",finishy=80);\"></v:shape>";
}else{
fb="";
xb="";
}
_12="<v:shape strokeweight=\"0\" filled=\"t\" stroked=\"f\" fillcolor=\"#808080\" coordorigin=\"0,0\" coordsize=\""+iw+","+(hh+_1a+_16)+"\" path=\"m "+ww+","+parseInt(_16*z)+" l "+ww+","+parseInt((_1a+_16)*z)+","+(iw-ww)+","+(_1a+hh+_16)+","+(iw-ww)+","+(hh+_16)+" x e\" style=\"position:absolute; margin:-1px 0 0 -1px; padding:0; top:"+(ih-hh+_16)+"px; left:0px; width:"+iw+"px;height:"+(hh+_1a+_16)+"px; flip: y; filter:flipv progid:DXImageTransform.Microsoft.Alpha(opacity="+_1c+",style=1,finishOpacity=0,startx=100,starty=0,finishx="+q+",finishy=90);\"><v:fill origin=\"0,0\" position=\"0,-"+((_1b/2)-0.5)+"\" size=\"1,"+(_1b)+"\" src=\""+_8.src+"\" type=\"frame\" /></v:shape>";
}
}
}
_13="</v:group>";
_a.resizeinternal=function(w,h,_30){
var _31=(h*_30);
var _32=h-_31;
var _33=this.down();
Element.setStyle(_33,{width:w+"px",height:h+"px"});
_33.coordsize=w+","+h;
var _34=_33.childNodes[2];
Element.setStyle(_34,{width:w+"px",height:_32+"px"});
var _35=this.bord*2;
var _36=_33.childNodes[3];
Element.setStyle(_36,{width:(w-_35)+"px",height:(_32-_35)+"px"});
var _37=_33.childNodes[1];
Element.setStyle(_37,{top:(_32+this.dist+this.bord)+"px",width:(w-_35)+"px",height:(_31-_35)+"px"});
var _38=_33.childNodes[0];
Element.setStyle(_38,{top:(_32+this.dist)+"px",width:w+"px",height:_31+"px"});
};
_a.dist=_16;
_a.bord=_1e;
_a.innerHTML=_10+_12+xb+_11+fb+_13;
_a.className=_e;
_a.style.cssText=_8.style.cssText;
_a.style.height=ih+_1a+_16+"px";
_a.width=iw;
_a.height=ih+_1a+_16;
_a.style.width=iw+"px";
_a.src=_8.src;
_a.alt=_8.alt;
if(_8.id!=""){
_a.id=_8.id;
}
if(_8.title!=""){
_a.title=_8.title;
}
if(_8.getAttribute("onclick")!=""){
_a.setAttribute("onclick",_8.getAttribute("onclick"));
}
_9.replaceChild(_a,_8);
_a.style.visibility="visible";
}
}
},addReflex:function(){
var _39=this.getImages(this.options.classname);
var _3a,_3b,_3c,_3d,_3e,_3f,tmp;
var i,j,_43,stl,_45,_46,_47,_48,_49,_4a,_4b,_4c,ih,iw;
var _4f,_50,_51,_52,_53,_54,_55,_56;
var _57=document.getElementsByTagName("img");
var _58="r";
var _59=Prototype.Browser.WebKit;
var _5a=Prototype.Browser.Opera;
for(i=0;i<_39.length;i++){
_3a=_39[i];
_3b=_3a.parentNode;
tmp=0;
_4f=0;
_50=0;
_51=0;
_52=33;
_53=33;
_54=0;
_55=0;
_56="#000000";
_3c=document.createElement("canvas");
if(_3c.getContext&&_3a.width>=32&&_3a.height>=32){
_3e=_3a.className.split(" ");
_55=this.getClassValue(_3e,"iborder");
_52=this.getClassValue(_3e,"iheight");
_53=this.getClassValue(_3e,"iopacity");
_54=this.getClassValue(_3e,"idistance");
_56=this.getClassColor(_3e,"icolor");
_51=this.getClassAttribute(_3e,"itiltleft");
_4f=this.getClassAttribute(_3e,"itiltright");
_50=this.getClassAttribute(_3e,"itiltnone");
if(_4f==true){
_58="r";
}
if(_50==true){
_58="n";
}
if(_51==true){
_58="l";
}
_3f=this.getClasses(_3e,this.options.classname);
ih=_3a.height;
iw=_3a.width;
_43=_54;
_4a=(_56!=0?_56:"#000000");
_49=(100-(_53>0?_53:33))/100;
_48=100/(_52>=10?_52:33);
_47=Math.floor(_3a.height/_48);
_55=Math.round(Math.min(Math.min(_55,_47/4),Math.max(iw,ih)/20));
_4c=12;
_4b=(_55>0?_55/2:0);
_3c.className=_3f;
_3c.style.cssText=_3a.style.cssText;
_3c.style.height=ih+_47+_43+"px";
_3c.width=iw;
_3c.style.width=iw+"px";
_3c.height=ih+_47+_43;
_3c.src=_3a.src;
_3c.alt=_3a.alt;
var hh=_3c.height;
var ww=_3c.width;
if(_3a.id!=""){
_3c.id=_3a.id;
}
if(_3a.title!=""){
_3c.title=_3a.title;
}
if(_3a.getAttribute("onclick")!=""){
_3c.setAttribute("onclick",_3a.getAttribute("onclick"));
}
_45=Math.floor(_3c.width/_4c);
_46=(_3c.width%_4c);
if(_58=="l"||_58=="r"){
var _5d=document.createElement("canvas");
if(_5d.getContext){
_5d.style.position="fixed";
_5d.style.left=-9999+"px";
_5d.style.top=0+"px";
_5d.height=_3c.height;
_5d.width=_3c.width;
_5d.style.height=_3c.height+"px";
_5d.style.width=_3c.width+"px";
if(_59){
_3b.appendChild(_5d);
}
}
}
if(_5a){
_3c.style.visibility="hidden";
}
_3d=_3c.getContext("2d");
_3b.replaceChild(_3c,_3a);
_3d.clearRect(0,0,_3c.width,_3c.height);
_3d.globalCompositeOperation="source-over";
_3d.fillStyle="rgba(0,0,0,0)";
_3d.fillRect(0,0,_3c.width,_3c.height);
_3d.save();
_3d.translate(0,_3c.height-(_59?0:1));
_3d.scale(1,-1);
_3d.drawImage(_3a,0,-(_3c.height-_47-_47-_43),_3c.width,_3c.height-_47-_43);
_3d.restore();
if(_55>0){
_3d.strokeStyle=_4a;
_3d.lineWidth=_55;
_3d.beginPath();
tmp=0;
if(_5a&&_58!="n"){
tmp=1;
}
if(_5a&&_58=="n"){
tmp=-1;
}
_3d.rect(_4b,_3c.height-_47+_4b+tmp,_3c.width-_55,_47);
_3d.closePath();
_3d.stroke();
}
if(!_59||_58=="n"){
tmp=0;
if(_5a){
tmp=1;
}
_3d.globalCompositeOperation="destination-out";
stl=_3d.createLinearGradient(0,_3c.height-_47-tmp,0,_3c.height);
stl.addColorStop(1,"rgba(0,0,0,1.0)");
stl.addColorStop(0,"rgba(0,0,0,"+_49+")");
_3d.fillStyle=stl;
}
if(_59){
_3d.beginPath();
_3d.rect(0,_3c.height-_47,_3c.width,_47);
_3d.closePath();
_3d.fill();
}else{
tmp=0;
if(_5a&&_58=="n"){
tmp=1;
}
_3d.fillRect(0,_3c.height-_47-tmp,_3c.width,_47+tmp);
}
_3d.globalCompositeOperation="source-over";
_3d.drawImage(_3a,0,0,iw,ih);
_3d.save();
if(_59&&_43>0&&_58!="n"){
_3d.fillStyle="#808080";
_3d.fillRect(0,_3c.height-_47-_43,_3c.width,_43);
}
if(_55>0){
if(_58=="n"){
_3d.beginPath();
_3d.rect(_4b,_4b,_3c.width-_55,_3c.height-_47-_43-_55);
_3d.closePath();
_3d.stroke();
}
}
if(_58=="l"||_58=="r"){
if(_5d.getContext){
_3d=_5d.getContext("2d");
globalCompositeOperation="source-over";
_3d.clearRect(0,0,_5d.width,_5d.height);
if(_58=="r"){
for(j=0;j<_45;j++){
_3d.drawImage(_3c,j*_4c,0,_4c,_5d.height,j*_4c,j*1,_4c,_5d.height-(j*2));
}
if(_46>0){
_46=_3c.width-(_45*_4c);
_3d.drawImage(_3c,j*_4c,0,_46,_5d.height,j*_4c,j*1,_46,_5d.height-(j*2));
}
}else{
for(j=0;j<_45;j++){
_3d.drawImage(_3c,j*_4c,0,_4c,_5d.height,j*_4c,(_45-j)*1,_4c,_5d.height-((_45-j)*2));
}
if(_46>0){
_46=_3c.width-(_45*_4c);
_3d.drawImage(_3c,j*_4c,0,_46,_5d.height,j*_4c,0,_46,_5d.height);
}
}
_3d.save();
if(_3c.getContext){
_3d=_3c.getContext("2d");
_3d.clearRect(0,0,_3c.width,_3c.height);
globalCompositeOperation="source-in";
if(_58=="r"){
this.clipPolyRight(_3d,_3c.width/20,0,_3c.width*0.95,_3c.height,_45+(_46>0?1:0),_43,_47);
}else{
this.clipPolyLeft(_3d,_3c.width/20,0,_3c.width*0.95,_3c.height,_45+(_46>0?1:0),_43,_47);
}
_3d.clip();
_3d.drawImage(_5d,parseInt(_3c.width/20),0,parseInt(_3c.width*0.9),_3c.height);
_3d.save();
if(_55>0){
_3d.lineWidth=_55;
if(_58=="r"){
this.strokePolyRight(_3d,_3c.width/20,0,_3c.width*0.95,_3c.height,_45+(_46>0?1:0),_43,_47,_4b);
_3d.stroke();
}else{
this.strokePolyLeft(_3d,_3c.width/20,0,_3c.width*0.95,_3c.height,_45+(_46>0?1:0),_43,_47,_4b);
_3d.stroke();
}
}
if(_59){
_3d.globalCompositeOperation="destination-out";
stl=_3d.createLinearGradient((_58=="l"?_3c.width:0),_3c.height-_47,(_58=="l"?_3c.width-parseInt(_4c/_48):parseInt(_4c/_48)),_3c.height);
stl.addColorStop(1,"rgba(255,0,0,1.0)");
stl.addColorStop(0,"rgba(255,0,0,"+_49+")");
_3d.fillStyle=stl;
this.clipReflex(_3d,_3c.width/20,0,_3c.width*0.95,_3c.height,_45+(_46>0?1:0),_43,_47,_58);
_3d.fill();
_3d.globalCompositeOperation="source-in";
this.clearReflex(_3d,_3c.width/20,0,_3c.width*0.95,_3c.height,_45+(_46>0?1:0),_43,_47,_58);
_3d.clip();
_3d.clearRect(0,0,_3c.width,_3c.height);
_3b.removeChild(_5d);
}
}
}
}
if(_5a){
_3c.style.visibility="visible";
}
if(this.options.alt){
if(_58=="r"){
_58="n";
}else{
if(_58=="n"){
_58="l";
}else{
if(_58=="l"){
_58="r";
}
}
}
}
_3d.save();
_3c.style.visibility="visible";
}
}
}});
document.observe('dom:loaded', function(){var gl = new Reflex();} );
function offsets()
{
  var deltaX = window.pageXOffset || document.documentElement.scrollLeft || document.body.scrollLeft || 0;
  var deltaY = window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop || 0;
  return {x:deltaX,y:deltaY};
}

function getwinwidth()
{
  if( window.innerWidth )
    return window.innerWidth;
  else if ( window.document.documentElement && document.documentElement.clientWidth )
    return document.documentElement.clientWidth;
  else
    return body.offsetWidth;
}

function getwinheight()
{
  if( window.innerHeight )
    return window.innerHeight;
  else if ( window.document.documentElement && document.documentElement.clientHeight )
    return document.documentElement.clientHeight;
  else
    return body.offsetHeight;
}

function createpopup(items, sid, urlsrc, scroll)
{
    var ret = new Popup( items, {
        onbeginexpand:function(thep){
            var tb = new Transbase({opacity:0.5,zindex:1001,backcolour:'#111111'});
            tb.show();
            var ss, txt;
            
            if ( Adito.Detector.isIE() )
            {
                ss = (scroll === null || scroll === 0)? 'yes': 'no';
                var ex = (scroll === null || scroll === 0)? 'overflow:auto;': 'overflow:hidden;';
                txt = '<iframe id="if'+sid+'" src="'+urlsrc+'" width="888" height="638" frameborder="no" scrolling="'+ss+'" vspace="0" hspace="0" style="width:888px;height:638px;'+ex+'"></iframe>';
            }
            else
            {
                ss = (scroll === null || scroll === 0)? 'overflow:auto;': 'overflow:hidden;';
                txt = '<object id="if'+sid+'" type="text/html" data="'+urlsrc+'" width="888" height="638" vspace="0" hspace="0" style="width:888px;height:638px;'+ss+'"></object>';
            }
                
            var thediv = document.createElement('div');
            Element.addClassName(thediv, 'outer');
            Element.setStyle(thediv, {width:'888px', height:'638px'});
            thediv.id='id'+sid;
            thediv.innerHTML = txt;
            thep.appendChild(thediv);
        },
        onendexpand:function(thep){
            $('id'+sid).style.visibility='visible';
        },
        onbeginshrink:function(thep){
            var tb = new Transbase();tb.hide();
            thep.removeChild( $('id'+sid));
        },
        popupshadowid: 'popupshadow'+sid,
        id:'popt'+sid});
    return ret;
}
function createdarkpopup(items, sid, urlsrc, scroll)
{
    var ret = new Popup( items, {
        popupclass: 'popup_movie',
        popupshadowclass: 'popupshadow_movie',
        popupimage: '/music/local/images/popup_movie.png',
        onbeginexpand:function(thep){
            var tb = new Transbase({opacity:0.5,zindex:1001,backcolour:'#111111'});
            tb.show();
            var ss, txt;
            
            if ( Adito.Detector.isIE() )
            {
                ss = (scroll === null || scroll === 0)? 'yes': 'no';
                var ex = (scroll === null || scroll === 0)? 'overflow:auto;': 'overflow:hidden;';
                txt = '<iframe id="if'+sid+'" src="'+urlsrc+'" width="888" height="444" frameborder="no" scrolling="'+ss+'" vspace="0" hspace="0" style="width:888px;height:444px;'+ex+'"></iframe>';
            }
            else
            {
                ss = (scroll === null || scroll === 0)? 'overflow:auto;': 'overflow:hidden;';
                txt = '<object id="if'+sid+'" type="text/html" data="'+urlsrc+'" width="888" height="444" vspace="0" hspace="0" style="width:888px;height:444px;'+ss+'"></object>';
            }

            var thediv = document.createElement('div');
            Element.addClassName(thediv, 'outer');
            Element.setStyle(thediv, {width:'888px', height:'444px'});
            thediv.id='id'+sid;
            thediv.innerHTML = txt;
            thep.appendChild(thediv);
        },
        onendexpand:function(thep){$('id'+sid).style.visibility='visible';},
        onbeginshrink:function(thep){var tb = new Transbase();tb.hide();thep.removeChild( $('id'+sid));},
        popupshadowid: 'popupshadow'+sid,
        id:'popt'+sid});
    return ret;
}

function getelheight(element)
{
  var els = element.style;
  var originalVisibility = els.visibility;
  var originalPosition = els.position;
  var originalDisplay = els.display;
  els.visibility = 'hidden';
  els.position = 'absolute';
  els.display = 'block';
  var originalHeight = element.clientHeight;
  els.display = originalDisplay;
  els.position = originalPosition;
  els.visibility = originalVisibility;
  return originalHeight;
}

function divslide(id)
{
  var elo = $(id);
  var isvisi = elo.style.display;
  if( isvisi == "none")
  {
    elo.style.height = "auto";
    var height = getelheight(elo);
    elo.style.display="block";
    elo.style.height= "0px";
    
    new Animator({onComplete:function(){
      elo.style.height='auto';
    },duration:500, interval:20,transition:Animator.tx.linear}).addSubject(new NumericalStyleSubject(elo, 'height', 0, height )).play();
  }
  else
  {
    new Animator({onComplete:function(){
      elo.style.display='none';
    },duration:500, interval:20,transition:Animator.tx.linear}).addSubject(new NumericalStyleSubject(elo, 'height', elo.offsetHeight, 0 )).play();
  }
}

var lastsubopen = null;
var lastopen = null;

function divfill(elo, varidx)
{
  elo = $(elo);

  var subel = varidx['el'];
  var pid = varidx['pid'];
  var proddescr = varidx['desc'];

  var isvisi = elo.style.display;
  if( isvisi == "none")
  {
    var thefunc = function()
      {
        elo.style.height = "auto";
        if ( lastopen != null )
        {
          lastopen.style.display = "none";
        }
          
        lastsubopen = subel;
        lastopen = elo;
      }
      
    var an = new Animator({ onComplete:thefunc, duration:500, interval:20,transition:Animator.tx.linear});

    if ( lastsubopen != null )
    {
      lastsubopen.innerHTML = "";
      an.addSubject(new NumericalStyleSubject(lastopen, 'height', lastopen.offsetHeight, 0 ));
    }

    innercompcust(subel, pid, proddescr );

    elo.style.height = "auto";
    var height = getelheight(elo);
    elo.style.display = "block";
    elo.style.height = "0px";
    
    an.addSubject(new NumericalStyleSubject(elo, 'height', 0, height ));    
    an.play();
  }
  else
  {
    new Animator({onComplete:function(){
      elo.style.display='none';
      if ( lastsubopen != null )
        lastsubopen.innerHTML = "";
      if ( lastopen == elo )
      {
        lastopen = null;
        lastsubopen = null;
      }
    },duration:500, interval:20,transition:Animator.tx.linear}).addSubject(new NumericalStyleSubject(elo, 'height', elo.offsetHeight, 0 )).play();
  }
}

function slidedivtype( pid, type )
{
  var el = document.getElementById(pid);
  var isvisi = el.style.display;
  if ( type == 'wav' || type == 'aiff' )
  {
    if( isvisi == "none")
    {
      el.style.height = "auto";
      var height = getelheight(el);
      el.style.display="block";
      el.style.height= "0px";
    
      new Animator({onComplete:function(){
        el.style.height='auto';
      },duration:500, interval:20,transition:Animator.tx.linear}).addSubject(new NumericalStyleSubject(el, 'height', 0, height )).play();
    }
  }
  else
  {
    if ( isvisi == "block" )
    {
      new Animator({onComplete:function(){
        el.style.display='none';
      },duration:500, interval:20,transition:Animator.tx.linear}).addSubject(new NumericalStyleSubject(el, 'height', el.offsetHeight, 0 )).play();
    }
  }
}

function innercompopt(el, pid, getaddr, proddescr )
{
  var element = $(el);
  var html = '<form id="wavgend'+pid+'" method="get" action="'+getaddr+'">';
  html += '<input type="hidden" id="pid" name="pid" value="'+pid+'" />';
  html += '<input type="hidden" id="grp" name="grp" value="'+proddescr+'" />';
  html += '</form>';
  element.innerHTML = html;
  document.forms["wavgend"+pid].submit();
}

function innercompcust(el, pid, proddescr)
{
  var element = $(el);
  
  var html = ['<div>',
  '<form id="custwavgen'+pid+'" method="get" action="/music/composerget.php">',
  '<input type="hidden" id="customwave" name="customwave" value="1" />',
  '<input type="hidden" id="pid" name="pid" value="'+pid+'" />',
  '<input type="hidden" id="grp" name="grp" value="'+proddescr+'" />',
  '<div style="float:left;width:80%;">',
  '<div id="formatdiv'+pid+'">',
  '<div style="height:4px;"></div>',
  '<div><span class="divtxt">Audio Format: </span>',
  '<input id="wavtype" name="wavtype" type="radio" value="wav" onchange="javascript:slidedivtype(\'encodingtype'+pid+'\',\'wav\');" checked="checked" /><span>&nbsp;Broadcast Wave</span>',
  '<input id="wavtype" name="wavtype" type="radio" value="aiff" onchange="javascript:slidedivtype(\'encodingtype'+pid+'\',\'aiff\');" /><span>&nbsp;Apple AIFF Format</span>',
  '<input id="wavtype" name="wavtype" type="radio" value="ogg" onchange="javascript:slidedivtype(\'encodingtype'+pid+'\',\'ogg\');" /><span>&nbsp;OGG Format</span>',
  '</div>',
  '<div id="encodingtype'+pid+'" style="display:block;overflow:hidden;height:auto;">',
  '<div><span class="divtxt">Audio Encoding Type: </span>',
  '<input id="subtype" name="subtype" type="radio" value="FORMAT_PCM_S8" /><span>&nbsp;Signed 8bit PCM</span>',
  '<input id="subtype" name="subtype" type="radio" value="FORMAT_PCM_U8" /><span>&nbsp;Unsigned 8bit PCM</span>',
  '<input id="subtype" name="subtype" type="radio" value="FORMAT_PCM_16" checked="checked" /><span>&nbsp;16bit PCM</span>',
  '<input id="subtype" name="subtype" type="radio" value="FORMAT_PCM_24" /><span>&nbsp;24bit PCM</span>',
  '<input id="subtype" name="subtype" type="radio" value="FORMAT_PCM_32" /><span>&nbsp;32bit PCM</span>',
  '<input id="subtype" name="subtype" type="radio" value="FORMAT_ULAW" /><span>&nbsp;ULAW</span>',
  '<input id="subtype" name="subtype" type="radio" value="FORMAT_ALAW" /><span>&nbsp;ALAW</span>',
  '<input id="subtype" name="subtype" type="radio" value="FORMAT_IMA_ADPCM" /><span>&nbsp;IMA ADPCM</span>',
  '</div>',
  '</div>',
  '<div style="height:4px;"></div>',
  '<div><span class="divtxt">Your Preferred Sample Rate: </span>',
  '<select name="format" id="format">',
  '<option value="8000">8K</option>',
  '<option value="16000">16K</option>', 
  '<option value="24000">24K</option>',
  '<option value="32000">32K</option>',
  '<option value="44056">44.1K pull-down ( 44.056K )</option>',
  '<option value="44100" selected="selected">44.1K</option>',
  '<option value="44056">44.1K pull-up ( 44.144K )</option>',
  '<option value="47952">48K pull-down ( 47.952K )</option>',
  '<option value="48000">48K</option>',
  '<option value="48048">48K pull-up ( 48.048K )</option>',
  '</select>',
  '</div>',
  '<div style="height:4px;"></div>',
  '</div>',
  '</div>',
  '<div style="float:right;width:18%;">',
  '<div>',
  '<div style="height:4px;"></div>',
  '<input type="image" src="/music/includes/languages/english/images/buttons/lpherecustom.png" value="download" />',
  '</div>',
  '</form>',
  '</div>' ].join("");

  element.innerHTML = html;
}

function setcheck( id, onimg, offimg )
{
  var checkimg = "sel_" + id;
  var checkinput = "el_" + id;

  var ctrl = document.getElementById( checkinput );
  var img = document.getElementById( checkimg );
  var im = null;
    
  if ( ctrl.value == '0' )
  {
    im = ( onimg == null )? '/music/images/decochken.gif': onimg;
    ctrl.value = '1';
  }
  else
  {
    im = ( offimg == null )? '/music/images/decochkdis.gif': offimg;
    ctrl.value = '0';
  }
  img.src = im;
}

var lastPopup = null;

function winopen( url, p1, p2 ) 
{
  var popwin = window.open( url, p1, p2 );
        
  if( typeof(popwin) != "undefined" && popwin )
  {
    popwin.focus(); 
  }

  lastPopup = popwin; 
}

function moveitem( dname, sdname, imgname, trans, dur )
{
  var el = document.getElementById(dname);
  var xoff = parseInt(el.style.left||el.offsetLeft);
  var from = 0, to = 0;
  if ( xoff < 0 )
  {
    from = xoff;
    to = 0;
  }
  else
  {    
    var elc = document.getElementById(sdname);
    from = 0;
    to = -elc.offsetWidth;
  }
  var durate = dur || 500;
  new Animator({onComplete:function()
  {
    Element.toggleClassName(document.getElementById(imgname), 'larrow');
  }, duration:durate, interval:20,transition:trans}).addSubject(new NumericalStyleSubject(el, 'left', from, to )).play();
}


function testfr(url)
{
  if ( typeof window.parent != "undefined" )
  {
    var baseel = window.parent.document.getElementById("subdiv");
    if ( baseel === null )
    {
      window.location.href = url;
    }
  }
}

function closedivs(vec)
{
  for ( var i=0,j=vec.length;i<j; i++)
  {
    document.getElementById(vec[i]).style.display="none";
  }
}

function addLoadEvent(func) {
  var oldonload = window.onload;
  if (typeof window.onload != 'function') {
    window.onload = func;
  } else {
    window.onload = function() {
      oldonload();
      func();
    }
  }
}

function addUnLoadEvent(func) {
    var oldonunload = window.onunload;
    if (typeof window.onunload != 'function') {
              window.onunload = func;
              } else {
              window.onunload = function() {
                      oldonunload();
                          func();
                            }
              }
}

/**
 * Dialog
 *
 * @author      Roland Franssen <franssen.roland@gmail.com>
 * @license     MIT
 * @version     2.1
 **/

var Dialogs = {
	Lang:{
		close:   '&nbsp;&times;&nbsp;',
		prev:    '&laquo; Previous',
		next:    'Next &raquo;',
		loading: 'Loading...',
		ok:      'OK',
		yes:     'Yes',
		no:      'No'
	},
	Default:{
		handle:         null,                    // css rule | element | null
		autoOpen:       false,                   // true | false
		background:     ['#000', '#fff'],        // array
		width:          'auto',                  // auto | max | integer
		height:         'auto',                  // auto | max | integer
		minWidth:       null,                    // null | pixel value
		minHeight:      null,                    // null | pixel value
		innerScroll:    true,                    // true | false
		opacity:        .75,                     // float | false
		margin:         10,                      // integer
		padding:        10,                      // integer
		title:          null,                    // string | null
		className:      null,                    // string | null
		content:        null,                    // string | element | array | object | function
		iframe:         null,                    // string | null
		target:{
		  id:           null,                    // string | null
		  auto:         true                     // true | false
		},
		ajax:{
		  url:          null,                    // string | null
		  jsonTemplate: null,                    // interpolation template string | null
		  options:      {}                       // default ajax options
		},
		close:{
		  link:         true,                    // true | false
		  esc:          true,                    // true | false
		  overlay:      true                     // true | false
		},
		afterOpen:      Prototype.emptyFunction, // function
		afterClose:     Prototype.emptyFunction, // function
		afterClick:     Prototype.emptyFunction, // function
		afterIframeLoad:Prototype.emptyFunction  // function
	},
	Browser:{
		IE6:(Prototype.Browser.IE && parseInt(navigator.appVersion) == 4 && navigator.userAgent.toLowerCase().indexOf('msie 6.') != -1)
	}
};

Object.extend(Dialogs, {
	_exec:false,
	_open:false,
	_elements:{
		overlay:['div', 'dialog-overlay', 'fixed'],
		container:['div', 'dialog-container', 'fixed'],
		content:['div', 'dialog-content'],
		loading:['div', 'dialog-loading'],
		top:['div', 'dialog-top'],
		bottom:['div', 'dialog-bottom'],
		title:['span', 'dialog-title'],
		close:['a', 'dialog-close'],
		next:['a', null, 'next'],
		prev:['a', null, 'prev'],
		curr:['span', null, 'curr']
	},
	fix:{
		scroll:Dialogs.Browser.IE6,
		select:Dialogs.Browser.IE6
	},
	view:function(){
		var view = document.viewport,
		    dim  = view.getDimensions(),
			data = {width:dim.width, height:dim.height};
		if(Dialogs.fix.scroll){
			var scroll = view.getScrollOffsets();
			data.top  = scroll.top;
			data.left = scroll.left;
		}
		return data;
	},
	elm:function(elm){
		return Dialogs._elements[elm];
	},
	load:function(domready){
		if(!!Dialogs._exec) return;
		Dialogs._exec = true;
		var e = Dialogs._elements;
		for(var x in e){
			var d = e[x],
			    a = {style:'display:none'};
			if(d[1]) a['id'] = d[1];
			if(d[2]) a['className'] = d[2];
			switch(d[0]){
				case 'a': a['href'] = 'javascript:;'; break;
			}
			var el = new Element(d[0], a);
			if(Dialogs.Lang[x]) el.update(Dialogs.Lang[x]);
			Dialogs._elements[x] = el;
		}
		var load = function(){
			var e = Dialogs._elements;
            var db = document.getElementsByTagName('body')[0];

			// $(document.body)
            $(db)
			.insert(e['overlay'])
			.insert(e['container']
				.insert(e['top']
					.insert(e['title'])
					.insert(e['close'])
				)
				.insert(e['content'])
				.insert(e['bottom']
					.insert(e['prev'])
					.insert(e['curr'])
					.insert(e['next'])
				)
			);
			if(Dialogs.Browser.IE6) e['top'].insert(new Element('div', {style:'clear:both'}));
		};
		if(!!domready) document.observe('dom:loaded', load);
		else load.call();
	},
	close:function(){
		[Dialogs.elm('title'), Dialogs.elm('content'), Dialogs.elm('curr')].invoke('update', '');
		for(var x in Dialogs._elements) Dialogs._elements[x].writeAttribute('style', 'display:none');
		Dialogs.elm('container').setStyle('left:-99999px;top:-99999px');
		if(Dialogs.fix.select)
			$$('select.dialog-hideselect').invoke('show').invoke('removeClassName', 'dialog-hideselect');
		Dialogs._open = false;
	},
	alert:function(s){
		var o = new Element('input', {value:Dialogs.Lang.ok, type:'button'}),
		    a = new Dialog({
				className:'alert',
				close:{link:false, esc:true},
				padding:20,
				content:function(){
					o.observe('click', Dialogs.close);
					return [s, '<br /><br />', o];
				},
				afterOpen:function(){
					o.focus();
				}
			});
		a.open();
	},
	confirm:function(s, y_call, n_call){
		var y = new Element('input', {value:Dialogs.Lang.yes, type:'button'}),
		    n = new Element('input', {value:Dialogs.Lang.no, type:'button'}),
		    c = new Dialog({
				className:'confirm',
				close:{link:false},
				padding:20,
				content:function(){
					y.observe('click', function(){
						if(Object.isFunction(y_call)) y_call();
						else Dialogs.close();
					});
					n.observe('click', function(){
						if(Object.isFunction(n_call)) n_call();
						else Dialogs.close();
					});
					return [s, '<br /><br />', y, n];
				},
				afterOpen:function(){
					y.focus();
				}
			});
		c.open();
	}
});
var Dialog = Class.create();
Dialog.prototype = {
	initialize:function(opt){
		this.opt = Object.extend(Object.clone(Dialogs.Default), opt || {});
		var c = this.opt.content;
		if(Object.isFunction(c))
			Object.extend(this.opt, {content:c()});
		c = this.opt.content;
		if(Object.isString(this.opt.target.id) || Object.isElement(this.opt.target.id)){
			var b = $(this.opt.target.id);
			Object.extend(this.opt, {content:b.innerHTML});
			if(this.opt.target.auto){
				var a = /#(.+)$/.exec(window.location);
				if(Object.isArray(a) && Object.isString(a[1])){
					a = a[1].split(',').last();
					if(a == b.identify()) this.open.bind(this).delay(1);
				}
			}
		}else if(Object.isHash(c))
			this.steps = {
				i:0,
				k:c.keys(),
				v:c.values(),
				m:c.size()
			};
		this.attachEvents();
		if(this.opt.autoOpen) this.open();
	},
	exec:function(bool){
		return Dialogs._open == this._open && Dialogs.elm('overlay').visible() && bool;
	},
	attachEvents:function(){
		Event.observe(window, 'resize', this.setDimensions.bindAsEventListener(this));
		if(Dialogs.fix.scroll)
			Event.observe(window, 'scroll', this.setScroll.bindAsEventListener(this));
		var handles = [];
		if(Object.isElement(this.opt.handle)) handles.push($(this.opt.handle));
		else if(Object.isArray(this.opt.handle)) this.opt.handle.each(function(handle){ handles.push($(handle)); });
		else if(Object.isString(this.opt.handle)) handles = $$(this.opt.handle);
		handles.invoke('show').invoke('observe', 'click', function(e){
			e.stop();
			if(Object.isFunction(this.opt.afterClick)) this.opt.afterClick(e);
			this.open();
		}.bindAsEventListener(this));
		Dialogs.elm('close').observe('click', function(){
			if(this.exec(this.opt.close.link)) this.close();
		}.bindAsEventListener(this));
		Dialogs.elm('overlay').observe('click', function(){
			if(this.exec(this.opt.close.overlay)) this.close();
		}.bindAsEventListener(this));
		document.observe('keyup', function(e){
			if(this.exec(this.opt.close.esc && (e.which || e.keyCode) == Event.KEY_ESC)) this.close();
		}.bindAsEventListener(this));
		if(this.steps){
			[Dialogs.elm('prev'), Dialogs.elm('next')].invoke('observe', 'click', this.setSteps.bindAsEventListener(this));
			document.observe('keydown', function(e){
				var c = e.which || e.keyCode;
				if(this.exec((c == Event.KEY_LEFT) || (c == Event.KEY_RIGHT))) this.setSteps(e);
			}.bindAsEventListener(this));
		}
	},
	setAuto:function(){
		this.auto = {max:0};
		var t = Dialogs.elm('title'), c = Dialogs.elm('close');
		[t,c].invoke('setStyle', 'float:none');
		$w('top content bottom').each(function(b){
			var e = Dialogs.elm(b);
			if(!e.visible()) this.auto[b] = {width:0,height:0};
			else{
				e.writeAttribute('style', 'display:inline;float:left;overflow:visible;white-space:nowrap');
				this.auto[b] = e.getDimensions();
				e.writeAttribute('style', 'overflow:hidden');
				if(b == 'content') this.auto[b].width += (parseInt(this.opt.padding) || 0) * 2;
				if(this.auto[b].width > this.auto.max) this.auto.max = this.auto[b].width;
			}
		}.bind(this));
		t.setStyle('float:left');
		c.setStyle('float:right');
	},
	setDimensions:function(){
		if(!this.exec(true)) return;
		this.setAuto();
		var a = this.auto,
		    d = Dialogs.view(),
		    t = Dialogs.elm('content'),
			c = Dialogs.elm('container'),
		    o = {
			  m:((parseInt(this.opt.margin) || 0) * 2),
			  p:((parseInt(this.opt.padding) || 0) * 2),
			  t:a.top.height,
			  b:a.bottom.height
			},
		    m = {width:(d.width-o.m), height:(d.height-o.m-o.t-o.b)},
		    h = this.opt.height,
			w = this.opt.width,
		    x = y = false;
		if(Object.isNumber(w)) w += o.p;
		if(w == 'max') w = m.width;
		if(!Object.isNumber(w)) w = a.max;
		if(w < (this.opt.minWidth || 0)) w = this.opt.minWidth || 0;
		if(w > m.width){ w = m.width; x = true }
		t.setStyle('width:'+(w-o.p)+'px;height:auto');
		if(Object.isNumber(h)) h += o.p;
		if(h == 'max') h = m.height;
		if(!Object.isNumber(h)) h = t.getHeight()+o.p;
		if(h < (this.opt.minHeight || 0)) w = this.opt.minHeight || 0;
		if(h > m.height){ h = m.height; y = true; } 
		t.setStyle('height:'+(h-o.p)+'px;padding:'+(o.p/2)+'px');
		if(this.opt.innerScroll && (x || y)) t.setStyle('overflow:scroll');
		var s = {w:w,h:(h+o.t+o.b)};
		c.setStyle('width:'+s.w+'px;height:'+s.h+'px;top:50%;left:50%;margin:-'+parseInt(s.h/2)+'px 0 0 -'+parseInt(s.w/2)+'px');
		if(Dialogs.fix.scroll){
			Dialogs.elm('overlay').setStyle('width:'+d.width+'px;height:'+d.height+'px');
			this.setScroll();
		}
	},
	setScroll:function(){
		if(!this.exec(true)) return;
		var v = Dialogs.view(),
			c = Dialogs.elm('container'),
			d = c.getDimensions(),
			t = v.top + parseInt((v.height - d.height) / 2),
			l = v.left + parseInt((v.width - d.width) / 2);
		c.setStyle('margin:0;top:'+t+'px;left:'+l+'px');
		Dialogs.elm('overlay').setStyle('margin:'+v.top+'px 0 0 '+v.left+'px');
	},
	setLoad:function(){
		var l = Dialogs.elm('loading').show(),
		    t = Dialogs.elm('content'),
		    b = t.down('#'+l.identify());
		if(!Object.isElement(b)) t.insert(l);
	},
	setAjax:function(){
		this.setLoad();
		var o = this.opt.ajax.options || {},
		    c = (o.onComplete && Object.isFunction(o.onComplete) ? o.onComplete : null),
		    a = function(t){
				var tpl = this.opt.ajax.jsonTemplate;
				if(t.responseJSON && Object.isString(tpl)) Dialogs.elm('content').update(tpl.interpolate(t.responseJSON));
				else Dialogs.elm('content').update(t.responseText || '');
				this.setImages();
				this.setDimensions();
				if(Object.isFunction(c)) c(t);
			}.bind(this);
		Object.extend(o, {onComplete:a});
		new Ajax.Request(this.opt.ajax.url, o);
	},
	setIframe:function(){
		this.setLoad();
		var f = new Element('iframe', {src:this.opt.iframe, frameborder:0, id:'dialog-iframe'});
		Dialogs.elm('content').insert(f);
		f.observe('load', function(){
			Dialogs.elm('loading').hide();
			f.setStyle('width:100%;height:100%');
			this.setDimensions();
			if(Object.isFunction(this.opt.afterIframeLoad)) this.opt.afterIframeLoad();
		}.bindAsEventListener(this));
	},
	setSteps:function(ev){
		if(!this.exec(true)) return;
		var m = this.steps.m,
		    s = false,
			n = Dialogs.elm('next'),
			p = Dialogs.elm('prev');
		if((ev.which || ev.keyCode) == Event.KEY_RIGHT || ev.element().hasClassName('next')){
			if(this.steps.i < (m - 1)) s = true;
			if(s) ++this.steps.i;
			if(((this.steps.i + 1) >= m) && n.visible()) n.hide();
			if(((this.steps.i - 1) >= 0) && !p.visible()) p.show();
		}else{
			if(this.steps.i > 0) s = true;
			if(s) --this.steps.i;
			if(((this.steps.i - 1) < 0) && p.visible()) p.hide();
			if(((this.steps.i + 1) <= m) && !n.visible()) n.show();
		}
		if(s) this.setContent();
	},
	setContent:function(){
		var c = this.opt.content,
		    t = Dialogs.elm('content');
		t.update('');
		if(Object.isString(c) || Object.isElement(c)) t.insert(c);
		else if(Object.isArray(c)) c.each(function(b){ t.insert(b); });
		else if(Object.isHash(c)){
			var b = Dialogs.elm('bottom');
			t.update('').insert(this.steps.v[this.steps.i]);
			Dialogs.elm('curr').update(this.steps.k[this.steps.i]);
			if(!b.visible()) b.show().childElements().invoke('show');
			if(this.steps.i <= 0) Dialogs.elm('prev').hide();
			if(this.steps.i >= (this.steps.m - 1)) Dialogs.elm('next').hide();
		}else if(Object.isString(this.opt.ajax.url)) this.setAjax();
		else if(Object.isString(this.opt.iframe)) this.setIframe();
		this.setImages();
		this.setDimensions.bind(this).defer();
	},
	setImages:function(){
		Dialogs.elm('content').select('img').each(function(el){
			el.onload = function(){
				this.setDimensions();
			}.bind(this);
		}.bind(this));
	},
	open:function(){
		if(Dialogs.fix.select)
			$$('select').select(function(el){ return el.visible(); }).invoke('hide').invoke('addClassName', 'dialog-hideselect');
		if(Object.isString(this.opt.title) || this.opt.close.link){
			if(Object.isString(this.opt.title)) Dialogs.elm('title').show().update(this.opt.title);
			if(this.opt.close.link) Dialogs.elm('close').show();
			else Dialogs.elm('close').hide();
			Dialogs.elm('top').show();
		}else Dialogs.elm('top').hide();
		var o = Dialogs.elm('overlay'), c = Dialogs.elm('container'), t = Dialogs.elm('content');
		[o, c, t].invoke('show');
		o.setOpacity(this.opt.opacity || 1).setStyle({background:this.opt.background[0] || '#000'});
		c.writeAttribute('style', 'left:-99999px;top:-99999px;background:'+(this.opt.background[1] || '#fff'));
		t.writeAttribute('class', this.opt.className || '');
		Dialogs._open = new Date().getTime();
		this._open = Dialogs._open;
		this.setContent();
		if(Object.isFunction(this.opt.afterOpen)) this.opt.afterOpen();
	},
	close:function(){
		Dialogs.close();
		if(Object.isFunction(this.opt.afterClose)) this.opt.afterClose();
	}
};

function modalyesno( t, r )
{
  Dialogs.load();
  
  Dialogs.confirm(t, function(){

    if ( typeof(r) == "function" )
    {
      Dialogs.close();
      r();
    }
    else
    {
      location.href=r;
    }
  });
}

function modalalert( t )
{
  Dialogs.load();
  Dialogs.alert(t);
}

function modalprogress( t, p )
{
  Dialogs.load();
  var html = [];
  html.push("<div id=\"comm_div\" style=\"border:1px black solid;position:relative;height:auto;\">" );
  html.push("<div id=\"comm_msg_div\" style=\"min-height:36px;\"></div>" );
  html.push("<div id=\"comm_progress_bar_div_wrapper\" style=\"position:relative;display:block;margin-top:10px;\">" );
  html.push("<div id=\"comm_progress_bar_div_inner\" style=\"border:1px solid black;position:relative;width:160px;height:15px\">" );
  html.push("<div id=\"comm_progress_bar_div\" style=\"height:100%;width:0%;background-color:#FF0084\"></div>" );
  html.push("</div>" );
  html.push("</div>" );
  html.push("</div>" );
  
  var conts = html.join('');
  var nd = new Dialog({handle:'#pid', className: 'progress', title:'Upload Status',content: conts, width:400, height:200});
  nd.open();
  document.getElementById('comm_div').changemsg = function(msg,perc,step,params){
    if(msg)
    {
      var el = document.getElementById('comm_msg_div');
      el.innerHTML='<span class="mblktxt">'+msg+'</span>';
      var w=Math.ceil(perc*document.getElementById("comm_progress_bar_div_inner").offsetWidth);
      var cpd = document.getElementById("comm_progress_bar_div");
      var an = new Animator({duration:100,interval:10,transition:Animator.tx.strongEaseIn});
      an.addSubject(new NumericalStyleSubject(cpd,"width",parseInt(cpd.style.width,10),w ));
      an.play();
    }
  };
}

