Mercurial > laserkard
view js-lib/prototype.js @ 85:da4f4f91f8aa laserkard
got three separate Raphael instances
author | Robert McIntyre <rlm@mit.edu> |
---|---|
date | Sun, 25 Jul 2010 22:36:16 -0400 |
parents | 343dc947f999 |
children |
line wrap: on
line source
1 /* Prototype JavaScript framework, version 1.6.12 * (c) 2005-2009 Sam Stephenson3 *4 * Prototype is freely distributable under the terms of an MIT-style license.5 * For details, see the Prototype web site: http://www.prototypejs.org/6 *7 *--------------------------------------------------------------------------*/9 var Prototype = {10 Version: '1.6.1',12 Browser: (function(){13 var ua = navigator.userAgent;14 var isOpera = Object.prototype.toString.call(window.opera) == '[object Opera]';15 return {16 IE: !!window.attachEvent && !isOpera,17 Opera: isOpera,18 WebKit: ua.indexOf('AppleWebKit/') > -1,19 Gecko: ua.indexOf('Gecko') > -1 && ua.indexOf('KHTML') === -1,20 MobileSafari: /Apple.*Mobile.*Safari/.test(ua)21 }22 })(),24 BrowserFeatures: {25 XPath: !!document.evaluate,26 SelectorsAPI: !!document.querySelector,27 ElementExtensions: (function() {28 var constructor = window.Element || window.HTMLElement;29 return !!(constructor && constructor.prototype);30 })(),31 SpecificElementExtensions: (function() {32 if (typeof window.HTMLDivElement !== 'undefined')33 return true;35 var div = document.createElement('div');36 var form = document.createElement('form');37 var isSupported = false;39 if (div['__proto__'] && (div['__proto__'] !== form['__proto__'])) {40 isSupported = true;41 }43 div = form = null;45 return isSupported;46 })()47 },49 ScriptFragment: '<script[^>]*>([\\S\\s]*?)<\/script>',50 JSONFilter: /^\/\*-secure-([\s\S]*)\*\/\s*$/,52 emptyFunction: function() { },53 K: function(x) { return x }54 };56 if (Prototype.Browser.MobileSafari)57 Prototype.BrowserFeatures.SpecificElementExtensions = false;60 var Abstract = { };63 var Try = {64 these: function() {65 var returnValue;67 for (var i = 0, length = arguments.length; i < length; i++) {68 var lambda = arguments[i];69 try {70 returnValue = lambda();71 break;72 } catch (e) { }73 }75 return returnValue;76 }77 };79 /* Based on Alex Arnell's inheritance implementation. */81 var Class = (function() {82 function subclass() {};83 function create() {84 var parent = null, properties = $A(arguments);85 if (Object.isFunction(properties[0]))86 parent = properties.shift();88 function klass() {89 this.initialize.apply(this, arguments);90 }92 Object.extend(klass, Class.Methods);93 klass.superclass = parent;94 klass.subclasses = [];96 if (parent) {97 subclass.prototype = parent.prototype;98 klass.prototype = new subclass;99 parent.subclasses.push(klass);100 }102 for (var i = 0; i < properties.length; i++)103 klass.addMethods(properties[i]);105 if (!klass.prototype.initialize)106 klass.prototype.initialize = Prototype.emptyFunction;108 klass.prototype.constructor = klass;109 return klass;110 }112 function addMethods(source) {113 var ancestor = this.superclass && this.superclass.prototype;114 var properties = Object.keys(source);116 if (!Object.keys({ toString: true }).length) {117 if (source.toString != Object.prototype.toString)118 properties.push("toString");119 if (source.valueOf != Object.prototype.valueOf)120 properties.push("valueOf");121 }123 for (var i = 0, length = properties.length; i < length; i++) {124 var property = properties[i], value = source[property];125 if (ancestor && Object.isFunction(value) &&126 value.argumentNames().first() == "$super") {127 var method = value;128 value = (function(m) {129 return function() { return ancestor[m].apply(this, arguments); };130 })(property).wrap(method);132 value.valueOf = method.valueOf.bind(method);133 value.toString = method.toString.bind(method);134 }135 this.prototype[property] = value;136 }138 return this;139 }141 return {142 create: create,143 Methods: {144 addMethods: addMethods145 }146 };147 })();148 (function() {150 var _toString = Object.prototype.toString;152 function extend(destination, source) {153 for (var property in source)154 destination[property] = source[property];155 return destination;156 }158 function inspect(object) {159 try {160 if (isUndefined(object)) return 'undefined';161 if (object === null) return 'null';162 return object.inspect ? object.inspect() : String(object);163 } catch (e) {164 if (e instanceof RangeError) return '...';165 throw e;166 }167 }169 function toJSON(object) {170 var type = typeof object;171 switch (type) {172 case 'undefined':173 case 'function':174 case 'unknown': return;175 case 'boolean': return object.toString();176 }178 if (object === null) return 'null';179 if (object.toJSON) return object.toJSON();180 if (isElement(object)) return;182 var results = [];183 for (var property in object) {184 var value = toJSON(object[property]);185 if (!isUndefined(value))186 results.push(property.toJSON() + ': ' + value);187 }189 return '{' + results.join(', ') + '}';190 }192 function toQueryString(object) {193 return $H(object).toQueryString();194 }196 function toHTML(object) {197 return object && object.toHTML ? object.toHTML() : String.interpret(object);198 }200 function keys(object) {201 var results = [];202 for (var property in object)203 results.push(property);204 return results;205 }207 function values(object) {208 var results = [];209 for (var property in object)210 results.push(object[property]);211 return results;212 }214 function clone(object) {215 return extend({ }, object);216 }218 function isElement(object) {219 return !!(object && object.nodeType == 1);220 }222 function isArray(object) {223 return _toString.call(object) == "[object Array]";224 }227 function isHash(object) {228 return object instanceof Hash;229 }231 function isFunction(object) {232 return typeof object === "function";233 }235 function isString(object) {236 return _toString.call(object) == "[object String]";237 }239 function isNumber(object) {240 return _toString.call(object) == "[object Number]";241 }243 function isUndefined(object) {244 return typeof object === "undefined";245 }247 extend(Object, {248 extend: extend,249 inspect: inspect,250 toJSON: toJSON,251 toQueryString: toQueryString,252 toHTML: toHTML,253 keys: keys,254 values: values,255 clone: clone,256 isElement: isElement,257 isArray: isArray,258 isHash: isHash,259 isFunction: isFunction,260 isString: isString,261 isNumber: isNumber,262 isUndefined: isUndefined263 });264 })();265 Object.extend(Function.prototype, (function() {266 var slice = Array.prototype.slice;268 function update(array, args) {269 var arrayLength = array.length, length = args.length;270 while (length--) array[arrayLength + length] = args[length];271 return array;272 }274 function merge(array, args) {275 array = slice.call(array, 0);276 return update(array, args);277 }279 function argumentNames() {280 var names = this.toString().match(/^[\s\(]*function[^(]*\(([^)]*)\)/)[1]281 .replace(/\/\/.*?[\r\n]|\/\*(?:.|[\r\n])*?\*\//g, '')282 .replace(/\s+/g, '').split(',');283 return names.length == 1 && !names[0] ? [] : names;284 }286 function bind(context) {287 if (arguments.length < 2 && Object.isUndefined(arguments[0])) return this;288 var __method = this, args = slice.call(arguments, 1);289 return function() {290 var a = merge(args, arguments);291 return __method.apply(context, a);292 }293 }295 function bindAsEventListener(context) {296 var __method = this, args = slice.call(arguments, 1);297 return function(event) {298 var a = update([event || window.event], args);299 return __method.apply(context, a);300 }301 }303 function curry() {304 if (!arguments.length) return this;305 var __method = this, args = slice.call(arguments, 0);306 return function() {307 var a = merge(args, arguments);308 return __method.apply(this, a);309 }310 }312 function delay(timeout) {313 var __method = this, args = slice.call(arguments, 1);314 timeout = timeout * 1000315 return window.setTimeout(function() {316 return __method.apply(__method, args);317 }, timeout);318 }320 function defer() {321 var args = update([0.01], arguments);322 return this.delay.apply(this, args);323 }325 function wrap(wrapper) {326 var __method = this;327 return function() {328 var a = update([__method.bind(this)], arguments);329 return wrapper.apply(this, a);330 }331 }333 function methodize() {334 if (this._methodized) return this._methodized;335 var __method = this;336 return this._methodized = function() {337 var a = update([this], arguments);338 return __method.apply(null, a);339 };340 }342 return {343 argumentNames: argumentNames,344 bind: bind,345 bindAsEventListener: bindAsEventListener,346 curry: curry,347 delay: delay,348 defer: defer,349 wrap: wrap,350 methodize: methodize351 }352 })());355 Date.prototype.toJSON = function() {356 return '"' + this.getUTCFullYear() + '-' +357 (this.getUTCMonth() + 1).toPaddedString(2) + '-' +358 this.getUTCDate().toPaddedString(2) + 'T' +359 this.getUTCHours().toPaddedString(2) + ':' +360 this.getUTCMinutes().toPaddedString(2) + ':' +361 this.getUTCSeconds().toPaddedString(2) + 'Z"';362 };365 RegExp.prototype.match = RegExp.prototype.test;367 RegExp.escape = function(str) {368 return String(str).replace(/([.*+?^=!:${}()|[\]\/\\])/g, '\\$1');369 };370 var PeriodicalExecuter = Class.create({371 initialize: function(callback, frequency) {372 this.callback = callback;373 this.frequency = frequency;374 this.currentlyExecuting = false;376 this.registerCallback();377 },379 registerCallback: function() {380 this.timer = setInterval(this.onTimerEvent.bind(this), this.frequency * 1000);381 },383 execute: function() {384 this.callback(this);385 },387 stop: function() {388 if (!this.timer) return;389 clearInterval(this.timer);390 this.timer = null;391 },393 onTimerEvent: function() {394 if (!this.currentlyExecuting) {395 try {396 this.currentlyExecuting = true;397 this.execute();398 this.currentlyExecuting = false;399 } catch(e) {400 this.currentlyExecuting = false;401 throw e;402 }403 }404 }405 });406 Object.extend(String, {407 interpret: function(value) {408 return value == null ? '' : String(value);409 },410 specialChar: {411 '\b': '\\b',412 '\t': '\\t',413 '\n': '\\n',414 '\f': '\\f',415 '\r': '\\r',416 '\\': '\\\\'417 }418 });420 Object.extend(String.prototype, (function() {422 function prepareReplacement(replacement) {423 if (Object.isFunction(replacement)) return replacement;424 var template = new Template(replacement);425 return function(match) { return template.evaluate(match) };426 }428 function gsub(pattern, replacement) {429 var result = '', source = this, match;430 replacement = prepareReplacement(replacement);432 if (Object.isString(pattern))433 pattern = RegExp.escape(pattern);435 if (!(pattern.length || pattern.source)) {436 replacement = replacement('');437 return replacement + source.split('').join(replacement) + replacement;438 }440 while (source.length > 0) {441 if (match = source.match(pattern)) {442 result += source.slice(0, match.index);443 result += String.interpret(replacement(match));444 source = source.slice(match.index + match[0].length);445 } else {446 result += source, source = '';447 }448 }449 return result;450 }452 function sub(pattern, replacement, count) {453 replacement = prepareReplacement(replacement);454 count = Object.isUndefined(count) ? 1 : count;456 return this.gsub(pattern, function(match) {457 if (--count < 0) return match[0];458 return replacement(match);459 });460 }462 function scan(pattern, iterator) {463 this.gsub(pattern, iterator);464 return String(this);465 }467 function truncate(length, truncation) {468 length = length || 30;469 truncation = Object.isUndefined(truncation) ? '...' : truncation;470 return this.length > length ?471 this.slice(0, length - truncation.length) + truncation : String(this);472 }474 function strip() {475 return this.replace(/^\s+/, '').replace(/\s+$/, '');476 }478 function stripTags() {479 return this.replace(/<\w+(\s+("[^"]*"|'[^']*'|[^>])+)?>|<\/\w+>/gi, '');480 }482 function stripScripts() {483 return this.replace(new RegExp(Prototype.ScriptFragment, 'img'), '');484 }486 function extractScripts() {487 var matchAll = new RegExp(Prototype.ScriptFragment, 'img');488 var matchOne = new RegExp(Prototype.ScriptFragment, 'im');489 return (this.match(matchAll) || []).map(function(scriptTag) {490 return (scriptTag.match(matchOne) || ['', ''])[1];491 });492 }494 function evalScripts() {495 return this.extractScripts().map(function(script) { return eval(script) });496 }498 function escapeHTML() {499 return this.replace(/&/g,'&').replace(/</g,'<').replace(/>/g,'>');500 }502 function unescapeHTML() {503 return this.stripTags().replace(/</g,'<').replace(/>/g,'>').replace(/&/g,'&');504 }507 function toQueryParams(separator) {508 var match = this.strip().match(/([^?#]*)(#.*)?$/);509 if (!match) return { };511 return match[1].split(separator || '&').inject({ }, function(hash, pair) {512 if ((pair = pair.split('='))[0]) {513 var key = decodeURIComponent(pair.shift());514 var value = pair.length > 1 ? pair.join('=') : pair[0];515 if (value != undefined) value = decodeURIComponent(value);517 if (key in hash) {518 if (!Object.isArray(hash[key])) hash[key] = [hash[key]];519 hash[key].push(value);520 }521 else hash[key] = value;522 }523 return hash;524 });525 }527 function toArray() {528 return this.split('');529 }531 function succ() {532 return this.slice(0, this.length - 1) +533 String.fromCharCode(this.charCodeAt(this.length - 1) + 1);534 }536 function times(count) {537 return count < 1 ? '' : new Array(count + 1).join(this);538 }540 function camelize() {541 var parts = this.split('-'), len = parts.length;542 if (len == 1) return parts[0];544 var camelized = this.charAt(0) == '-'545 ? parts[0].charAt(0).toUpperCase() + parts[0].substring(1)546 : parts[0];548 for (var i = 1; i < len; i++)549 camelized += parts[i].charAt(0).toUpperCase() + parts[i].substring(1);551 return camelized;552 }554 function capitalize() {555 return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase();556 }558 function underscore() {559 return this.replace(/::/g, '/')560 .replace(/([A-Z]+)([A-Z][a-z])/g, '$1_$2')561 .replace(/([a-z\d])([A-Z])/g, '$1_$2')562 .replace(/-/g, '_')563 .toLowerCase();564 }566 function dasherize() {567 return this.replace(/_/g, '-');568 }570 function inspect(useDoubleQuotes) {571 var escapedString = this.replace(/[\x00-\x1f\\]/g, function(character) {572 if (character in String.specialChar) {573 return String.specialChar[character];574 }575 return '\\u00' + character.charCodeAt().toPaddedString(2, 16);576 });577 if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"';578 return "'" + escapedString.replace(/'/g, '\\\'') + "'";579 }581 function toJSON() {582 return this.inspect(true);583 }585 function unfilterJSON(filter) {586 return this.replace(filter || Prototype.JSONFilter, '$1');587 }589 function isJSON() {590 var str = this;591 if (str.blank()) return false;592 str = this.replace(/\\./g, '@').replace(/"[^"\\\n\r]*"/g, '');593 return (/^[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t]*$/).test(str);594 }596 function evalJSON(sanitize) {597 var json = this.unfilterJSON();598 try {599 if (!sanitize || json.isJSON()) return eval('(' + json + ')');600 } catch (e) { }601 throw new SyntaxError('Badly formed JSON string: ' + this.inspect());602 }604 function include(pattern) {605 return this.indexOf(pattern) > -1;606 }608 function startsWith(pattern) {609 return this.indexOf(pattern) === 0;610 }612 function endsWith(pattern) {613 var d = this.length - pattern.length;614 return d >= 0 && this.lastIndexOf(pattern) === d;615 }617 function empty() {618 return this == '';619 }621 function blank() {622 return /^\s*$/.test(this);623 }625 function interpolate(object, pattern) {626 return new Template(this, pattern).evaluate(object);627 }629 return {630 gsub: gsub,631 sub: sub,632 scan: scan,633 truncate: truncate,634 strip: String.prototype.trim ? String.prototype.trim : strip,635 stripTags: stripTags,636 stripScripts: stripScripts,637 extractScripts: extractScripts,638 evalScripts: evalScripts,639 escapeHTML: escapeHTML,640 unescapeHTML: unescapeHTML,641 toQueryParams: toQueryParams,642 parseQuery: toQueryParams,643 toArray: toArray,644 succ: succ,645 times: times,646 camelize: camelize,647 capitalize: capitalize,648 underscore: underscore,649 dasherize: dasherize,650 inspect: inspect,651 toJSON: toJSON,652 unfilterJSON: unfilterJSON,653 isJSON: isJSON,654 evalJSON: evalJSON,655 include: include,656 startsWith: startsWith,657 endsWith: endsWith,658 empty: empty,659 blank: blank,660 interpolate: interpolate661 };662 })());664 var Template = Class.create({665 initialize: function(template, pattern) {666 this.template = template.toString();667 this.pattern = pattern || Template.Pattern;668 },670 evaluate: function(object) {671 if (object && Object.isFunction(object.toTemplateReplacements))672 object = object.toTemplateReplacements();674 return this.template.gsub(this.pattern, function(match) {675 if (object == null) return (match[1] + '');677 var before = match[1] || '';678 if (before == '\\') return match[2];680 var ctx = object, expr = match[3];681 var pattern = /^([^.[]+|\[((?:.*?[^\\])?)\])(\.|\[|$)/;682 match = pattern.exec(expr);683 if (match == null) return before;685 while (match != null) {686 var comp = match[1].startsWith('[') ? match[2].replace(/\\\\]/g, ']') : match[1];687 ctx = ctx[comp];688 if (null == ctx || '' == match[3]) break;689 expr = expr.substring('[' == match[3] ? match[1].length : match[0].length);690 match = pattern.exec(expr);691 }693 return before + String.interpret(ctx);694 });695 }696 });697 Template.Pattern = /(^|.|\r|\n)(#\{(.*?)\})/;699 var $break = { };701 var Enumerable = (function() {702 function each(iterator, context) {703 var index = 0;704 try {705 this._each(function(value) {706 iterator.call(context, value, index++);707 });708 } catch (e) {709 if (e != $break) throw e;710 }711 return this;712 }714 function eachSlice(number, iterator, context) {715 var index = -number, slices = [], array = this.toArray();716 if (number < 1) return array;717 while ((index += number) < array.length)718 slices.push(array.slice(index, index+number));719 return slices.collect(iterator, context);720 }722 function all(iterator, context) {723 iterator = iterator || Prototype.K;724 var result = true;725 this.each(function(value, index) {726 result = result && !!iterator.call(context, value, index);727 if (!result) throw $break;728 });729 return result;730 }732 function any(iterator, context) {733 iterator = iterator || Prototype.K;734 var result = false;735 this.each(function(value, index) {736 if (result = !!iterator.call(context, value, index))737 throw $break;738 });739 return result;740 }742 function collect(iterator, context) {743 iterator = iterator || Prototype.K;744 var results = [];745 this.each(function(value, index) {746 results.push(iterator.call(context, value, index));747 });748 return results;749 }751 function detect(iterator, context) {752 var result;753 this.each(function(value, index) {754 if (iterator.call(context, value, index)) {755 result = value;756 throw $break;757 }758 });759 return result;760 }762 function findAll(iterator, context) {763 var results = [];764 this.each(function(value, index) {765 if (iterator.call(context, value, index))766 results.push(value);767 });768 return results;769 }771 function grep(filter, iterator, context) {772 iterator = iterator || Prototype.K;773 var results = [];775 if (Object.isString(filter))776 filter = new RegExp(RegExp.escape(filter));778 this.each(function(value, index) {779 if (filter.match(value))780 results.push(iterator.call(context, value, index));781 });782 return results;783 }785 function include(object) {786 if (Object.isFunction(this.indexOf))787 if (this.indexOf(object) != -1) return true;789 var found = false;790 this.each(function(value) {791 if (value == object) {792 found = true;793 throw $break;794 }795 });796 return found;797 }799 function inGroupsOf(number, fillWith) {800 fillWith = Object.isUndefined(fillWith) ? null : fillWith;801 return this.eachSlice(number, function(slice) {802 while(slice.length < number) slice.push(fillWith);803 return slice;804 });805 }807 function inject(memo, iterator, context) {808 this.each(function(value, index) {809 memo = iterator.call(context, memo, value, index);810 });811 return memo;812 }814 function invoke(method) {815 var args = $A(arguments).slice(1);816 return this.map(function(value) {817 return value[method].apply(value, args);818 });819 }821 function max(iterator, context) {822 iterator = iterator || Prototype.K;823 var result;824 this.each(function(value, index) {825 value = iterator.call(context, value, index);826 if (result == null || value >= result)827 result = value;828 });829 return result;830 }832 function min(iterator, context) {833 iterator = iterator || Prototype.K;834 var result;835 this.each(function(value, index) {836 value = iterator.call(context, value, index);837 if (result == null || value < result)838 result = value;839 });840 return result;841 }843 function partition(iterator, context) {844 iterator = iterator || Prototype.K;845 var trues = [], falses = [];846 this.each(function(value, index) {847 (iterator.call(context, value, index) ?848 trues : falses).push(value);849 });850 return [trues, falses];851 }853 function pluck(property) {854 var results = [];855 this.each(function(value) {856 results.push(value[property]);857 });858 return results;859 }861 function reject(iterator, context) {862 var results = [];863 this.each(function(value, index) {864 if (!iterator.call(context, value, index))865 results.push(value);866 });867 return results;868 }870 function sortBy(iterator, context) {871 return this.map(function(value, index) {872 return {873 value: value,874 criteria: iterator.call(context, value, index)875 };876 }).sort(function(left, right) {877 var a = left.criteria, b = right.criteria;878 return a < b ? -1 : a > b ? 1 : 0;879 }).pluck('value');880 }882 function toArray() {883 return this.map();884 }886 function zip() {887 var iterator = Prototype.K, args = $A(arguments);888 if (Object.isFunction(args.last()))889 iterator = args.pop();891 var collections = [this].concat(args).map($A);892 return this.map(function(value, index) {893 return iterator(collections.pluck(index));894 });895 }897 function size() {898 return this.toArray().length;899 }901 function inspect() {902 return '#<Enumerable:' + this.toArray().inspect() + '>';903 }913 return {914 each: each,915 eachSlice: eachSlice,916 all: all,917 every: all,918 any: any,919 some: any,920 collect: collect,921 map: collect,922 detect: detect,923 findAll: findAll,924 select: findAll,925 filter: findAll,926 grep: grep,927 include: include,928 member: include,929 inGroupsOf: inGroupsOf,930 inject: inject,931 invoke: invoke,932 max: max,933 min: min,934 partition: partition,935 pluck: pluck,936 reject: reject,937 sortBy: sortBy,938 toArray: toArray,939 entries: toArray,940 zip: zip,941 size: size,942 inspect: inspect,943 find: detect944 };945 })();946 function $A(iterable) {947 if (!iterable) return [];948 if ('toArray' in Object(iterable)) return iterable.toArray();949 var length = iterable.length || 0, results = new Array(length);950 while (length--) results[length] = iterable[length];951 return results;952 }954 function $w(string) {955 if (!Object.isString(string)) return [];956 string = string.strip();957 return string ? string.split(/\s+/) : [];958 }960 Array.from = $A;963 (function() {964 var arrayProto = Array.prototype,965 slice = arrayProto.slice,966 _each = arrayProto.forEach; // use native browser JS 1.6 implementation if available968 function each(iterator) {969 for (var i = 0, length = this.length; i < length; i++)970 iterator(this[i]);971 }972 if (!_each) _each = each;974 function clear() {975 this.length = 0;976 return this;977 }979 function first() {980 return this[0];981 }983 function last() {984 return this[this.length - 1];985 }987 function compact() {988 return this.select(function(value) {989 return value != null;990 });991 }993 function flatten() {994 return this.inject([], function(array, value) {995 if (Object.isArray(value))996 return array.concat(value.flatten());997 array.push(value);998 return array;999 });1000 }1002 function without() {1003 var values = slice.call(arguments, 0);1004 return this.select(function(value) {1005 return !values.include(value);1006 });1007 }1009 function reverse(inline) {1010 return (inline !== false ? this : this.toArray())._reverse();1011 }1013 function uniq(sorted) {1014 return this.inject([], function(array, value, index) {1015 if (0 == index || (sorted ? array.last() != value : !array.include(value)))1016 array.push(value);1017 return array;1018 });1019 }1021 function intersect(array) {1022 return this.uniq().findAll(function(item) {1023 return array.detect(function(value) { return item === value });1024 });1025 }1028 function clone() {1029 return slice.call(this, 0);1030 }1032 function size() {1033 return this.length;1034 }1036 function inspect() {1037 return '[' + this.map(Object.inspect).join(', ') + ']';1038 }1040 function toJSON() {1041 var results = [];1042 this.each(function(object) {1043 var value = Object.toJSON(object);1044 if (!Object.isUndefined(value)) results.push(value);1045 });1046 return '[' + results.join(', ') + ']';1047 }1049 function indexOf(item, i) {1050 i || (i = 0);1051 var length = this.length;1052 if (i < 0) i = length + i;1053 for (; i < length; i++)1054 if (this[i] === item) return i;1055 return -1;1056 }1058 function lastIndexOf(item, i) {1059 i = isNaN(i) ? this.length : (i < 0 ? this.length + i : i) + 1;1060 var n = this.slice(0, i).reverse().indexOf(item);1061 return (n < 0) ? n : i - n - 1;1062 }1064 function concat() {1065 var array = slice.call(this, 0), item;1066 for (var i = 0, length = arguments.length; i < length; i++) {1067 item = arguments[i];1068 if (Object.isArray(item) && !('callee' in item)) {1069 for (var j = 0, arrayLength = item.length; j < arrayLength; j++)1070 array.push(item[j]);1071 } else {1072 array.push(item);1073 }1074 }1075 return array;1076 }1078 Object.extend(arrayProto, Enumerable);1080 if (!arrayProto._reverse)1081 arrayProto._reverse = arrayProto.reverse;1083 Object.extend(arrayProto, {1084 _each: _each,1085 clear: clear,1086 first: first,1087 last: last,1088 compact: compact,1089 flatten: flatten,1090 without: without,1091 reverse: reverse,1092 uniq: uniq,1093 intersect: intersect,1094 clone: clone,1095 toArray: clone,1096 size: size,1097 inspect: inspect,1098 toJSON: toJSON1099 });1101 var CONCAT_ARGUMENTS_BUGGY = (function() {1102 return [].concat(arguments)[0][0] !== 1;1103 })(1,2)1105 if (CONCAT_ARGUMENTS_BUGGY) arrayProto.concat = concat;1107 if (!arrayProto.indexOf) arrayProto.indexOf = indexOf;1108 if (!arrayProto.lastIndexOf) arrayProto.lastIndexOf = lastIndexOf;1109 })();1110 function $H(object) {1111 return new Hash(object);1112 };1114 var Hash = Class.create(Enumerable, (function() {1115 function initialize(object) {1116 this._object = Object.isHash(object) ? object.toObject() : Object.clone(object);1117 }1119 function _each(iterator) {1120 for (var key in this._object) {1121 var value = this._object[key], pair = [key, value];1122 pair.key = key;1123 pair.value = value;1124 iterator(pair);1125 }1126 }1128 function set(key, value) {1129 return this._object[key] = value;1130 }1132 function get(key) {1133 if (this._object[key] !== Object.prototype[key])1134 return this._object[key];1135 }1137 function unset(key) {1138 var value = this._object[key];1139 delete this._object[key];1140 return value;1141 }1143 function toObject() {1144 return Object.clone(this._object);1145 }1147 function keys() {1148 return this.pluck('key');1149 }1151 function values() {1152 return this.pluck('value');1153 }1155 function index(value) {1156 var match = this.detect(function(pair) {1157 return pair.value === value;1158 });1159 return match && match.key;1160 }1162 function merge(object) {1163 return this.clone().update(object);1164 }1166 function update(object) {1167 return new Hash(object).inject(this, function(result, pair) {1168 result.set(pair.key, pair.value);1169 return result;1170 });1171 }1173 function toQueryPair(key, value) {1174 if (Object.isUndefined(value)) return key;1175 return key + '=' + encodeURIComponent(String.interpret(value));1176 }1178 function toQueryString() {1179 return this.inject([], function(results, pair) {1180 var key = encodeURIComponent(pair.key), values = pair.value;1182 if (values && typeof values == 'object') {1183 if (Object.isArray(values))1184 return results.concat(values.map(toQueryPair.curry(key)));1185 } else results.push(toQueryPair(key, values));1186 return results;1187 }).join('&');1188 }1190 function inspect() {1191 return '#<Hash:{' + this.map(function(pair) {1192 return pair.map(Object.inspect).join(': ');1193 }).join(', ') + '}>';1194 }1196 function toJSON() {1197 return Object.toJSON(this.toObject());1198 }1200 function clone() {1201 return new Hash(this);1202 }1204 return {1205 initialize: initialize,1206 _each: _each,1207 set: set,1208 get: get,1209 unset: unset,1210 toObject: toObject,1211 toTemplateReplacements: toObject,1212 keys: keys,1213 values: values,1214 index: index,1215 merge: merge,1216 update: update,1217 toQueryString: toQueryString,1218 inspect: inspect,1219 toJSON: toJSON,1220 clone: clone1221 };1222 })());1224 Hash.from = $H;1225 Object.extend(Number.prototype, (function() {1226 function toColorPart() {1227 return this.toPaddedString(2, 16);1228 }1230 function succ() {1231 return this + 1;1232 }1234 function times(iterator, context) {1235 $R(0, this, true).each(iterator, context);1236 return this;1237 }1239 function toPaddedString(length, radix) {1240 var string = this.toString(radix || 10);1241 return '0'.times(length - string.length) + string;1242 }1244 function toJSON() {1245 return isFinite(this) ? this.toString() : 'null';1246 }1248 function abs() {1249 return Math.abs(this);1250 }1252 function round() {1253 return Math.round(this);1254 }1256 function ceil() {1257 return Math.ceil(this);1258 }1260 function floor() {1261 return Math.floor(this);1262 }1264 return {1265 toColorPart: toColorPart,1266 succ: succ,1267 times: times,1268 toPaddedString: toPaddedString,1269 toJSON: toJSON,1270 abs: abs,1271 round: round,1272 ceil: ceil,1273 floor: floor1274 };1275 })());1277 function $R(start, end, exclusive) {1278 return new ObjectRange(start, end, exclusive);1279 }1281 var ObjectRange = Class.create(Enumerable, (function() {1282 function initialize(start, end, exclusive) {1283 this.start = start;1284 this.end = end;1285 this.exclusive = exclusive;1286 }1288 function _each(iterator) {1289 var value = this.start;1290 while (this.include(value)) {1291 iterator(value);1292 value = value.succ();1293 }1294 }1296 function include(value) {1297 if (value < this.start)1298 return false;1299 if (this.exclusive)1300 return value < this.end;1301 return value <= this.end;1302 }1304 return {1305 initialize: initialize,1306 _each: _each,1307 include: include1308 };1309 })());1313 var Ajax = {1314 getTransport: function() {1315 return Try.these(1316 function() {return new XMLHttpRequest()},1317 function() {return new ActiveXObject('Msxml2.XMLHTTP')},1318 function() {return new ActiveXObject('Microsoft.XMLHTTP')}1319 ) || false;1320 },1322 activeRequestCount: 01323 };1325 Ajax.Responders = {1326 responders: [],1328 _each: function(iterator) {1329 this.responders._each(iterator);1330 },1332 register: function(responder) {1333 if (!this.include(responder))1334 this.responders.push(responder);1335 },1337 unregister: function(responder) {1338 this.responders = this.responders.without(responder);1339 },1341 dispatch: function(callback, request, transport, json) {1342 this.each(function(responder) {1343 if (Object.isFunction(responder[callback])) {1344 try {1345 responder[callback].apply(responder, [request, transport, json]);1346 } catch (e) { }1347 }1348 });1349 }1350 };1352 Object.extend(Ajax.Responders, Enumerable);1354 Ajax.Responders.register({1355 onCreate: function() { Ajax.activeRequestCount++ },1356 onComplete: function() { Ajax.activeRequestCount-- }1357 });1358 Ajax.Base = Class.create({1359 initialize: function(options) {1360 this.options = {1361 method: 'post',1362 asynchronous: true,1363 contentType: 'application/x-www-form-urlencoded',1364 encoding: 'UTF-8',1365 parameters: '',1366 evalJSON: true,1367 evalJS: true1368 };1369 Object.extend(this.options, options || { });1371 this.options.method = this.options.method.toLowerCase();1373 if (Object.isString(this.options.parameters))1374 this.options.parameters = this.options.parameters.toQueryParams();1375 else if (Object.isHash(this.options.parameters))1376 this.options.parameters = this.options.parameters.toObject();1377 }1378 });1379 Ajax.Request = Class.create(Ajax.Base, {1380 _complete: false,1382 initialize: function($super, url, options) {1383 $super(options);1384 this.transport = Ajax.getTransport();1385 this.request(url);1386 },1388 request: function(url) {1389 this.url = url;1390 this.method = this.options.method;1391 var params = Object.clone(this.options.parameters);1393 if (!['get', 'post'].include(this.method)) {1394 params['_method'] = this.method;1395 this.method = 'post';1396 }1398 this.parameters = params;1400 if (params = Object.toQueryString(params)) {1401 if (this.method == 'get')1402 this.url += (this.url.include('?') ? '&' : '?') + params;1403 else if (/Konqueror|Safari|KHTML/.test(navigator.userAgent))1404 params += '&_=';1405 }1407 try {1408 var response = new Ajax.Response(this);1409 if (this.options.onCreate) this.options.onCreate(response);1410 Ajax.Responders.dispatch('onCreate', this, response);1412 this.transport.open(this.method.toUpperCase(), this.url,1413 this.options.asynchronous);1415 if (this.options.asynchronous) this.respondToReadyState.bind(this).defer(1);1417 this.transport.onreadystatechange = this.onStateChange.bind(this);1418 this.setRequestHeaders();1420 this.body = this.method == 'post' ? (this.options.postBody || params) : null;1421 this.transport.send(this.body);1423 /* Force Firefox to handle ready state 4 for synchronous requests */1424 if (!this.options.asynchronous && this.transport.overrideMimeType)1425 this.onStateChange();1427 }1428 catch (e) {1429 this.dispatchException(e);1430 }1431 },1433 onStateChange: function() {1434 var readyState = this.transport.readyState;1435 if (readyState > 1 && !((readyState == 4) && this._complete))1436 this.respondToReadyState(this.transport.readyState);1437 },1439 setRequestHeaders: function() {1440 var headers = {1441 'X-Requested-With': 'XMLHttpRequest',1442 'X-Prototype-Version': Prototype.Version,1443 'Accept': 'text/javascript, text/html, application/xml, text/xml, */*'1444 };1446 if (this.method == 'post') {1447 headers['Content-type'] = this.options.contentType +1448 (this.options.encoding ? '; charset=' + this.options.encoding : '');1450 /* Force "Connection: close" for older Mozilla browsers to work1451 * around a bug where XMLHttpRequest sends an incorrect1452 * Content-length header. See Mozilla Bugzilla #246651.1453 */1454 if (this.transport.overrideMimeType &&1455 (navigator.userAgent.match(/Gecko\/(\d{4})/) || [0,2005])[1] < 2005)1456 headers['Connection'] = 'close';1457 }1459 if (typeof this.options.requestHeaders == 'object') {1460 var extras = this.options.requestHeaders;1462 if (Object.isFunction(extras.push))1463 for (var i = 0, length = extras.length; i < length; i += 2)1464 headers[extras[i]] = extras[i+1];1465 else1466 $H(extras).each(function(pair) { headers[pair.key] = pair.value });1467 }1469 for (var name in headers)1470 this.transport.setRequestHeader(name, headers[name]);1471 },1473 success: function() {1474 var status = this.getStatus();1475 return !status || (status >= 200 && status < 300);1476 },1478 getStatus: function() {1479 try {1480 return this.transport.status || 0;1481 } catch (e) { return 0 }1482 },1484 respondToReadyState: function(readyState) {1485 var state = Ajax.Request.Events[readyState], response = new Ajax.Response(this);1487 if (state == 'Complete') {1488 try {1489 this._complete = true;1490 (this.options['on' + response.status]1491 || this.options['on' + (this.success() ? 'Success' : 'Failure')]1492 || Prototype.emptyFunction)(response, response.headerJSON);1493 } catch (e) {1494 this.dispatchException(e);1495 }1497 var contentType = response.getHeader('Content-type');1498 if (this.options.evalJS == 'force'1499 || (this.options.evalJS && this.isSameOrigin() && contentType1500 && contentType.match(/^\s*(text|application)\/(x-)?(java|ecma)script(;.*)?\s*$/i)))1501 this.evalResponse();1502 }1504 try {1505 (this.options['on' + state] || Prototype.emptyFunction)(response, response.headerJSON);1506 Ajax.Responders.dispatch('on' + state, this, response, response.headerJSON);1507 } catch (e) {1508 this.dispatchException(e);1509 }1511 if (state == 'Complete') {1512 this.transport.onreadystatechange = Prototype.emptyFunction;1513 }1514 },1516 isSameOrigin: function() {1517 var m = this.url.match(/^\s*https?:\/\/[^\/]*/);1518 return !m || (m[0] == '#{protocol}//#{domain}#{port}'.interpolate({1519 protocol: location.protocol,1520 domain: document.domain,1521 port: location.port ? ':' + location.port : ''1522 }));1523 },1525 getHeader: function(name) {1526 try {1527 return this.transport.getResponseHeader(name) || null;1528 } catch (e) { return null; }1529 },1531 evalResponse: function() {1532 try {1533 return eval((this.transport.responseText || '').unfilterJSON());1534 } catch (e) {1535 this.dispatchException(e);1536 }1537 },1539 dispatchException: function(exception) {1540 (this.options.onException || Prototype.emptyFunction)(this, exception);1541 Ajax.Responders.dispatch('onException', this, exception);1542 }1543 });1545 Ajax.Request.Events =1546 ['Uninitialized', 'Loading', 'Loaded', 'Interactive', 'Complete'];1555 Ajax.Response = Class.create({1556 initialize: function(request){1557 this.request = request;1558 var transport = this.transport = request.transport,1559 readyState = this.readyState = transport.readyState;1561 if((readyState > 2 && !Prototype.Browser.IE) || readyState == 4) {1562 this.status = this.getStatus();1563 this.statusText = this.getStatusText();1564 this.responseText = String.interpret(transport.responseText);1565 this.headerJSON = this._getHeaderJSON();1566 }1568 if(readyState == 4) {1569 var xml = transport.responseXML;1570 this.responseXML = Object.isUndefined(xml) ? null : xml;1571 this.responseJSON = this._getResponseJSON();1572 }1573 },1575 status: 0,1577 statusText: '',1579 getStatus: Ajax.Request.prototype.getStatus,1581 getStatusText: function() {1582 try {1583 return this.transport.statusText || '';1584 } catch (e) { return '' }1585 },1587 getHeader: Ajax.Request.prototype.getHeader,1589 getAllHeaders: function() {1590 try {1591 return this.getAllResponseHeaders();1592 } catch (e) { return null }1593 },1595 getResponseHeader: function(name) {1596 return this.transport.getResponseHeader(name);1597 },1599 getAllResponseHeaders: function() {1600 return this.transport.getAllResponseHeaders();1601 },1603 _getHeaderJSON: function() {1604 var json = this.getHeader('X-JSON');1605 if (!json) return null;1606 json = decodeURIComponent(escape(json));1607 try {1608 return json.evalJSON(this.request.options.sanitizeJSON ||1609 !this.request.isSameOrigin());1610 } catch (e) {1611 this.request.dispatchException(e);1612 }1613 },1615 _getResponseJSON: function() {1616 var options = this.request.options;1617 if (!options.evalJSON || (options.evalJSON != 'force' &&1618 !(this.getHeader('Content-type') || '').include('application/json')) ||1619 this.responseText.blank())1620 return null;1621 try {1622 return this.responseText.evalJSON(options.sanitizeJSON ||1623 !this.request.isSameOrigin());1624 } catch (e) {1625 this.request.dispatchException(e);1626 }1627 }1628 });1630 Ajax.Updater = Class.create(Ajax.Request, {1631 initialize: function($super, container, url, options) {1632 this.container = {1633 success: (container.success || container),1634 failure: (container.failure || (container.success ? null : container))1635 };1637 options = Object.clone(options);1638 var onComplete = options.onComplete;1639 options.onComplete = (function(response, json) {1640 this.updateContent(response.responseText);1641 if (Object.isFunction(onComplete)) onComplete(response, json);1642 }).bind(this);1644 $super(url, options);1645 },1647 updateContent: function(responseText) {1648 var receiver = this.container[this.success() ? 'success' : 'failure'],1649 options = this.options;1651 if (!options.evalScripts) responseText = responseText.stripScripts();1653 if (receiver = $(receiver)) {1654 if (options.insertion) {1655 if (Object.isString(options.insertion)) {1656 var insertion = { }; insertion[options.insertion] = responseText;1657 receiver.insert(insertion);1658 }1659 else options.insertion(receiver, responseText);1660 }1661 else receiver.update(responseText);1662 }1663 }1664 });1666 Ajax.PeriodicalUpdater = Class.create(Ajax.Base, {1667 initialize: function($super, container, url, options) {1668 $super(options);1669 this.onComplete = this.options.onComplete;1671 this.frequency = (this.options.frequency || 2);1672 this.decay = (this.options.decay || 1);1674 this.updater = { };1675 this.container = container;1676 this.url = url;1678 this.start();1679 },1681 start: function() {1682 this.options.onComplete = this.updateComplete.bind(this);1683 this.onTimerEvent();1684 },1686 stop: function() {1687 this.updater.options.onComplete = undefined;1688 clearTimeout(this.timer);1689 (this.onComplete || Prototype.emptyFunction).apply(this, arguments);1690 },1692 updateComplete: function(response) {1693 if (this.options.decay) {1694 this.decay = (response.responseText == this.lastText ?1695 this.decay * this.options.decay : 1);1697 this.lastText = response.responseText;1698 }1699 this.timer = this.onTimerEvent.bind(this).delay(this.decay * this.frequency);1700 },1702 onTimerEvent: function() {1703 this.updater = new Ajax.Updater(this.container, this.url, this.options);1704 }1705 });1709 function $(element) {1710 if (arguments.length > 1) {1711 for (var i = 0, elements = [], length = arguments.length; i < length; i++)1712 elements.push($(arguments[i]));1713 return elements;1714 }1715 if (Object.isString(element))1716 element = document.getElementById(element);1717 return Element.extend(element);1718 }1720 if (Prototype.BrowserFeatures.XPath) {1721 document._getElementsByXPath = function(expression, parentElement) {1722 var results = [];1723 var query = document.evaluate(expression, $(parentElement) || document,1724 null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);1725 for (var i = 0, length = query.snapshotLength; i < length; i++)1726 results.push(Element.extend(query.snapshotItem(i)));1727 return results;1728 };1729 }1731 /*--------------------------------------------------------------------------*/1733 if (!window.Node) var Node = { };1735 if (!Node.ELEMENT_NODE) {1736 Object.extend(Node, {1737 ELEMENT_NODE: 1,1738 ATTRIBUTE_NODE: 2,1739 TEXT_NODE: 3,1740 CDATA_SECTION_NODE: 4,1741 ENTITY_REFERENCE_NODE: 5,1742 ENTITY_NODE: 6,1743 PROCESSING_INSTRUCTION_NODE: 7,1744 COMMENT_NODE: 8,1745 DOCUMENT_NODE: 9,1746 DOCUMENT_TYPE_NODE: 10,1747 DOCUMENT_FRAGMENT_NODE: 11,1748 NOTATION_NODE: 121749 });1750 }1753 (function(global) {1755 var SETATTRIBUTE_IGNORES_NAME = (function(){1756 var elForm = document.createElement("form");1757 var elInput = document.createElement("input");1758 var root = document.documentElement;1759 elInput.setAttribute("name", "test");1760 elForm.appendChild(elInput);1761 root.appendChild(elForm);1762 var isBuggy = elForm.elements1763 ? (typeof elForm.elements.test == "undefined")1764 : null;1765 root.removeChild(elForm);1766 elForm = elInput = null;1767 return isBuggy;1768 })();1770 var element = global.Element;1771 global.Element = function(tagName, attributes) {1772 attributes = attributes || { };1773 tagName = tagName.toLowerCase();1774 var cache = Element.cache;1775 if (SETATTRIBUTE_IGNORES_NAME && attributes.name) {1776 tagName = '<' + tagName + ' name="' + attributes.name + '">';1777 delete attributes.name;1778 return Element.writeAttribute(document.createElement(tagName), attributes);1779 }1780 if (!cache[tagName]) cache[tagName] = Element.extend(document.createElement(tagName));1781 return Element.writeAttribute(cache[tagName].cloneNode(false), attributes);1782 };1783 Object.extend(global.Element, element || { });1784 if (element) global.Element.prototype = element.prototype;1785 })(this);1787 Element.cache = { };1788 Element.idCounter = 1;1790 Element.Methods = {1791 visible: function(element) {1792 return $(element).style.display != 'none';1793 },1795 toggle: function(element) {1796 element = $(element);1797 Element[Element.visible(element) ? 'hide' : 'show'](element);1798 return element;1799 },1802 hide: function(element) {1803 element = $(element);1804 element.style.display = 'none';1805 return element;1806 },1808 show: function(element) {1809 element = $(element);1810 element.style.display = '';1811 return element;1812 },1814 remove: function(element) {1815 element = $(element);1816 element.parentNode.removeChild(element);1817 return element;1818 },1820 update: (function(){1822 var SELECT_ELEMENT_INNERHTML_BUGGY = (function(){1823 var el = document.createElement("select"),1824 isBuggy = true;1825 el.innerHTML = "<option value=\"test\">test</option>";1826 if (el.options && el.options[0]) {1827 isBuggy = el.options[0].nodeName.toUpperCase() !== "OPTION";1828 }1829 el = null;1830 return isBuggy;1831 })();1833 var TABLE_ELEMENT_INNERHTML_BUGGY = (function(){1834 try {1835 var el = document.createElement("table");1836 if (el && el.tBodies) {1837 el.innerHTML = "<tbody><tr><td>test</td></tr></tbody>";1838 var isBuggy = typeof el.tBodies[0] == "undefined";1839 el = null;1840 return isBuggy;1841 }1842 } catch (e) {1843 return true;1844 }1845 })();1847 var SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING = (function () {1848 var s = document.createElement("script"),1849 isBuggy = false;1850 try {1851 s.appendChild(document.createTextNode(""));1852 isBuggy = !s.firstChild ||1853 s.firstChild && s.firstChild.nodeType !== 3;1854 } catch (e) {1855 isBuggy = true;1856 }1857 s = null;1858 return isBuggy;1859 })();1861 function update(element, content) {1862 element = $(element);1864 if (content && content.toElement)1865 content = content.toElement();1867 if (Object.isElement(content))1868 return element.update().insert(content);1870 content = Object.toHTML(content);1872 var tagName = element.tagName.toUpperCase();1874 if (tagName === 'SCRIPT' && SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING) {1875 element.text = content;1876 return element;1877 }1879 if (SELECT_ELEMENT_INNERHTML_BUGGY || TABLE_ELEMENT_INNERHTML_BUGGY) {1880 if (tagName in Element._insertionTranslations.tags) {1881 while (element.firstChild) {1882 element.removeChild(element.firstChild);1883 }1884 Element._getContentFromAnonymousElement(tagName, content.stripScripts())1885 .each(function(node) {1886 element.appendChild(node)1887 });1888 }1889 else {1890 element.innerHTML = content.stripScripts();1891 }1892 }1893 else {1894 element.innerHTML = content.stripScripts();1895 }1897 content.evalScripts.bind(content).defer();1898 return element;1899 }1901 return update;1902 })(),1904 replace: function(element, content) {1905 element = $(element);1906 if (content && content.toElement) content = content.toElement();1907 else if (!Object.isElement(content)) {1908 content = Object.toHTML(content);1909 var range = element.ownerDocument.createRange();1910 range.selectNode(element);1911 content.evalScripts.bind(content).defer();1912 content = range.createContextualFragment(content.stripScripts());1913 }1914 element.parentNode.replaceChild(content, element);1915 return element;1916 },1918 insert: function(element, insertions) {1919 element = $(element);1921 if (Object.isString(insertions) || Object.isNumber(insertions) ||1922 Object.isElement(insertions) || (insertions && (insertions.toElement || insertions.toHTML)))1923 insertions = {bottom:insertions};1925 var content, insert, tagName, childNodes;1927 for (var position in insertions) {1928 content = insertions[position];1929 position = position.toLowerCase();1930 insert = Element._insertionTranslations[position];1932 if (content && content.toElement) content = content.toElement();1933 if (Object.isElement(content)) {1934 insert(element, content);1935 continue;1936 }1938 content = Object.toHTML(content);1940 tagName = ((position == 'before' || position == 'after')1941 ? element.parentNode : element).tagName.toUpperCase();1943 childNodes = Element._getContentFromAnonymousElement(tagName, content.stripScripts());1945 if (position == 'top' || position == 'after') childNodes.reverse();1946 childNodes.each(insert.curry(element));1948 content.evalScripts.bind(content).defer();1949 }1951 return element;1952 },1954 wrap: function(element, wrapper, attributes) {1955 element = $(element);1956 if (Object.isElement(wrapper))1957 $(wrapper).writeAttribute(attributes || { });1958 else if (Object.isString(wrapper)) wrapper = new Element(wrapper, attributes);1959 else wrapper = new Element('div', wrapper);1960 if (element.parentNode)1961 element.parentNode.replaceChild(wrapper, element);1962 wrapper.appendChild(element);1963 return wrapper;1964 },1966 inspect: function(element) {1967 element = $(element);1968 var result = '<' + element.tagName.toLowerCase();1969 $H({'id': 'id', 'className': 'class'}).each(function(pair) {1970 var property = pair.first(), attribute = pair.last();1971 var value = (element[property] || '').toString();1972 if (value) result += ' ' + attribute + '=' + value.inspect(true);1973 });1974 return result + '>';1975 },1977 recursivelyCollect: function(element, property) {1978 element = $(element);1979 var elements = [];1980 while (element = element[property])1981 if (element.nodeType == 1)1982 elements.push(Element.extend(element));1983 return elements;1984 },1986 ancestors: function(element) {1987 return Element.recursivelyCollect(element, 'parentNode');1988 },1990 descendants: function(element) {1991 return Element.select(element, "*");1992 },1994 firstDescendant: function(element) {1995 element = $(element).firstChild;1996 while (element && element.nodeType != 1) element = element.nextSibling;1997 return $(element);1998 },2000 immediateDescendants: function(element) {2001 if (!(element = $(element).firstChild)) return [];2002 while (element && element.nodeType != 1) element = element.nextSibling;2003 if (element) return [element].concat($(element).nextSiblings());2004 return [];2005 },2007 previousSiblings: function(element) {2008 return Element.recursivelyCollect(element, 'previousSibling');2009 },2011 nextSiblings: function(element) {2012 return Element.recursivelyCollect(element, 'nextSibling');2013 },2015 siblings: function(element) {2016 element = $(element);2017 return Element.previousSiblings(element).reverse()2018 .concat(Element.nextSiblings(element));2019 },2021 match: function(element, selector) {2022 if (Object.isString(selector))2023 selector = new Selector(selector);2024 return selector.match($(element));2025 },2027 up: function(element, expression, index) {2028 element = $(element);2029 if (arguments.length == 1) return $(element.parentNode);2030 var ancestors = Element.ancestors(element);2031 return Object.isNumber(expression) ? ancestors[expression] :2032 Selector.findElement(ancestors, expression, index);2033 },2035 down: function(element, expression, index) {2036 element = $(element);2037 if (arguments.length == 1) return Element.firstDescendant(element);2038 return Object.isNumber(expression) ? Element.descendants(element)[expression] :2039 Element.select(element, expression)[index || 0];2040 },2042 previous: function(element, expression, index) {2043 element = $(element);2044 if (arguments.length == 1) return $(Selector.handlers.previousElementSibling(element));2045 var previousSiblings = Element.previousSiblings(element);2046 return Object.isNumber(expression) ? previousSiblings[expression] :2047 Selector.findElement(previousSiblings, expression, index);2048 },2050 next: function(element, expression, index) {2051 element = $(element);2052 if (arguments.length == 1) return $(Selector.handlers.nextElementSibling(element));2053 var nextSiblings = Element.nextSiblings(element);2054 return Object.isNumber(expression) ? nextSiblings[expression] :2055 Selector.findElement(nextSiblings, expression, index);2056 },2059 select: function(element) {2060 var args = Array.prototype.slice.call(arguments, 1);2061 return Selector.findChildElements(element, args);2062 },2064 adjacent: function(element) {2065 var args = Array.prototype.slice.call(arguments, 1);2066 return Selector.findChildElements(element.parentNode, args).without(element);2067 },2069 identify: function(element) {2070 element = $(element);2071 var id = Element.readAttribute(element, 'id');2072 if (id) return id;2073 do { id = 'anonymous_element_' + Element.idCounter++ } while ($(id));2074 Element.writeAttribute(element, 'id', id);2075 return id;2076 },2078 readAttribute: function(element, name) {2079 element = $(element);2080 if (Prototype.Browser.IE) {2081 var t = Element._attributeTranslations.read;2082 if (t.values[name]) return t.values[name](element, name);2083 if (t.names[name]) name = t.names[name];2084 if (name.include(':')) {2085 return (!element.attributes || !element.attributes[name]) ? null :2086 element.attributes[name].value;2087 }2088 }2089 return element.getAttribute(name);2090 },2092 writeAttribute: function(element, name, value) {2093 element = $(element);2094 var attributes = { }, t = Element._attributeTranslations.write;2096 if (typeof name == 'object') attributes = name;2097 else attributes[name] = Object.isUndefined(value) ? true : value;2099 for (var attr in attributes) {2100 name = t.names[attr] || attr;2101 value = attributes[attr];2102 if (t.values[attr]) name = t.values[attr](element, value);2103 if (value === false || value === null)2104 element.removeAttribute(name);2105 else if (value === true)2106 element.setAttribute(name, name);2107 else element.setAttribute(name, value);2108 }2109 return element;2110 },2112 getHeight: function(element) {2113 return Element.getDimensions(element).height;2114 },2116 getWidth: function(element) {2117 return Element.getDimensions(element).width;2118 },2120 classNames: function(element) {2121 return new Element.ClassNames(element);2122 },2124 hasClassName: function(element, className) {2125 if (!(element = $(element))) return;2126 var elementClassName = element.className;2127 return (elementClassName.length > 0 && (elementClassName == className ||2128 new RegExp("(^|\\s)" + className + "(\\s|$)").test(elementClassName)));2129 },2131 addClassName: function(element, className) {2132 if (!(element = $(element))) return;2133 if (!Element.hasClassName(element, className))2134 element.className += (element.className ? ' ' : '') + className;2135 return element;2136 },2138 removeClassName: function(element, className) {2139 if (!(element = $(element))) return;2140 element.className = element.className.replace(2141 new RegExp("(^|\\s+)" + className + "(\\s+|$)"), ' ').strip();2142 return element;2143 },2145 toggleClassName: function(element, className) {2146 if (!(element = $(element))) return;2147 return Element[Element.hasClassName(element, className) ?2148 'removeClassName' : 'addClassName'](element, className);2149 },2151 cleanWhitespace: function(element) {2152 element = $(element);2153 var node = element.firstChild;2154 while (node) {2155 var nextNode = node.nextSibling;2156 if (node.nodeType == 3 && !/\S/.test(node.nodeValue))2157 element.removeChild(node);2158 node = nextNode;2159 }2160 return element;2161 },2163 empty: function(element) {2164 return $(element).innerHTML.blank();2165 },2167 descendantOf: function(element, ancestor) {2168 element = $(element), ancestor = $(ancestor);2170 if (element.compareDocumentPosition)2171 return (element.compareDocumentPosition(ancestor) & 8) === 8;2173 if (ancestor.contains)2174 return ancestor.contains(element) && ancestor !== element;2176 while (element = element.parentNode)2177 if (element == ancestor) return true;2179 return false;2180 },2182 scrollTo: function(element) {2183 element = $(element);2184 var pos = Element.cumulativeOffset(element);2185 window.scrollTo(pos[0], pos[1]);2186 return element;2187 },2189 getStyle: function(element, style) {2190 element = $(element);2191 style = style == 'float' ? 'cssFloat' : style.camelize();2192 var value = element.style[style];2193 if (!value || value == 'auto') {2194 var css = document.defaultView.getComputedStyle(element, null);2195 value = css ? css[style] : null;2196 }2197 if (style == 'opacity') return value ? parseFloat(value) : 1.0;2198 return value == 'auto' ? null : value;2199 },2201 getOpacity: function(element) {2202 return $(element).getStyle('opacity');2203 },2205 setStyle: function(element, styles) {2206 element = $(element);2207 var elementStyle = element.style, match;2208 if (Object.isString(styles)) {2209 element.style.cssText += ';' + styles;2210 return styles.include('opacity') ?2211 element.setOpacity(styles.match(/opacity:\s*(\d?\.?\d*)/)[1]) : element;2212 }2213 for (var property in styles)2214 if (property == 'opacity') element.setOpacity(styles[property]);2215 else2216 elementStyle[(property == 'float' || property == 'cssFloat') ?2217 (Object.isUndefined(elementStyle.styleFloat) ? 'cssFloat' : 'styleFloat') :2218 property] = styles[property];2220 return element;2221 },2223 setOpacity: function(element, value) {2224 element = $(element);2225 element.style.opacity = (value == 1 || value === '') ? '' :2226 (value < 0.00001) ? 0 : value;2227 return element;2228 },2230 getDimensions: function(element) {2231 element = $(element);2232 var display = Element.getStyle(element, 'display');2233 if (display != 'none' && display != null) // Safari bug2234 return {width: element.offsetWidth, height: element.offsetHeight};2236 var els = element.style;2237 var originalVisibility = els.visibility;2238 var originalPosition = els.position;2239 var originalDisplay = els.display;2240 els.visibility = 'hidden';2241 if (originalPosition != 'fixed') // Switching fixed to absolute causes issues in Safari2242 els.position = 'absolute';2243 els.display = 'block';2244 var originalWidth = element.clientWidth;2245 var originalHeight = element.clientHeight;2246 els.display = originalDisplay;2247 els.position = originalPosition;2248 els.visibility = originalVisibility;2249 return {width: originalWidth, height: originalHeight};2250 },2252 makePositioned: function(element) {2253 element = $(element);2254 var pos = Element.getStyle(element, 'position');2255 if (pos == 'static' || !pos) {2256 element._madePositioned = true;2257 element.style.position = 'relative';2258 if (Prototype.Browser.Opera) {2259 element.style.top = 0;2260 element.style.left = 0;2261 }2262 }2263 return element;2264 },2266 undoPositioned: function(element) {2267 element = $(element);2268 if (element._madePositioned) {2269 element._madePositioned = undefined;2270 element.style.position =2271 element.style.top =2272 element.style.left =2273 element.style.bottom =2274 element.style.right = '';2275 }2276 return element;2277 },2279 makeClipping: function(element) {2280 element = $(element);2281 if (element._overflow) return element;2282 element._overflow = Element.getStyle(element, 'overflow') || 'auto';2283 if (element._overflow !== 'hidden')2284 element.style.overflow = 'hidden';2285 return element;2286 },2288 undoClipping: function(element) {2289 element = $(element);2290 if (!element._overflow) return element;2291 element.style.overflow = element._overflow == 'auto' ? '' : element._overflow;2292 element._overflow = null;2293 return element;2294 },2296 cumulativeOffset: function(element) {2297 var valueT = 0, valueL = 0;2298 do {2299 valueT += element.offsetTop || 0;2300 valueL += element.offsetLeft || 0;2301 element = element.offsetParent;2302 } while (element);2303 return Element._returnOffset(valueL, valueT);2304 },2306 positionedOffset: function(element) {2307 var valueT = 0, valueL = 0;2308 do {2309 valueT += element.offsetTop || 0;2310 valueL += element.offsetLeft || 0;2311 element = element.offsetParent;2312 if (element) {2313 if (element.tagName.toUpperCase() == 'BODY') break;2314 var p = Element.getStyle(element, 'position');2315 if (p !== 'static') break;2316 }2317 } while (element);2318 return Element._returnOffset(valueL, valueT);2319 },2321 absolutize: function(element) {2322 element = $(element);2323 if (Element.getStyle(element, 'position') == 'absolute') return element;2325 var offsets = Element.positionedOffset(element);2326 var top = offsets[1];2327 var left = offsets[0];2328 var width = element.clientWidth;2329 var height = element.clientHeight;2331 element._originalLeft = left - parseFloat(element.style.left || 0);2332 element._originalTop = top - parseFloat(element.style.top || 0);2333 element._originalWidth = element.style.width;2334 element._originalHeight = element.style.height;2336 element.style.position = 'absolute';2337 element.style.top = top + 'px';2338 element.style.left = left + 'px';2339 element.style.width = width + 'px';2340 element.style.height = height + 'px';2341 return element;2342 },2344 relativize: function(element) {2345 element = $(element);2346 if (Element.getStyle(element, 'position') == 'relative') return element;2348 element.style.position = 'relative';2349 var top = parseFloat(element.style.top || 0) - (element._originalTop || 0);2350 var left = parseFloat(element.style.left || 0) - (element._originalLeft || 0);2352 element.style.top = top + 'px';2353 element.style.left = left + 'px';2354 element.style.height = element._originalHeight;2355 element.style.width = element._originalWidth;2356 return element;2357 },2359 cumulativeScrollOffset: function(element) {2360 var valueT = 0, valueL = 0;2361 do {2362 valueT += element.scrollTop || 0;2363 valueL += element.scrollLeft || 0;2364 element = element.parentNode;2365 } while (element);2366 return Element._returnOffset(valueL, valueT);2367 },2369 getOffsetParent: function(element) {2370 if (element.offsetParent) return $(element.offsetParent);2371 if (element == document.body) return $(element);2373 while ((element = element.parentNode) && element != document.body)2374 if (Element.getStyle(element, 'position') != 'static')2375 return $(element);2377 return $(document.body);2378 },2380 viewportOffset: function(forElement) {2381 var valueT = 0, valueL = 0;2383 var element = forElement;2384 do {2385 valueT += element.offsetTop || 0;2386 valueL += element.offsetLeft || 0;2388 if (element.offsetParent == document.body &&2389 Element.getStyle(element, 'position') == 'absolute') break;2391 } while (element = element.offsetParent);2393 element = forElement;2394 do {2395 if (!Prototype.Browser.Opera || (element.tagName && (element.tagName.toUpperCase() == 'BODY'))) {2396 valueT -= element.scrollTop || 0;2397 valueL -= element.scrollLeft || 0;2398 }2399 } while (element = element.parentNode);2401 return Element._returnOffset(valueL, valueT);2402 },2404 clonePosition: function(element, source) {2405 var options = Object.extend({2406 setLeft: true,2407 setTop: true,2408 setWidth: true,2409 setHeight: true,2410 offsetTop: 0,2411 offsetLeft: 02412 }, arguments[2] || { });2414 source = $(source);2415 var p = Element.viewportOffset(source);2417 element = $(element);2418 var delta = [0, 0];2419 var parent = null;2420 if (Element.getStyle(element, 'position') == 'absolute') {2421 parent = Element.getOffsetParent(element);2422 delta = Element.viewportOffset(parent);2423 }2425 if (parent == document.body) {2426 delta[0] -= document.body.offsetLeft;2427 delta[1] -= document.body.offsetTop;2428 }2430 if (options.setLeft) element.style.left = (p[0] - delta[0] + options.offsetLeft) + 'px';2431 if (options.setTop) element.style.top = (p[1] - delta[1] + options.offsetTop) + 'px';2432 if (options.setWidth) element.style.width = source.offsetWidth + 'px';2433 if (options.setHeight) element.style.height = source.offsetHeight + 'px';2434 return element;2435 }2436 };2438 Object.extend(Element.Methods, {2439 getElementsBySelector: Element.Methods.select,2441 childElements: Element.Methods.immediateDescendants2442 });2444 Element._attributeTranslations = {2445 write: {2446 names: {2447 className: 'class',2448 htmlFor: 'for'2449 },2450 values: { }2451 }2452 };2454 if (Prototype.Browser.Opera) {2455 Element.Methods.getStyle = Element.Methods.getStyle.wrap(2456 function(proceed, element, style) {2457 switch (style) {2458 case 'left': case 'top': case 'right': case 'bottom':2459 if (proceed(element, 'position') === 'static') return null;2460 case 'height': case 'width':2461 if (!Element.visible(element)) return null;2463 var dim = parseInt(proceed(element, style), 10);2465 if (dim !== element['offset' + style.capitalize()])2466 return dim + 'px';2468 var properties;2469 if (style === 'height') {2470 properties = ['border-top-width', 'padding-top',2471 'padding-bottom', 'border-bottom-width'];2472 }2473 else {2474 properties = ['border-left-width', 'padding-left',2475 'padding-right', 'border-right-width'];2476 }2477 return properties.inject(dim, function(memo, property) {2478 var val = proceed(element, property);2479 return val === null ? memo : memo - parseInt(val, 10);2480 }) + 'px';2481 default: return proceed(element, style);2482 }2483 }2484 );2486 Element.Methods.readAttribute = Element.Methods.readAttribute.wrap(2487 function(proceed, element, attribute) {2488 if (attribute === 'title') return element.title;2489 return proceed(element, attribute);2490 }2491 );2492 }2494 else if (Prototype.Browser.IE) {2495 Element.Methods.getOffsetParent = Element.Methods.getOffsetParent.wrap(2496 function(proceed, element) {2497 element = $(element);2498 try { element.offsetParent }2499 catch(e) { return $(document.body) }2500 var position = element.getStyle('position');2501 if (position !== 'static') return proceed(element);2502 element.setStyle({ position: 'relative' });2503 var value = proceed(element);2504 element.setStyle({ position: position });2505 return value;2506 }2507 );2509 $w('positionedOffset viewportOffset').each(function(method) {2510 Element.Methods[method] = Element.Methods[method].wrap(2511 function(proceed, element) {2512 element = $(element);2513 try { element.offsetParent }2514 catch(e) { return Element._returnOffset(0,0) }2515 var position = element.getStyle('position');2516 if (position !== 'static') return proceed(element);2517 var offsetParent = element.getOffsetParent();2518 if (offsetParent && offsetParent.getStyle('position') === 'fixed')2519 offsetParent.setStyle({ zoom: 1 });2520 element.setStyle({ position: 'relative' });2521 var value = proceed(element);2522 element.setStyle({ position: position });2523 return value;2524 }2525 );2526 });2528 Element.Methods.cumulativeOffset = Element.Methods.cumulativeOffset.wrap(2529 function(proceed, element) {2530 try { element.offsetParent }2531 catch(e) { return Element._returnOffset(0,0) }2532 return proceed(element);2533 }2534 );2536 Element.Methods.getStyle = function(element, style) {2537 element = $(element);2538 style = (style == 'float' || style == 'cssFloat') ? 'styleFloat' : style.camelize();2539 var value = element.style[style];2540 if (!value && element.currentStyle) value = element.currentStyle[style];2542 if (style == 'opacity') {2543 if (value = (element.getStyle('filter') || '').match(/alpha\(opacity=(.*)\)/))2544 if (value[1]) return parseFloat(value[1]) / 100;2545 return 1.0;2546 }2548 if (value == 'auto') {2549 if ((style == 'width' || style == 'height') && (element.getStyle('display') != 'none'))2550 return element['offset' + style.capitalize()] + 'px';2551 return null;2552 }2553 return value;2554 };2556 Element.Methods.setOpacity = function(element, value) {2557 function stripAlpha(filter){2558 return filter.replace(/alpha\([^\)]*\)/gi,'');2559 }2560 element = $(element);2561 var currentStyle = element.currentStyle;2562 if ((currentStyle && !currentStyle.hasLayout) ||2563 (!currentStyle && element.style.zoom == 'normal'))2564 element.style.zoom = 1;2566 var filter = element.getStyle('filter'), style = element.style;2567 if (value == 1 || value === '') {2568 (filter = stripAlpha(filter)) ?2569 style.filter = filter : style.removeAttribute('filter');2570 return element;2571 } else if (value < 0.00001) value = 0;2572 style.filter = stripAlpha(filter) +2573 'alpha(opacity=' + (value * 100) + ')';2574 return element;2575 };2577 Element._attributeTranslations = (function(){2579 var classProp = 'className';2580 var forProp = 'for';2582 var el = document.createElement('div');2584 el.setAttribute(classProp, 'x');2586 if (el.className !== 'x') {2587 el.setAttribute('class', 'x');2588 if (el.className === 'x') {2589 classProp = 'class';2590 }2591 }2592 el = null;2594 el = document.createElement('label');2595 el.setAttribute(forProp, 'x');2596 if (el.htmlFor !== 'x') {2597 el.setAttribute('htmlFor', 'x');2598 if (el.htmlFor === 'x') {2599 forProp = 'htmlFor';2600 }2601 }2602 el = null;2604 return {2605 read: {2606 names: {2607 'class': classProp,2608 'className': classProp,2609 'for': forProp,2610 'htmlFor': forProp2611 },2612 values: {2613 _getAttr: function(element, attribute) {2614 return element.getAttribute(attribute);2615 },2616 _getAttr2: function(element, attribute) {2617 return element.getAttribute(attribute, 2);2618 },2619 _getAttrNode: function(element, attribute) {2620 var node = element.getAttributeNode(attribute);2621 return node ? node.value : "";2622 },2623 _getEv: (function(){2625 var el = document.createElement('div');2626 el.onclick = Prototype.emptyFunction;2627 var value = el.getAttribute('onclick');2628 var f;2630 if (String(value).indexOf('{') > -1) {2631 f = function(element, attribute) {2632 attribute = element.getAttribute(attribute);2633 if (!attribute) return null;2634 attribute = attribute.toString();2635 attribute = attribute.split('{')[1];2636 attribute = attribute.split('}')[0];2637 return attribute.strip();2638 };2639 }2640 else if (value === '') {2641 f = function(element, attribute) {2642 attribute = element.getAttribute(attribute);2643 if (!attribute) return null;2644 return attribute.strip();2645 };2646 }2647 el = null;2648 return f;2649 })(),2650 _flag: function(element, attribute) {2651 return $(element).hasAttribute(attribute) ? attribute : null;2652 },2653 style: function(element) {2654 return element.style.cssText.toLowerCase();2655 },2656 title: function(element) {2657 return element.title;2658 }2659 }2660 }2661 }2662 })();2664 Element._attributeTranslations.write = {2665 names: Object.extend({2666 cellpadding: 'cellPadding',2667 cellspacing: 'cellSpacing'2668 }, Element._attributeTranslations.read.names),2669 values: {2670 checked: function(element, value) {2671 element.checked = !!value;2672 },2674 style: function(element, value) {2675 element.style.cssText = value ? value : '';2676 }2677 }2678 };2680 Element._attributeTranslations.has = {};2682 $w('colSpan rowSpan vAlign dateTime accessKey tabIndex ' +2683 'encType maxLength readOnly longDesc frameBorder').each(function(attr) {2684 Element._attributeTranslations.write.names[attr.toLowerCase()] = attr;2685 Element._attributeTranslations.has[attr.toLowerCase()] = attr;2686 });2688 (function(v) {2689 Object.extend(v, {2690 href: v._getAttr2,2691 src: v._getAttr2,2692 type: v._getAttr,2693 action: v._getAttrNode,2694 disabled: v._flag,2695 checked: v._flag,2696 readonly: v._flag,2697 multiple: v._flag,2698 onload: v._getEv,2699 onunload: v._getEv,2700 onclick: v._getEv,2701 ondblclick: v._getEv,2702 onmousedown: v._getEv,2703 onmouseup: v._getEv,2704 onmouseover: v._getEv,2705 onmousemove: v._getEv,2706 onmouseout: v._getEv,2707 onfocus: v._getEv,2708 onblur: v._getEv,2709 onkeypress: v._getEv,2710 onkeydown: v._getEv,2711 onkeyup: v._getEv,2712 onsubmit: v._getEv,2713 onreset: v._getEv,2714 onselect: v._getEv,2715 onchange: v._getEv2716 });2717 })(Element._attributeTranslations.read.values);2719 if (Prototype.BrowserFeatures.ElementExtensions) {2720 (function() {2721 function _descendants(element) {2722 var nodes = element.getElementsByTagName('*'), results = [];2723 for (var i = 0, node; node = nodes[i]; i++)2724 if (node.tagName !== "!") // Filter out comment nodes.2725 results.push(node);2726 return results;2727 }2729 Element.Methods.down = function(element, expression, index) {2730 element = $(element);2731 if (arguments.length == 1) return element.firstDescendant();2732 return Object.isNumber(expression) ? _descendants(element)[expression] :2733 Element.select(element, expression)[index || 0];2734 }2735 })();2736 }2738 }2740 else if (Prototype.Browser.Gecko && /rv:1\.8\.0/.test(navigator.userAgent)) {2741 Element.Methods.setOpacity = function(element, value) {2742 element = $(element);2743 element.style.opacity = (value == 1) ? 0.999999 :2744 (value === '') ? '' : (value < 0.00001) ? 0 : value;2745 return element;2746 };2747 }2749 else if (Prototype.Browser.WebKit) {2750 Element.Methods.setOpacity = function(element, value) {2751 element = $(element);2752 element.style.opacity = (value == 1 || value === '') ? '' :2753 (value < 0.00001) ? 0 : value;2755 if (value == 1)2756 if(element.tagName.toUpperCase() == 'IMG' && element.width) {2757 element.width++; element.width--;2758 } else try {2759 var n = document.createTextNode(' ');2760 element.appendChild(n);2761 element.removeChild(n);2762 } catch (e) { }2764 return element;2765 };2767 Element.Methods.cumulativeOffset = function(element) {2768 var valueT = 0, valueL = 0;2769 do {2770 valueT += element.offsetTop || 0;2771 valueL += element.offsetLeft || 0;2772 if (element.offsetParent == document.body)2773 if (Element.getStyle(element, 'position') == 'absolute') break;2775 element = element.offsetParent;2776 } while (element);2778 return Element._returnOffset(valueL, valueT);2779 };2780 }2782 if ('outerHTML' in document.documentElement) {2783 Element.Methods.replace = function(element, content) {2784 element = $(element);2786 if (content && content.toElement) content = content.toElement();2787 if (Object.isElement(content)) {2788 element.parentNode.replaceChild(content, element);2789 return element;2790 }2792 content = Object.toHTML(content);2793 var parent = element.parentNode, tagName = parent.tagName.toUpperCase();2795 if (Element._insertionTranslations.tags[tagName]) {2796 var nextSibling = element.next();2797 var fragments = Element._getContentFromAnonymousElement(tagName, content.stripScripts());2798 parent.removeChild(element);2799 if (nextSibling)2800 fragments.each(function(node) { parent.insertBefore(node, nextSibling) });2801 else2802 fragments.each(function(node) { parent.appendChild(node) });2803 }2804 else element.outerHTML = content.stripScripts();2806 content.evalScripts.bind(content).defer();2807 return element;2808 };2809 }2811 Element._returnOffset = function(l, t) {2812 var result = [l, t];2813 result.left = l;2814 result.top = t;2815 return result;2816 };2818 Element._getContentFromAnonymousElement = function(tagName, html) {2819 var div = new Element('div'), t = Element._insertionTranslations.tags[tagName];2820 if (t) {2821 div.innerHTML = t[0] + html + t[1];2822 t[2].times(function() { div = div.firstChild });2823 } else div.innerHTML = html;2824 return $A(div.childNodes);2825 };2827 Element._insertionTranslations = {2828 before: function(element, node) {2829 element.parentNode.insertBefore(node, element);2830 },2831 top: function(element, node) {2832 element.insertBefore(node, element.firstChild);2833 },2834 bottom: function(element, node) {2835 element.appendChild(node);2836 },2837 after: function(element, node) {2838 element.parentNode.insertBefore(node, element.nextSibling);2839 },2840 tags: {2841 TABLE: ['<table>', '</table>', 1],2842 TBODY: ['<table><tbody>', '</tbody></table>', 2],2843 TR: ['<table><tbody><tr>', '</tr></tbody></table>', 3],2844 TD: ['<table><tbody><tr><td>', '</td></tr></tbody></table>', 4],2845 SELECT: ['<select>', '</select>', 1]2846 }2847 };2849 (function() {2850 var tags = Element._insertionTranslations.tags;2851 Object.extend(tags, {2852 THEAD: tags.TBODY,2853 TFOOT: tags.TBODY,2854 TH: tags.TD2855 });2856 })();2858 Element.Methods.Simulated = {2859 hasAttribute: function(element, attribute) {2860 attribute = Element._attributeTranslations.has[attribute] || attribute;2861 var node = $(element).getAttributeNode(attribute);2862 return !!(node && node.specified);2863 }2864 };2866 Element.Methods.ByTag = { };2868 Object.extend(Element, Element.Methods);2870 (function(div) {2872 if (!Prototype.BrowserFeatures.ElementExtensions && div['__proto__']) {2873 window.HTMLElement = { };2874 window.HTMLElement.prototype = div['__proto__'];2875 Prototype.BrowserFeatures.ElementExtensions = true;2876 }2878 div = null;2880 })(document.createElement('div'))2882 Element.extend = (function() {2884 function checkDeficiency(tagName) {2885 if (typeof window.Element != 'undefined') {2886 var proto = window.Element.prototype;2887 if (proto) {2888 var id = '_' + (Math.random()+'').slice(2);2889 var el = document.createElement(tagName);2890 proto[id] = 'x';2891 var isBuggy = (el[id] !== 'x');2892 delete proto[id];2893 el = null;2894 return isBuggy;2895 }2896 }2897 return false;2898 }2900 function extendElementWith(element, methods) {2901 for (var property in methods) {2902 var value = methods[property];2903 if (Object.isFunction(value) && !(property in element))2904 element[property] = value.methodize();2905 }2906 }2908 var HTMLOBJECTELEMENT_PROTOTYPE_BUGGY = checkDeficiency('object');2910 if (Prototype.BrowserFeatures.SpecificElementExtensions) {2911 if (HTMLOBJECTELEMENT_PROTOTYPE_BUGGY) {2912 return function(element) {2913 if (element && typeof element._extendedByPrototype == 'undefined') {2914 var t = element.tagName;2915 if (t && (/^(?:object|applet|embed)$/i.test(t))) {2916 extendElementWith(element, Element.Methods);2917 extendElementWith(element, Element.Methods.Simulated);2918 extendElementWith(element, Element.Methods.ByTag[t.toUpperCase()]);2919 }2920 }2921 return element;2922 }2923 }2924 return Prototype.K;2925 }2927 var Methods = { }, ByTag = Element.Methods.ByTag;2929 var extend = Object.extend(function(element) {2930 if (!element || typeof element._extendedByPrototype != 'undefined' ||2931 element.nodeType != 1 || element == window) return element;2933 var methods = Object.clone(Methods),2934 tagName = element.tagName.toUpperCase();2936 if (ByTag[tagName]) Object.extend(methods, ByTag[tagName]);2938 extendElementWith(element, methods);2940 element._extendedByPrototype = Prototype.emptyFunction;2941 return element;2943 }, {2944 refresh: function() {2945 if (!Prototype.BrowserFeatures.ElementExtensions) {2946 Object.extend(Methods, Element.Methods);2947 Object.extend(Methods, Element.Methods.Simulated);2948 }2949 }2950 });2952 extend.refresh();2953 return extend;2954 })();2956 Element.hasAttribute = function(element, attribute) {2957 if (element.hasAttribute) return element.hasAttribute(attribute);2958 return Element.Methods.Simulated.hasAttribute(element, attribute);2959 };2961 Element.addMethods = function(methods) {2962 var F = Prototype.BrowserFeatures, T = Element.Methods.ByTag;2964 if (!methods) {2965 Object.extend(Form, Form.Methods);2966 Object.extend(Form.Element, Form.Element.Methods);2967 Object.extend(Element.Methods.ByTag, {2968 "FORM": Object.clone(Form.Methods),2969 "INPUT": Object.clone(Form.Element.Methods),2970 "SELECT": Object.clone(Form.Element.Methods),2971 "TEXTAREA": Object.clone(Form.Element.Methods)2972 });2973 }2975 if (arguments.length == 2) {2976 var tagName = methods;2977 methods = arguments[1];2978 }2980 if (!tagName) Object.extend(Element.Methods, methods || { });2981 else {2982 if (Object.isArray(tagName)) tagName.each(extend);2983 else extend(tagName);2984 }2986 function extend(tagName) {2987 tagName = tagName.toUpperCase();2988 if (!Element.Methods.ByTag[tagName])2989 Element.Methods.ByTag[tagName] = { };2990 Object.extend(Element.Methods.ByTag[tagName], methods);2991 }2993 function copy(methods, destination, onlyIfAbsent) {2994 onlyIfAbsent = onlyIfAbsent || false;2995 for (var property in methods) {2996 var value = methods[property];2997 if (!Object.isFunction(value)) continue;2998 if (!onlyIfAbsent || !(property in destination))2999 destination[property] = value.methodize();3000 }3001 }3003 function findDOMClass(tagName) {3004 var klass;3005 var trans = {3006 "OPTGROUP": "OptGroup", "TEXTAREA": "TextArea", "P": "Paragraph",3007 "FIELDSET": "FieldSet", "UL": "UList", "OL": "OList", "DL": "DList",3008 "DIR": "Directory", "H1": "Heading", "H2": "Heading", "H3": "Heading",3009 "H4": "Heading", "H5": "Heading", "H6": "Heading", "Q": "Quote",3010 "INS": "Mod", "DEL": "Mod", "A": "Anchor", "IMG": "Image", "CAPTION":3011 "TableCaption", "COL": "TableCol", "COLGROUP": "TableCol", "THEAD":3012 "TableSection", "TFOOT": "TableSection", "TBODY": "TableSection", "TR":3013 "TableRow", "TH": "TableCell", "TD": "TableCell", "FRAMESET":3014 "FrameSet", "IFRAME": "IFrame"3015 };3016 if (trans[tagName]) klass = 'HTML' + trans[tagName] + 'Element';3017 if (window[klass]) return window[klass];3018 klass = 'HTML' + tagName + 'Element';3019 if (window[klass]) return window[klass];3020 klass = 'HTML' + tagName.capitalize() + 'Element';3021 if (window[klass]) return window[klass];3023 var element = document.createElement(tagName);3024 var proto = element['__proto__'] || element.constructor.prototype;3025 element = null;3026 return proto;3027 }3029 var elementPrototype = window.HTMLElement ? HTMLElement.prototype :3030 Element.prototype;3032 if (F.ElementExtensions) {3033 copy(Element.Methods, elementPrototype);3034 copy(Element.Methods.Simulated, elementPrototype, true);3035 }3037 if (F.SpecificElementExtensions) {3038 for (var tag in Element.Methods.ByTag) {3039 var klass = findDOMClass(tag);3040 if (Object.isUndefined(klass)) continue;3041 copy(T[tag], klass.prototype);3042 }3043 }3045 Object.extend(Element, Element.Methods);3046 delete Element.ByTag;3048 if (Element.extend.refresh) Element.extend.refresh();3049 Element.cache = { };3050 };3053 document.viewport = {3055 getDimensions: function() {3056 return { width: this.getWidth(), height: this.getHeight() };3057 },3059 getScrollOffsets: function() {3060 return Element._returnOffset(3061 window.pageXOffset || document.documentElement.scrollLeft || document.body.scrollLeft,3062 window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop);3063 }3064 };3066 (function(viewport) {3067 var B = Prototype.Browser, doc = document, element, property = {};3069 function getRootElement() {3070 if (B.WebKit && !doc.evaluate)3071 return document;3073 if (B.Opera && window.parseFloat(window.opera.version()) < 9.5)3074 return document.body;3076 return document.documentElement;3077 }3079 function define(D) {3080 if (!element) element = getRootElement();3082 property[D] = 'client' + D;3084 viewport['get' + D] = function() { return element[property[D]] };3085 return viewport['get' + D]();3086 }3088 viewport.getWidth = define.curry('Width');3090 viewport.getHeight = define.curry('Height');3091 })(document.viewport);3094 Element.Storage = {3095 UID: 13096 };3098 Element.addMethods({3099 getStorage: function(element) {3100 if (!(element = $(element))) return;3102 var uid;3103 if (element === window) {3104 uid = 0;3105 } else {3106 if (typeof element._prototypeUID === "undefined")3107 element._prototypeUID = [Element.Storage.UID++];3108 uid = element._prototypeUID[0];3109 }3111 if (!Element.Storage[uid])3112 Element.Storage[uid] = $H();3114 return Element.Storage[uid];3115 },3117 store: function(element, key, value) {3118 if (!(element = $(element))) return;3120 if (arguments.length === 2) {3121 Element.getStorage(element).update(key);3122 } else {3123 Element.getStorage(element).set(key, value);3124 }3126 return element;3127 },3129 retrieve: function(element, key, defaultValue) {3130 if (!(element = $(element))) return;3131 var hash = Element.getStorage(element), value = hash.get(key);3133 if (Object.isUndefined(value)) {3134 hash.set(key, defaultValue);3135 value = defaultValue;3136 }3138 return value;3139 },3141 clone: function(element, deep) {3142 if (!(element = $(element))) return;3143 var clone = element.cloneNode(deep);3144 clone._prototypeUID = void 0;3145 if (deep) {3146 var descendants = Element.select(clone, '*'),3147 i = descendants.length;3148 while (i--) {3149 descendants[i]._prototypeUID = void 0;3150 }3151 }3152 return Element.extend(clone);3153 }3154 });3155 /* Portions of the Selector class are derived from Jack Slocum's DomQuery,3156 * part of YUI-Ext version 0.40, distributed under the terms of an MIT-style3157 * license. Please see http://www.yui-ext.com/ for more information. */3159 var Selector = Class.create({3160 initialize: function(expression) {3161 this.expression = expression.strip();3163 if (this.shouldUseSelectorsAPI()) {3164 this.mode = 'selectorsAPI';3165 } else if (this.shouldUseXPath()) {3166 this.mode = 'xpath';3167 this.compileXPathMatcher();3168 } else {3169 this.mode = "normal";3170 this.compileMatcher();3171 }3173 },3175 shouldUseXPath: (function() {3177 var IS_DESCENDANT_SELECTOR_BUGGY = (function(){3178 var isBuggy = false;3179 if (document.evaluate && window.XPathResult) {3180 var el = document.createElement('div');3181 el.innerHTML = '<ul><li></li></ul><div><ul><li></li></ul></div>';3183 var xpath = ".//*[local-name()='ul' or local-name()='UL']" +3184 "//*[local-name()='li' or local-name()='LI']";3186 var result = document.evaluate(xpath, el, null,3187 XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);3189 isBuggy = (result.snapshotLength !== 2);3190 el = null;3191 }3192 return isBuggy;3193 })();3195 return function() {3196 if (!Prototype.BrowserFeatures.XPath) return false;3198 var e = this.expression;3200 if (Prototype.Browser.WebKit &&3201 (e.include("-of-type") || e.include(":empty")))3202 return false;3204 if ((/(\[[\w-]*?:|:checked)/).test(e))3205 return false;3207 if (IS_DESCENDANT_SELECTOR_BUGGY) return false;3209 return true;3210 }3212 })(),3214 shouldUseSelectorsAPI: function() {3215 if (!Prototype.BrowserFeatures.SelectorsAPI) return false;3217 if (Selector.CASE_INSENSITIVE_CLASS_NAMES) return false;3219 if (!Selector._div) Selector._div = new Element('div');3221 try {3222 Selector._div.querySelector(this.expression);3223 } catch(e) {3224 return false;3225 }3227 return true;3228 },3230 compileMatcher: function() {3231 var e = this.expression, ps = Selector.patterns, h = Selector.handlers,3232 c = Selector.criteria, le, p, m, len = ps.length, name;3234 if (Selector._cache[e]) {3235 this.matcher = Selector._cache[e];3236 return;3237 }3239 this.matcher = ["this.matcher = function(root) {",3240 "var r = root, h = Selector.handlers, c = false, n;"];3242 while (e && le != e && (/\S/).test(e)) {3243 le = e;3244 for (var i = 0; i<len; i++) {3245 p = ps[i].re;3246 name = ps[i].name;3247 if (m = e.match(p)) {3248 this.matcher.push(Object.isFunction(c[name]) ? c[name](m) :3249 new Template(c[name]).evaluate(m));3250 e = e.replace(m[0], '');3251 break;3252 }3253 }3254 }3256 this.matcher.push("return h.unique(n);\n}");3257 eval(this.matcher.join('\n'));3258 Selector._cache[this.expression] = this.matcher;3259 },3261 compileXPathMatcher: function() {3262 var e = this.expression, ps = Selector.patterns,3263 x = Selector.xpath, le, m, len = ps.length, name;3265 if (Selector._cache[e]) {3266 this.xpath = Selector._cache[e]; return;3267 }3269 this.matcher = ['.//*'];3270 while (e && le != e && (/\S/).test(e)) {3271 le = e;3272 for (var i = 0; i<len; i++) {3273 name = ps[i].name;3274 if (m = e.match(ps[i].re)) {3275 this.matcher.push(Object.isFunction(x[name]) ? x[name](m) :3276 new Template(x[name]).evaluate(m));3277 e = e.replace(m[0], '');3278 break;3279 }3280 }3281 }3283 this.xpath = this.matcher.join('');3284 Selector._cache[this.expression] = this.xpath;3285 },3287 findElements: function(root) {3288 root = root || document;3289 var e = this.expression, results;3291 switch (this.mode) {3292 case 'selectorsAPI':3293 if (root !== document) {3294 var oldId = root.id, id = $(root).identify();3295 id = id.replace(/([\.:])/g, "\\$1");3296 e = "#" + id + " " + e;3297 }3299 results = $A(root.querySelectorAll(e)).map(Element.extend);3300 root.id = oldId;3302 return results;3303 case 'xpath':3304 return document._getElementsByXPath(this.xpath, root);3305 default:3306 return this.matcher(root);3307 }3308 },3310 match: function(element) {3311 this.tokens = [];3313 var e = this.expression, ps = Selector.patterns, as = Selector.assertions;3314 var le, p, m, len = ps.length, name;3316 while (e && le !== e && (/\S/).test(e)) {3317 le = e;3318 for (var i = 0; i<len; i++) {3319 p = ps[i].re;3320 name = ps[i].name;3321 if (m = e.match(p)) {3322 if (as[name]) {3323 this.tokens.push([name, Object.clone(m)]);3324 e = e.replace(m[0], '');3325 } else {3326 return this.findElements(document).include(element);3327 }3328 }3329 }3330 }3332 var match = true, name, matches;3333 for (var i = 0, token; token = this.tokens[i]; i++) {3334 name = token[0], matches = token[1];3335 if (!Selector.assertions[name](element, matches)) {3336 match = false; break;3337 }3338 }3340 return match;3341 },3343 toString: function() {3344 return this.expression;3345 },3347 inspect: function() {3348 return "#<Selector:" + this.expression.inspect() + ">";3349 }3350 });3352 if (Prototype.BrowserFeatures.SelectorsAPI &&3353 document.compatMode === 'BackCompat') {3354 Selector.CASE_INSENSITIVE_CLASS_NAMES = (function(){3355 var div = document.createElement('div'),3356 span = document.createElement('span');3358 div.id = "prototype_test_id";3359 span.className = 'Test';3360 div.appendChild(span);3361 var isIgnored = (div.querySelector('#prototype_test_id .test') !== null);3362 div = span = null;3363 return isIgnored;3364 })();3365 }3367 Object.extend(Selector, {3368 _cache: { },3370 xpath: {3371 descendant: "//*",3372 child: "/*",3373 adjacent: "/following-sibling::*[1]",3374 laterSibling: '/following-sibling::*',3375 tagName: function(m) {3376 if (m[1] == '*') return '';3377 return "[local-name()='" + m[1].toLowerCase() +3378 "' or local-name()='" + m[1].toUpperCase() + "']";3379 },3380 className: "[contains(concat(' ', @class, ' '), ' #{1} ')]",3381 id: "[@id='#{1}']",3382 attrPresence: function(m) {3383 m[1] = m[1].toLowerCase();3384 return new Template("[@#{1}]").evaluate(m);3385 },3386 attr: function(m) {3387 m[1] = m[1].toLowerCase();3388 m[3] = m[5] || m[6];3389 return new Template(Selector.xpath.operators[m[2]]).evaluate(m);3390 },3391 pseudo: function(m) {3392 var h = Selector.xpath.pseudos[m[1]];3393 if (!h) return '';3394 if (Object.isFunction(h)) return h(m);3395 return new Template(Selector.xpath.pseudos[m[1]]).evaluate(m);3396 },3397 operators: {3398 '=': "[@#{1}='#{3}']",3399 '!=': "[@#{1}!='#{3}']",3400 '^=': "[starts-with(@#{1}, '#{3}')]",3401 '$=': "[substring(@#{1}, (string-length(@#{1}) - string-length('#{3}') + 1))='#{3}']",3402 '*=': "[contains(@#{1}, '#{3}')]",3403 '~=': "[contains(concat(' ', @#{1}, ' '), ' #{3} ')]",3404 '|=': "[contains(concat('-', @#{1}, '-'), '-#{3}-')]"3405 },3406 pseudos: {3407 'first-child': '[not(preceding-sibling::*)]',3408 'last-child': '[not(following-sibling::*)]',3409 'only-child': '[not(preceding-sibling::* or following-sibling::*)]',3410 'empty': "[count(*) = 0 and (count(text()) = 0)]",3411 'checked': "[@checked]",3412 'disabled': "[(@disabled) and (@type!='hidden')]",3413 'enabled': "[not(@disabled) and (@type!='hidden')]",3414 'not': function(m) {3415 var e = m[6], p = Selector.patterns,3416 x = Selector.xpath, le, v, len = p.length, name;3418 var exclusion = [];3419 while (e && le != e && (/\S/).test(e)) {3420 le = e;3421 for (var i = 0; i<len; i++) {3422 name = p[i].name3423 if (m = e.match(p[i].re)) {3424 v = Object.isFunction(x[name]) ? x[name](m) : new Template(x[name]).evaluate(m);3425 exclusion.push("(" + v.substring(1, v.length - 1) + ")");3426 e = e.replace(m[0], '');3427 break;3428 }3429 }3430 }3431 return "[not(" + exclusion.join(" and ") + ")]";3432 },3433 'nth-child': function(m) {3434 return Selector.xpath.pseudos.nth("(count(./preceding-sibling::*) + 1) ", m);3435 },3436 'nth-last-child': function(m) {3437 return Selector.xpath.pseudos.nth("(count(./following-sibling::*) + 1) ", m);3438 },3439 'nth-of-type': function(m) {3440 return Selector.xpath.pseudos.nth("position() ", m);3441 },3442 'nth-last-of-type': function(m) {3443 return Selector.xpath.pseudos.nth("(last() + 1 - position()) ", m);3444 },3445 'first-of-type': function(m) {3446 m[6] = "1"; return Selector.xpath.pseudos['nth-of-type'](m);3447 },3448 'last-of-type': function(m) {3449 m[6] = "1"; return Selector.xpath.pseudos['nth-last-of-type'](m);3450 },3451 'only-of-type': function(m) {3452 var p = Selector.xpath.pseudos; return p['first-of-type'](m) + p['last-of-type'](m);3453 },3454 nth: function(fragment, m) {3455 var mm, formula = m[6], predicate;3456 if (formula == 'even') formula = '2n+0';3457 if (formula == 'odd') formula = '2n+1';3458 if (mm = formula.match(/^(\d+)$/)) // digit only3459 return '[' + fragment + "= " + mm[1] + ']';3460 if (mm = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b3461 if (mm[1] == "-") mm[1] = -1;3462 var a = mm[1] ? Number(mm[1]) : 1;3463 var b = mm[2] ? Number(mm[2]) : 0;3464 predicate = "[((#{fragment} - #{b}) mod #{a} = 0) and " +3465 "((#{fragment} - #{b}) div #{a} >= 0)]";3466 return new Template(predicate).evaluate({3467 fragment: fragment, a: a, b: b });3468 }3469 }3470 }3471 },3473 criteria: {3474 tagName: 'n = h.tagName(n, r, "#{1}", c); c = false;',3475 className: 'n = h.className(n, r, "#{1}", c); c = false;',3476 id: 'n = h.id(n, r, "#{1}", c); c = false;',3477 attrPresence: 'n = h.attrPresence(n, r, "#{1}", c); c = false;',3478 attr: function(m) {3479 m[3] = (m[5] || m[6]);3480 return new Template('n = h.attr(n, r, "#{1}", "#{3}", "#{2}", c); c = false;').evaluate(m);3481 },3482 pseudo: function(m) {3483 if (m[6]) m[6] = m[6].replace(/"/g, '\\"');3484 return new Template('n = h.pseudo(n, "#{1}", "#{6}", r, c); c = false;').evaluate(m);3485 },3486 descendant: 'c = "descendant";',3487 child: 'c = "child";',3488 adjacent: 'c = "adjacent";',3489 laterSibling: 'c = "laterSibling";'3490 },3492 patterns: [3493 { name: 'laterSibling', re: /^\s*~\s*/ },3494 { name: 'child', re: /^\s*>\s*/ },3495 { name: 'adjacent', re: /^\s*\+\s*/ },3496 { name: 'descendant', re: /^\s/ },3498 { name: 'tagName', re: /^\s*(\*|[\w\-]+)(\b|$)?/ },3499 { name: 'id', re: /^#([\w\-\*]+)(\b|$)/ },3500 { name: 'className', re: /^\.([\w\-\*]+)(\b|$)/ },3501 { name: 'pseudo', re: /^:((first|last|nth|nth-last|only)(-child|-of-type)|empty|checked|(en|dis)abled|not)(\((.*?)\))?(\b|$|(?=\s|[:+~>]))/ },3502 { name: 'attrPresence', re: /^\[((?:[\w-]+:)?[\w-]+)\]/ },3503 { name: 'attr', re: /\[((?:[\w-]*:)?[\w-]+)\s*(?:([!^$*~|]?=)\s*((['"])([^\4]*?)\4|([^'"][^\]]*?)))?\]/ }3504 ],3506 assertions: {3507 tagName: function(element, matches) {3508 return matches[1].toUpperCase() == element.tagName.toUpperCase();3509 },3511 className: function(element, matches) {3512 return Element.hasClassName(element, matches[1]);3513 },3515 id: function(element, matches) {3516 return element.id === matches[1];3517 },3519 attrPresence: function(element, matches) {3520 return Element.hasAttribute(element, matches[1]);3521 },3523 attr: function(element, matches) {3524 var nodeValue = Element.readAttribute(element, matches[1]);3525 return nodeValue && Selector.operators[matches[2]](nodeValue, matches[5] || matches[6]);3526 }3527 },3529 handlers: {3530 concat: function(a, b) {3531 for (var i = 0, node; node = b[i]; i++)3532 a.push(node);3533 return a;3534 },3536 mark: function(nodes) {3537 var _true = Prototype.emptyFunction;3538 for (var i = 0, node; node = nodes[i]; i++)3539 node._countedByPrototype = _true;3540 return nodes;3541 },3543 unmark: (function(){3545 var PROPERTIES_ATTRIBUTES_MAP = (function(){3546 var el = document.createElement('div'),3547 isBuggy = false,3548 propName = '_countedByPrototype',3549 value = 'x'3550 el[propName] = value;3551 isBuggy = (el.getAttribute(propName) === value);3552 el = null;3553 return isBuggy;3554 })();3556 return PROPERTIES_ATTRIBUTES_MAP ?3557 function(nodes) {3558 for (var i = 0, node; node = nodes[i]; i++)3559 node.removeAttribute('_countedByPrototype');3560 return nodes;3561 } :3562 function(nodes) {3563 for (var i = 0, node; node = nodes[i]; i++)3564 node._countedByPrototype = void 0;3565 return nodes;3566 }3567 })(),3569 index: function(parentNode, reverse, ofType) {3570 parentNode._countedByPrototype = Prototype.emptyFunction;3571 if (reverse) {3572 for (var nodes = parentNode.childNodes, i = nodes.length - 1, j = 1; i >= 0; i--) {3573 var node = nodes[i];3574 if (node.nodeType == 1 && (!ofType || node._countedByPrototype)) node.nodeIndex = j++;3575 }3576 } else {3577 for (var i = 0, j = 1, nodes = parentNode.childNodes; node = nodes[i]; i++)3578 if (node.nodeType == 1 && (!ofType || node._countedByPrototype)) node.nodeIndex = j++;3579 }3580 },3582 unique: function(nodes) {3583 if (nodes.length == 0) return nodes;3584 var results = [], n;3585 for (var i = 0, l = nodes.length; i < l; i++)3586 if (typeof (n = nodes[i])._countedByPrototype == 'undefined') {3587 n._countedByPrototype = Prototype.emptyFunction;3588 results.push(Element.extend(n));3589 }3590 return Selector.handlers.unmark(results);3591 },3593 descendant: function(nodes) {3594 var h = Selector.handlers;3595 for (var i = 0, results = [], node; node = nodes[i]; i++)3596 h.concat(results, node.getElementsByTagName('*'));3597 return results;3598 },3600 child: function(nodes) {3601 var h = Selector.handlers;3602 for (var i = 0, results = [], node; node = nodes[i]; i++) {3603 for (var j = 0, child; child = node.childNodes[j]; j++)3604 if (child.nodeType == 1 && child.tagName != '!') results.push(child);3605 }3606 return results;3607 },3609 adjacent: function(nodes) {3610 for (var i = 0, results = [], node; node = nodes[i]; i++) {3611 var next = this.nextElementSibling(node);3612 if (next) results.push(next);3613 }3614 return results;3615 },3617 laterSibling: function(nodes) {3618 var h = Selector.handlers;3619 for (var i = 0, results = [], node; node = nodes[i]; i++)3620 h.concat(results, Element.nextSiblings(node));3621 return results;3622 },3624 nextElementSibling: function(node) {3625 while (node = node.nextSibling)3626 if (node.nodeType == 1) return node;3627 return null;3628 },3630 previousElementSibling: function(node) {3631 while (node = node.previousSibling)3632 if (node.nodeType == 1) return node;3633 return null;3634 },3636 tagName: function(nodes, root, tagName, combinator) {3637 var uTagName = tagName.toUpperCase();3638 var results = [], h = Selector.handlers;3639 if (nodes) {3640 if (combinator) {3641 if (combinator == "descendant") {3642 for (var i = 0, node; node = nodes[i]; i++)3643 h.concat(results, node.getElementsByTagName(tagName));3644 return results;3645 } else nodes = this[combinator](nodes);3646 if (tagName == "*") return nodes;3647 }3648 for (var i = 0, node; node = nodes[i]; i++)3649 if (node.tagName.toUpperCase() === uTagName) results.push(node);3650 return results;3651 } else return root.getElementsByTagName(tagName);3652 },3654 id: function(nodes, root, id, combinator) {3655 var targetNode = $(id), h = Selector.handlers;3657 if (root == document) {3658 if (!targetNode) return [];3659 if (!nodes) return [targetNode];3660 } else {3661 if (!root.sourceIndex || root.sourceIndex < 1) {3662 var nodes = root.getElementsByTagName('*');3663 for (var j = 0, node; node = nodes[j]; j++) {3664 if (node.id === id) return [node];3665 }3666 }3667 }3669 if (nodes) {3670 if (combinator) {3671 if (combinator == 'child') {3672 for (var i = 0, node; node = nodes[i]; i++)3673 if (targetNode.parentNode == node) return [targetNode];3674 } else if (combinator == 'descendant') {3675 for (var i = 0, node; node = nodes[i]; i++)3676 if (Element.descendantOf(targetNode, node)) return [targetNode];3677 } else if (combinator == 'adjacent') {3678 for (var i = 0, node; node = nodes[i]; i++)3679 if (Selector.handlers.previousElementSibling(targetNode) == node)3680 return [targetNode];3681 } else nodes = h[combinator](nodes);3682 }3683 for (var i = 0, node; node = nodes[i]; i++)3684 if (node == targetNode) return [targetNode];3685 return [];3686 }3687 return (targetNode && Element.descendantOf(targetNode, root)) ? [targetNode] : [];3688 },3690 className: function(nodes, root, className, combinator) {3691 if (nodes && combinator) nodes = this[combinator](nodes);3692 return Selector.handlers.byClassName(nodes, root, className);3693 },3695 byClassName: function(nodes, root, className) {3696 if (!nodes) nodes = Selector.handlers.descendant([root]);3697 var needle = ' ' + className + ' ';3698 for (var i = 0, results = [], node, nodeClassName; node = nodes[i]; i++) {3699 nodeClassName = node.className;3700 if (nodeClassName.length == 0) continue;3701 if (nodeClassName == className || (' ' + nodeClassName + ' ').include(needle))3702 results.push(node);3703 }3704 return results;3705 },3707 attrPresence: function(nodes, root, attr, combinator) {3708 if (!nodes) nodes = root.getElementsByTagName("*");3709 if (nodes && combinator) nodes = this[combinator](nodes);3710 var results = [];3711 for (var i = 0, node; node = nodes[i]; i++)3712 if (Element.hasAttribute(node, attr)) results.push(node);3713 return results;3714 },3716 attr: function(nodes, root, attr, value, operator, combinator) {3717 if (!nodes) nodes = root.getElementsByTagName("*");3718 if (nodes && combinator) nodes = this[combinator](nodes);3719 var handler = Selector.operators[operator], results = [];3720 for (var i = 0, node; node = nodes[i]; i++) {3721 var nodeValue = Element.readAttribute(node, attr);3722 if (nodeValue === null) continue;3723 if (handler(nodeValue, value)) results.push(node);3724 }3725 return results;3726 },3728 pseudo: function(nodes, name, value, root, combinator) {3729 if (nodes && combinator) nodes = this[combinator](nodes);3730 if (!nodes) nodes = root.getElementsByTagName("*");3731 return Selector.pseudos[name](nodes, value, root);3732 }3733 },3735 pseudos: {3736 'first-child': function(nodes, value, root) {3737 for (var i = 0, results = [], node; node = nodes[i]; i++) {3738 if (Selector.handlers.previousElementSibling(node)) continue;3739 results.push(node);3740 }3741 return results;3742 },3743 'last-child': function(nodes, value, root) {3744 for (var i = 0, results = [], node; node = nodes[i]; i++) {3745 if (Selector.handlers.nextElementSibling(node)) continue;3746 results.push(node);3747 }3748 return results;3749 },3750 'only-child': function(nodes, value, root) {3751 var h = Selector.handlers;3752 for (var i = 0, results = [], node; node = nodes[i]; i++)3753 if (!h.previousElementSibling(node) && !h.nextElementSibling(node))3754 results.push(node);3755 return results;3756 },3757 'nth-child': function(nodes, formula, root) {3758 return Selector.pseudos.nth(nodes, formula, root);3759 },3760 'nth-last-child': function(nodes, formula, root) {3761 return Selector.pseudos.nth(nodes, formula, root, true);3762 },3763 'nth-of-type': function(nodes, formula, root) {3764 return Selector.pseudos.nth(nodes, formula, root, false, true);3765 },3766 'nth-last-of-type': function(nodes, formula, root) {3767 return Selector.pseudos.nth(nodes, formula, root, true, true);3768 },3769 'first-of-type': function(nodes, formula, root) {3770 return Selector.pseudos.nth(nodes, "1", root, false, true);3771 },3772 'last-of-type': function(nodes, formula, root) {3773 return Selector.pseudos.nth(nodes, "1", root, true, true);3774 },3775 'only-of-type': function(nodes, formula, root) {3776 var p = Selector.pseudos;3777 return p['last-of-type'](p['first-of-type'](nodes, formula, root), formula, root);3778 },3780 getIndices: function(a, b, total) {3781 if (a == 0) return b > 0 ? [b] : [];3782 return $R(1, total).inject([], function(memo, i) {3783 if (0 == (i - b) % a && (i - b) / a >= 0) memo.push(i);3784 return memo;3785 });3786 },3788 nth: function(nodes, formula, root, reverse, ofType) {3789 if (nodes.length == 0) return [];3790 if (formula == 'even') formula = '2n+0';3791 if (formula == 'odd') formula = '2n+1';3792 var h = Selector.handlers, results = [], indexed = [], m;3793 h.mark(nodes);3794 for (var i = 0, node; node = nodes[i]; i++) {3795 if (!node.parentNode._countedByPrototype) {3796 h.index(node.parentNode, reverse, ofType);3797 indexed.push(node.parentNode);3798 }3799 }3800 if (formula.match(/^\d+$/)) { // just a number3801 formula = Number(formula);3802 for (var i = 0, node; node = nodes[i]; i++)3803 if (node.nodeIndex == formula) results.push(node);3804 } else if (m = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b3805 if (m[1] == "-") m[1] = -1;3806 var a = m[1] ? Number(m[1]) : 1;3807 var b = m[2] ? Number(m[2]) : 0;3808 var indices = Selector.pseudos.getIndices(a, b, nodes.length);3809 for (var i = 0, node, l = indices.length; node = nodes[i]; i++) {3810 for (var j = 0; j < l; j++)3811 if (node.nodeIndex == indices[j]) results.push(node);3812 }3813 }3814 h.unmark(nodes);3815 h.unmark(indexed);3816 return results;3817 },3819 'empty': function(nodes, value, root) {3820 for (var i = 0, results = [], node; node = nodes[i]; i++) {3821 if (node.tagName == '!' || node.firstChild) continue;3822 results.push(node);3823 }3824 return results;3825 },3827 'not': function(nodes, selector, root) {3828 var h = Selector.handlers, selectorType, m;3829 var exclusions = new Selector(selector).findElements(root);3830 h.mark(exclusions);3831 for (var i = 0, results = [], node; node = nodes[i]; i++)3832 if (!node._countedByPrototype) results.push(node);3833 h.unmark(exclusions);3834 return results;3835 },3837 'enabled': function(nodes, value, root) {3838 for (var i = 0, results = [], node; node = nodes[i]; i++)3839 if (!node.disabled && (!node.type || node.type !== 'hidden'))3840 results.push(node);3841 return results;3842 },3844 'disabled': function(nodes, value, root) {3845 for (var i = 0, results = [], node; node = nodes[i]; i++)3846 if (node.disabled) results.push(node);3847 return results;3848 },3850 'checked': function(nodes, value, root) {3851 for (var i = 0, results = [], node; node = nodes[i]; i++)3852 if (node.checked) results.push(node);3853 return results;3854 }3855 },3857 operators: {3858 '=': function(nv, v) { return nv == v; },3859 '!=': function(nv, v) { return nv != v; },3860 '^=': function(nv, v) { return nv == v || nv && nv.startsWith(v); },3861 '$=': function(nv, v) { return nv == v || nv && nv.endsWith(v); },3862 '*=': function(nv, v) { return nv == v || nv && nv.include(v); },3863 '~=': function(nv, v) { return (' ' + nv + ' ').include(' ' + v + ' '); },3864 '|=': function(nv, v) { return ('-' + (nv || "").toUpperCase() +3865 '-').include('-' + (v || "").toUpperCase() + '-'); }3866 },3868 split: function(expression) {3869 var expressions = [];3870 expression.scan(/(([\w#:.~>+()\s-]+|\*|\[.*?\])+)\s*(,|$)/, function(m) {3871 expressions.push(m[1].strip());3872 });3873 return expressions;3874 },3876 matchElements: function(elements, expression) {3877 var matches = $$(expression), h = Selector.handlers;3878 h.mark(matches);3879 for (var i = 0, results = [], element; element = elements[i]; i++)3880 if (element._countedByPrototype) results.push(element);3881 h.unmark(matches);3882 return results;3883 },3885 findElement: function(elements, expression, index) {3886 if (Object.isNumber(expression)) {3887 index = expression; expression = false;3888 }3889 return Selector.matchElements(elements, expression || '*')[index || 0];3890 },3892 findChildElements: function(element, expressions) {3893 expressions = Selector.split(expressions.join(','));3894 var results = [], h = Selector.handlers;3895 for (var i = 0, l = expressions.length, selector; i < l; i++) {3896 selector = new Selector(expressions[i].strip());3897 h.concat(results, selector.findElements(element));3898 }3899 return (l > 1) ? h.unique(results) : results;3900 }3901 });3903 if (Prototype.Browser.IE) {3904 Object.extend(Selector.handlers, {3905 concat: function(a, b) {3906 for (var i = 0, node; node = b[i]; i++)3907 if (node.tagName !== "!") a.push(node);3908 return a;3909 }3910 });3911 }3913 function $$() {3914 return Selector.findChildElements(document, $A(arguments));3915 }3917 var Form = {3918 reset: function(form) {3919 form = $(form);3920 form.reset();3921 return form;3922 },3924 serializeElements: function(elements, options) {3925 if (typeof options != 'object') options = { hash: !!options };3926 else if (Object.isUndefined(options.hash)) options.hash = true;3927 var key, value, submitted = false, submit = options.submit;3929 var data = elements.inject({ }, function(result, element) {3930 if (!element.disabled && element.name) {3931 key = element.name; value = $(element).getValue();3932 if (value != null && element.type != 'file' && (element.type != 'submit' || (!submitted &&3933 submit !== false && (!submit || key == submit) && (submitted = true)))) {3934 if (key in result) {3935 if (!Object.isArray(result[key])) result[key] = [result[key]];3936 result[key].push(value);3937 }3938 else result[key] = value;3939 }3940 }3941 return result;3942 });3944 return options.hash ? data : Object.toQueryString(data);3945 }3946 };3948 Form.Methods = {3949 serialize: function(form, options) {3950 return Form.serializeElements(Form.getElements(form), options);3951 },3953 getElements: function(form) {3954 var elements = $(form).getElementsByTagName('*'),3955 element,3956 arr = [ ],3957 serializers = Form.Element.Serializers;3958 for (var i = 0; element = elements[i]; i++) {3959 arr.push(element);3960 }3961 return arr.inject([], function(elements, child) {3962 if (serializers[child.tagName.toLowerCase()])3963 elements.push(Element.extend(child));3964 return elements;3965 })3966 },3968 getInputs: function(form, typeName, name) {3969 form = $(form);3970 var inputs = form.getElementsByTagName('input');3972 if (!typeName && !name) return $A(inputs).map(Element.extend);3974 for (var i = 0, matchingInputs = [], length = inputs.length; i < length; i++) {3975 var input = inputs[i];3976 if ((typeName && input.type != typeName) || (name && input.name != name))3977 continue;3978 matchingInputs.push(Element.extend(input));3979 }3981 return matchingInputs;3982 },3984 disable: function(form) {3985 form = $(form);3986 Form.getElements(form).invoke('disable');3987 return form;3988 },3990 enable: function(form) {3991 form = $(form);3992 Form.getElements(form).invoke('enable');3993 return form;3994 },3996 findFirstElement: function(form) {3997 var elements = $(form).getElements().findAll(function(element) {3998 return 'hidden' != element.type && !element.disabled;3999 });4000 var firstByIndex = elements.findAll(function(element) {4001 return element.hasAttribute('tabIndex') && element.tabIndex >= 0;4002 }).sortBy(function(element) { return element.tabIndex }).first();4004 return firstByIndex ? firstByIndex : elements.find(function(element) {4005 return /^(?:input|select|textarea)$/i.test(element.tagName);4006 });4007 },4009 focusFirstElement: function(form) {4010 form = $(form);4011 form.findFirstElement().activate();4012 return form;4013 },4015 request: function(form, options) {4016 form = $(form), options = Object.clone(options || { });4018 var params = options.parameters, action = form.readAttribute('action') || '';4019 if (action.blank()) action = window.location.href;4020 options.parameters = form.serialize(true);4022 if (params) {4023 if (Object.isString(params)) params = params.toQueryParams();4024 Object.extend(options.parameters, params);4025 }4027 if (form.hasAttribute('method') && !options.method)4028 options.method = form.method;4030 return new Ajax.Request(action, options);4031 }4032 };4034 /*--------------------------------------------------------------------------*/4037 Form.Element = {4038 focus: function(element) {4039 $(element).focus();4040 return element;4041 },4043 select: function(element) {4044 $(element).select();4045 return element;4046 }4047 };4049 Form.Element.Methods = {4051 serialize: function(element) {4052 element = $(element);4053 if (!element.disabled && element.name) {4054 var value = element.getValue();4055 if (value != undefined) {4056 var pair = { };4057 pair[element.name] = value;4058 return Object.toQueryString(pair);4059 }4060 }4061 return '';4062 },4064 getValue: function(element) {4065 element = $(element);4066 var method = element.tagName.toLowerCase();4067 return Form.Element.Serializers[method](element);4068 },4070 setValue: function(element, value) {4071 element = $(element);4072 var method = element.tagName.toLowerCase();4073 Form.Element.Serializers[method](element, value);4074 return element;4075 },4077 clear: function(element) {4078 $(element).value = '';4079 return element;4080 },4082 present: function(element) {4083 return $(element).value != '';4084 },4086 activate: function(element) {4087 element = $(element);4088 try {4089 element.focus();4090 if (element.select && (element.tagName.toLowerCase() != 'input' ||4091 !(/^(?:button|reset|submit)$/i.test(element.type))))4092 element.select();4093 } catch (e) { }4094 return element;4095 },4097 disable: function(element) {4098 element = $(element);4099 element.disabled = true;4100 return element;4101 },4103 enable: function(element) {4104 element = $(element);4105 element.disabled = false;4106 return element;4107 }4108 };4110 /*--------------------------------------------------------------------------*/4112 var Field = Form.Element;4114 var $F = Form.Element.Methods.getValue;4116 /*--------------------------------------------------------------------------*/4118 Form.Element.Serializers = {4119 input: function(element, value) {4120 switch (element.type.toLowerCase()) {4121 case 'checkbox':4122 case 'radio':4123 return Form.Element.Serializers.inputSelector(element, value);4124 default:4125 return Form.Element.Serializers.textarea(element, value);4126 }4127 },4129 inputSelector: function(element, value) {4130 if (Object.isUndefined(value)) return element.checked ? element.value : null;4131 else element.checked = !!value;4132 },4134 textarea: function(element, value) {4135 if (Object.isUndefined(value)) return element.value;4136 else element.value = value;4137 },4139 select: function(element, value) {4140 if (Object.isUndefined(value))4141 return this[element.type == 'select-one' ?4142 'selectOne' : 'selectMany'](element);4143 else {4144 var opt, currentValue, single = !Object.isArray(value);4145 for (var i = 0, length = element.length; i < length; i++) {4146 opt = element.options[i];4147 currentValue = this.optionValue(opt);4148 if (single) {4149 if (currentValue == value) {4150 opt.selected = true;4151 return;4152 }4153 }4154 else opt.selected = value.include(currentValue);4155 }4156 }4157 },4159 selectOne: function(element) {4160 var index = element.selectedIndex;4161 return index >= 0 ? this.optionValue(element.options[index]) : null;4162 },4164 selectMany: function(element) {4165 var values, length = element.length;4166 if (!length) return null;4168 for (var i = 0, values = []; i < length; i++) {4169 var opt = element.options[i];4170 if (opt.selected) values.push(this.optionValue(opt));4171 }4172 return values;4173 },4175 optionValue: function(opt) {4176 return Element.extend(opt).hasAttribute('value') ? opt.value : opt.text;4177 }4178 };4180 /*--------------------------------------------------------------------------*/4183 Abstract.TimedObserver = Class.create(PeriodicalExecuter, {4184 initialize: function($super, element, frequency, callback) {4185 $super(callback, frequency);4186 this.element = $(element);4187 this.lastValue = this.getValue();4188 },4190 execute: function() {4191 var value = this.getValue();4192 if (Object.isString(this.lastValue) && Object.isString(value) ?4193 this.lastValue != value : String(this.lastValue) != String(value)) {4194 this.callback(this.element, value);4195 this.lastValue = value;4196 }4197 }4198 });4200 Form.Element.Observer = Class.create(Abstract.TimedObserver, {4201 getValue: function() {4202 return Form.Element.getValue(this.element);4203 }4204 });4206 Form.Observer = Class.create(Abstract.TimedObserver, {4207 getValue: function() {4208 return Form.serialize(this.element);4209 }4210 });4212 /*--------------------------------------------------------------------------*/4214 Abstract.EventObserver = Class.create({4215 initialize: function(element, callback) {4216 this.element = $(element);4217 this.callback = callback;4219 this.lastValue = this.getValue();4220 if (this.element.tagName.toLowerCase() == 'form')4221 this.registerFormCallbacks();4222 else4223 this.registerCallback(this.element);4224 },4226 onElementEvent: function() {4227 var value = this.getValue();4228 if (this.lastValue != value) {4229 this.callback(this.element, value);4230 this.lastValue = value;4231 }4232 },4234 registerFormCallbacks: function() {4235 Form.getElements(this.element).each(this.registerCallback, this);4236 },4238 registerCallback: function(element) {4239 if (element.type) {4240 switch (element.type.toLowerCase()) {4241 case 'checkbox':4242 case 'radio':4243 Event.observe(element, 'click', this.onElementEvent.bind(this));4244 break;4245 default:4246 Event.observe(element, 'change', this.onElementEvent.bind(this));4247 break;4248 }4249 }4250 }4251 });4253 Form.Element.EventObserver = Class.create(Abstract.EventObserver, {4254 getValue: function() {4255 return Form.Element.getValue(this.element);4256 }4257 });4259 Form.EventObserver = Class.create(Abstract.EventObserver, {4260 getValue: function() {4261 return Form.serialize(this.element);4262 }4263 });4264 (function() {4266 var Event = {4267 KEY_BACKSPACE: 8,4268 KEY_TAB: 9,4269 KEY_RETURN: 13,4270 KEY_ESC: 27,4271 KEY_LEFT: 37,4272 KEY_UP: 38,4273 KEY_RIGHT: 39,4274 KEY_DOWN: 40,4275 KEY_DELETE: 46,4276 KEY_HOME: 36,4277 KEY_END: 35,4278 KEY_PAGEUP: 33,4279 KEY_PAGEDOWN: 34,4280 KEY_INSERT: 45,4282 cache: {}4283 };4285 var docEl = document.documentElement;4286 var MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED = 'onmouseenter' in docEl4287 && 'onmouseleave' in docEl;4289 var _isButton;4290 if (Prototype.Browser.IE) {4291 var buttonMap = { 0: 1, 1: 4, 2: 2 };4292 _isButton = function(event, code) {4293 return event.button === buttonMap[code];4294 };4295 } else if (Prototype.Browser.WebKit) {4296 _isButton = function(event, code) {4297 switch (code) {4298 case 0: return event.which == 1 && !event.metaKey;4299 case 1: return event.which == 1 && event.metaKey;4300 default: return false;4301 }4302 };4303 } else {4304 _isButton = function(event, code) {4305 return event.which ? (event.which === code + 1) : (event.button === code);4306 };4307 }4309 function isLeftClick(event) { return _isButton(event, 0) }4311 function isMiddleClick(event) { return _isButton(event, 1) }4313 function isRightClick(event) { return _isButton(event, 2) }4315 function element(event) {4316 event = Event.extend(event);4318 var node = event.target, type = event.type,4319 currentTarget = event.currentTarget;4321 if (currentTarget && currentTarget.tagName) {4322 if (type === 'load' || type === 'error' ||4323 (type === 'click' && currentTarget.tagName.toLowerCase() === 'input'4324 && currentTarget.type === 'radio'))4325 node = currentTarget;4326 }4328 if (node.nodeType == Node.TEXT_NODE)4329 node = node.parentNode;4331 return Element.extend(node);4332 }4334 function findElement(event, expression) {4335 var element = Event.element(event);4336 if (!expression) return element;4337 var elements = [element].concat(element.ancestors());4338 return Selector.findElement(elements, expression, 0);4339 }4341 function pointer(event) {4342 return { x: pointerX(event), y: pointerY(event) };4343 }4345 function pointerX(event) {4346 var docElement = document.documentElement,4347 body = document.body || { scrollLeft: 0 };4349 return event.pageX || (event.clientX +4350 (docElement.scrollLeft || body.scrollLeft) -4351 (docElement.clientLeft || 0));4352 }4354 function pointerY(event) {4355 var docElement = document.documentElement,4356 body = document.body || { scrollTop: 0 };4358 return event.pageY || (event.clientY +4359 (docElement.scrollTop || body.scrollTop) -4360 (docElement.clientTop || 0));4361 }4364 function stop(event) {4365 Event.extend(event);4366 event.preventDefault();4367 event.stopPropagation();4369 event.stopped = true;4370 }4372 Event.Methods = {4373 isLeftClick: isLeftClick,4374 isMiddleClick: isMiddleClick,4375 isRightClick: isRightClick,4377 element: element,4378 findElement: findElement,4380 pointer: pointer,4381 pointerX: pointerX,4382 pointerY: pointerY,4384 stop: stop4385 };4388 var methods = Object.keys(Event.Methods).inject({ }, function(m, name) {4389 m[name] = Event.Methods[name].methodize();4390 return m;4391 });4393 if (Prototype.Browser.IE) {4394 function _relatedTarget(event) {4395 var element;4396 switch (event.type) {4397 case 'mouseover': element = event.fromElement; break;4398 case 'mouseout': element = event.toElement; break;4399 default: return null;4400 }4401 return Element.extend(element);4402 }4404 Object.extend(methods, {4405 stopPropagation: function() { this.cancelBubble = true },4406 preventDefault: function() { this.returnValue = false },4407 inspect: function() { return '[object Event]' }4408 });4410 Event.extend = function(event, element) {4411 if (!event) return false;4412 if (event._extendedByPrototype) return event;4414 event._extendedByPrototype = Prototype.emptyFunction;4415 var pointer = Event.pointer(event);4417 Object.extend(event, {4418 target: event.srcElement || element,4419 relatedTarget: _relatedTarget(event),4420 pageX: pointer.x,4421 pageY: pointer.y4422 });4424 return Object.extend(event, methods);4425 };4426 } else {4427 Event.prototype = window.Event.prototype || document.createEvent('HTMLEvents').__proto__;4428 Object.extend(Event.prototype, methods);4429 Event.extend = Prototype.K;4430 }4432 function _createResponder(element, eventName, handler) {4433 var registry = Element.retrieve(element, 'prototype_event_registry');4435 if (Object.isUndefined(registry)) {4436 CACHE.push(element);4437 registry = Element.retrieve(element, 'prototype_event_registry', $H());4438 }4440 var respondersForEvent = registry.get(eventName);4441 if (Object.isUndefined(respondersForEvent)) {4442 respondersForEvent = [];4443 registry.set(eventName, respondersForEvent);4444 }4446 if (respondersForEvent.pluck('handler').include(handler)) return false;4448 var responder;4449 if (eventName.include(":")) {4450 responder = function(event) {4451 if (Object.isUndefined(event.eventName))4452 return false;4454 if (event.eventName !== eventName)4455 return false;4457 Event.extend(event, element);4458 handler.call(element, event);4459 };4460 } else {4461 if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED &&4462 (eventName === "mouseenter" || eventName === "mouseleave")) {4463 if (eventName === "mouseenter" || eventName === "mouseleave") {4464 responder = function(event) {4465 Event.extend(event, element);4467 var parent = event.relatedTarget;4468 while (parent && parent !== element) {4469 try { parent = parent.parentNode; }4470 catch(e) { parent = element; }4471 }4473 if (parent === element) return;4475 handler.call(element, event);4476 };4477 }4478 } else {4479 responder = function(event) {4480 Event.extend(event, element);4481 handler.call(element, event);4482 };4483 }4484 }4486 responder.handler = handler;4487 respondersForEvent.push(responder);4488 return responder;4489 }4491 function _destroyCache() {4492 for (var i = 0, length = CACHE.length; i < length; i++) {4493 Event.stopObserving(CACHE[i]);4494 CACHE[i] = null;4495 }4496 }4498 var CACHE = [];4500 if (Prototype.Browser.IE)4501 window.attachEvent('onunload', _destroyCache);4503 if (Prototype.Browser.WebKit)4504 window.addEventListener('unload', Prototype.emptyFunction, false);4507 var _getDOMEventName = Prototype.K;4509 if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED) {4510 _getDOMEventName = function(eventName) {4511 var translations = { mouseenter: "mouseover", mouseleave: "mouseout" };4512 return eventName in translations ? translations[eventName] : eventName;4513 };4514 }4516 function observe(element, eventName, handler) {4517 element = $(element);4519 var responder = _createResponder(element, eventName, handler);4521 if (!responder) return element;4523 if (eventName.include(':')) {4524 if (element.addEventListener)4525 element.addEventListener("dataavailable", responder, false);4526 else {4527 element.attachEvent("ondataavailable", responder);4528 element.attachEvent("onfilterchange", responder);4529 }4530 } else {4531 var actualEventName = _getDOMEventName(eventName);4533 if (element.addEventListener)4534 element.addEventListener(actualEventName, responder, false);4535 else4536 element.attachEvent("on" + actualEventName, responder);4537 }4539 return element;4540 }4542 function stopObserving(element, eventName, handler) {4543 element = $(element);4545 var registry = Element.retrieve(element, 'prototype_event_registry');4547 if (Object.isUndefined(registry)) return element;4549 if (eventName && !handler) {4550 var responders = registry.get(eventName);4552 if (Object.isUndefined(responders)) return element;4554 responders.each( function(r) {4555 Element.stopObserving(element, eventName, r.handler);4556 });4557 return element;4558 } else if (!eventName) {4559 registry.each( function(pair) {4560 var eventName = pair.key, responders = pair.value;4562 responders.each( function(r) {4563 Element.stopObserving(element, eventName, r.handler);4564 });4565 });4566 return element;4567 }4569 var responders = registry.get(eventName);4571 if (!responders) return;4573 var responder = responders.find( function(r) { return r.handler === handler; });4574 if (!responder) return element;4576 var actualEventName = _getDOMEventName(eventName);4578 if (eventName.include(':')) {4579 if (element.removeEventListener)4580 element.removeEventListener("dataavailable", responder, false);4581 else {4582 element.detachEvent("ondataavailable", responder);4583 element.detachEvent("onfilterchange", responder);4584 }4585 } else {4586 if (element.removeEventListener)4587 element.removeEventListener(actualEventName, responder, false);4588 else4589 element.detachEvent('on' + actualEventName, responder);4590 }4592 registry.set(eventName, responders.without(responder));4594 return element;4595 }4597 function fire(element, eventName, memo, bubble) {4598 element = $(element);4600 if (Object.isUndefined(bubble))4601 bubble = true;4603 if (element == document && document.createEvent && !element.dispatchEvent)4604 element = document.documentElement;4606 var event;4607 if (document.createEvent) {4608 event = document.createEvent('HTMLEvents');4609 event.initEvent('dataavailable', true, true);4610 } else {4611 event = document.createEventObject();4612 event.eventType = bubble ? 'ondataavailable' : 'onfilterchange';4613 }4615 event.eventName = eventName;4616 event.memo = memo || { };4618 if (document.createEvent)4619 element.dispatchEvent(event);4620 else4621 element.fireEvent(event.eventType, event);4623 return Event.extend(event);4624 }4627 Object.extend(Event, Event.Methods);4629 Object.extend(Event, {4630 fire: fire,4631 observe: observe,4632 stopObserving: stopObserving4633 });4635 Element.addMethods({4636 fire: fire,4638 observe: observe,4640 stopObserving: stopObserving4641 });4643 Object.extend(document, {4644 fire: fire.methodize(),4646 observe: observe.methodize(),4648 stopObserving: stopObserving.methodize(),4650 loaded: false4651 });4653 if (window.Event) Object.extend(window.Event, Event);4654 else window.Event = Event;4655 })();4657 (function() {4658 /* Support for the DOMContentLoaded event is based on work by Dan Webb,4659 Matthias Miller, Dean Edwards, John Resig, and Diego Perini. */4661 var timer;4663 function fireContentLoadedEvent() {4664 if (document.loaded) return;4665 if (timer) window.clearTimeout(timer);4666 document.loaded = true;4667 document.fire('dom:loaded');4668 }4670 function checkReadyState() {4671 if (document.readyState === 'complete') {4672 document.stopObserving('readystatechange', checkReadyState);4673 fireContentLoadedEvent();4674 }4675 }4677 function pollDoScroll() {4678 try { document.documentElement.doScroll('left'); }4679 catch(e) {4680 timer = pollDoScroll.defer();4681 return;4682 }4683 fireContentLoadedEvent();4684 }4686 if (document.addEventListener) {4687 document.addEventListener('DOMContentLoaded', fireContentLoadedEvent, false);4688 } else {4689 document.observe('readystatechange', checkReadyState);4690 if (window == top)4691 timer = pollDoScroll.defer();4692 }4694 Event.observe(window, 'load', fireContentLoadedEvent);4695 })();4697 Element.addMethods();4699 /*------------------------------- DEPRECATED -------------------------------*/4701 Hash.toQueryString = Object.toQueryString;4703 var Toggle = { display: Element.toggle };4705 Element.Methods.childOf = Element.Methods.descendantOf;4707 var Insertion = {4708 Before: function(element, content) {4709 return Element.insert(element, {before:content});4710 },4712 Top: function(element, content) {4713 return Element.insert(element, {top:content});4714 },4716 Bottom: function(element, content) {4717 return Element.insert(element, {bottom:content});4718 },4720 After: function(element, content) {4721 return Element.insert(element, {after:content});4722 }4723 };4725 var $continue = new Error('"throw $continue" is deprecated, use "return" instead');4727 var Position = {4728 includeScrollOffsets: false,4730 prepare: function() {4731 this.deltaX = window.pageXOffset4732 || document.documentElement.scrollLeft4733 || document.body.scrollLeft4734 || 0;4735 this.deltaY = window.pageYOffset4736 || document.documentElement.scrollTop4737 || document.body.scrollTop4738 || 0;4739 },4741 within: function(element, x, y) {4742 if (this.includeScrollOffsets)4743 return this.withinIncludingScrolloffsets(element, x, y);4744 this.xcomp = x;4745 this.ycomp = y;4746 this.offset = Element.cumulativeOffset(element);4748 return (y >= this.offset[1] &&4749 y < this.offset[1] + element.offsetHeight &&4750 x >= this.offset[0] &&4751 x < this.offset[0] + element.offsetWidth);4752 },4754 withinIncludingScrolloffsets: function(element, x, y) {4755 var offsetcache = Element.cumulativeScrollOffset(element);4757 this.xcomp = x + offsetcache[0] - this.deltaX;4758 this.ycomp = y + offsetcache[1] - this.deltaY;4759 this.offset = Element.cumulativeOffset(element);4761 return (this.ycomp >= this.offset[1] &&4762 this.ycomp < this.offset[1] + element.offsetHeight &&4763 this.xcomp >= this.offset[0] &&4764 this.xcomp < this.offset[0] + element.offsetWidth);4765 },4767 overlap: function(mode, element) {4768 if (!mode) return 0;4769 if (mode == 'vertical')4770 return ((this.offset[1] + element.offsetHeight) - this.ycomp) /4771 element.offsetHeight;4772 if (mode == 'horizontal')4773 return ((this.offset[0] + element.offsetWidth) - this.xcomp) /4774 element.offsetWidth;4775 },4778 cumulativeOffset: Element.Methods.cumulativeOffset,4780 positionedOffset: Element.Methods.positionedOffset,4782 absolutize: function(element) {4783 Position.prepare();4784 return Element.absolutize(element);4785 },4787 relativize: function(element) {4788 Position.prepare();4789 return Element.relativize(element);4790 },4792 realOffset: Element.Methods.cumulativeScrollOffset,4794 offsetParent: Element.Methods.getOffsetParent,4796 page: Element.Methods.viewportOffset,4798 clone: function(source, target, options) {4799 options = options || { };4800 return Element.clonePosition(target, source, options);4801 }4802 };4804 /*--------------------------------------------------------------------------*/4806 if (!document.getElementsByClassName) document.getElementsByClassName = function(instanceMethods){4807 function iter(name) {4808 return name.blank() ? null : "[contains(concat(' ', @class, ' '), ' " + name + " ')]";4809 }4811 instanceMethods.getElementsByClassName = Prototype.BrowserFeatures.XPath ?4812 function(element, className) {4813 className = className.toString().strip();4814 var cond = /\s/.test(className) ? $w(className).map(iter).join('') : iter(className);4815 return cond ? document._getElementsByXPath('.//*' + cond, element) : [];4816 } : function(element, className) {4817 className = className.toString().strip();4818 var elements = [], classNames = (/\s/.test(className) ? $w(className) : null);4819 if (!classNames && !className) return elements;4821 var nodes = $(element).getElementsByTagName('*');4822 className = ' ' + className + ' ';4824 for (var i = 0, child, cn; child = nodes[i]; i++) {4825 if (child.className && (cn = ' ' + child.className + ' ') && (cn.include(className) ||4826 (classNames && classNames.all(function(name) {4827 return !name.toString().blank() && cn.include(' ' + name + ' ');4828 }))))4829 elements.push(Element.extend(child));4830 }4831 return elements;4832 };4834 return function(className, parentElement) {4835 return $(parentElement || document.body).getElementsByClassName(className);4836 };4837 }(Element.Methods);4839 /*--------------------------------------------------------------------------*/4841 Element.ClassNames = Class.create();4842 Element.ClassNames.prototype = {4843 initialize: function(element) {4844 this.element = $(element);4845 },4847 _each: function(iterator) {4848 this.element.className.split(/\s+/).select(function(name) {4849 return name.length > 0;4850 })._each(iterator);4851 },4853 set: function(className) {4854 this.element.className = className;4855 },4857 add: function(classNameToAdd) {4858 if (this.include(classNameToAdd)) return;4859 this.set($A(this).concat(classNameToAdd).join(' '));4860 },4862 remove: function(classNameToRemove) {4863 if (!this.include(classNameToRemove)) return;4864 this.set($A(this).without(classNameToRemove).join(' '));4865 },4867 toString: function() {4868 return $A(this).join(' ');4869 }4870 };4872 Object.extend(Element.ClassNames.prototype, Enumerable);4874 /*--------------------------------------------------------------------------*/