WordPress.org

Make WordPress Core

Ticket #4265: 4265.diff

File 4265.diff, 65.2 KB (added by rob1n, 8 years ago)
  • wp-includes/js/prototype.js

     
    1 /*  Prototype JavaScript framework, version 1.5.0 
     1/*  Prototype JavaScript framework, version 1.5.1 
    22 *  (c) 2005-2007 Sam Stephenson 
    33 * 
    44 *  Prototype is freely distributable under the terms of an MIT-style license. 
    5  *  For details, see the Prototype web site: http://prototype.conio.net/ 
     5 *  For details, see the Prototype web site: http://www.prototypejs.org/ 
    66 * 
    77/*--------------------------------------------------------------------------*/ 
    88 
    99var Prototype = { 
    10   Version: '1.5.0', 
     10  Version: '1.5.1', 
     11 
     12  Browser: { 
     13    IE:     !!(window.attachEvent && !window.opera), 
     14    Opera:  !!window.opera, 
     15    WebKit: navigator.userAgent.indexOf('AppleWebKit/') > -1, 
     16    Gecko:  navigator.userAgent.indexOf('Gecko') > -1 && navigator.userAgent.indexOf('KHTML') == -1 
     17  }, 
     18 
    1119  BrowserFeatures: { 
    12     XPath: !!document.evaluate 
     20    XPath: !!document.evaluate, 
     21    ElementExtensions: !!window.HTMLElement, 
     22    SpecificElementExtensions: 
     23      (document.createElement('div').__proto__ !== 
     24       document.createElement('form').__proto__) 
    1325  }, 
    1426 
    15   ScriptFragment: '(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)', 
    16   emptyFunction: function() {}, 
     27  ScriptFragment: '<script[^>]*>([\u0001-\uFFFF]*?)</script>', 
     28  JSONFilter: /^\/\*-secure-\s*(.*)\s*\*\/\s*$/, 
     29 
     30  emptyFunction: function() { }, 
    1731  K: function(x) { return x } 
    1832} 
    1933 
     
    4660    } 
    4761  }, 
    4862 
     63  toJSON: function(object) { 
     64    var type = typeof object; 
     65    switch(type) { 
     66      case 'undefined': 
     67      case 'function': 
     68      case 'unknown': return; 
     69      case 'boolean': return object.toString(); 
     70    } 
     71    if (object === null) return 'null'; 
     72    if (object.toJSON) return object.toJSON(); 
     73    if (object.ownerDocument === document) return; 
     74    var results = []; 
     75    for (var property in object) { 
     76      var value = Object.toJSON(object[property]); 
     77      if (value !== undefined) 
     78        results.push(property.toJSON() + ': ' + value); 
     79    } 
     80    return '{' + results.join(', ') + '}'; 
     81  }, 
     82 
    4983  keys: function(object) { 
    5084    var keys = []; 
    5185    for (var property in object) 
     
    75109Function.prototype.bindAsEventListener = function(object) { 
    76110  var __method = this, args = $A(arguments), object = args.shift(); 
    77111  return function(event) { 
    78     return __method.apply(object, [( event || window.event)].concat(args).concat($A(arguments))); 
     112    return __method.apply(object, [event || window.event].concat(args)); 
    79113  } 
    80114} 
    81115 
    82116Object.extend(Number.prototype, { 
    83117  toColorPart: function() { 
    84     var digits = this.toString(16); 
    85     if (this < 16) return '0' + digits; 
    86     return digits; 
     118    return this.toPaddedString(2, 16); 
    87119  }, 
    88120 
    89121  succ: function() { 
     
    93125  times: function(iterator) { 
    94126    $R(0, this, true).each(iterator); 
    95127    return this; 
     128  }, 
     129 
     130  toPaddedString: function(length, radix) { 
     131    var string = this.toString(radix || 10); 
     132    return '0'.times(length - string.length) + string; 
     133  }, 
     134 
     135  toJSON: function() { 
     136    return isFinite(this) ? this.toString() : 'null'; 
    96137  } 
    97138}); 
    98139 
     140Date.prototype.toJSON = function() { 
     141  return '"' + this.getFullYear() + '-' + 
     142    (this.getMonth() + 1).toPaddedString(2) + '-' + 
     143    this.getDate().toPaddedString(2) + 'T' + 
     144    this.getHours().toPaddedString(2) + ':' + 
     145    this.getMinutes().toPaddedString(2) + ':' + 
     146    this.getSeconds().toPaddedString(2) + '"'; 
     147}; 
     148 
    99149var Try = { 
    100150  these: function() { 
    101151    var returnValue; 
     
    145195    } 
    146196  } 
    147197} 
    148 String.interpret = function(value){ 
    149   return value == null ? '' : String(value); 
    150 } 
     198Object.extend(String, { 
     199  interpret: function(value) { 
     200    return value == null ? '' : String(value); 
     201  }, 
     202  specialChar: { 
     203    '\b': '\\b', 
     204    '\t': '\\t', 
     205    '\n': '\\n', 
     206    '\f': '\\f', 
     207    '\r': '\\r', 
     208    '\\': '\\\\' 
     209  } 
     210}); 
    151211 
    152212Object.extend(String.prototype, { 
    153213  gsub: function(pattern, replacement) { 
     
    213273  }, 
    214274 
    215275  escapeHTML: function() { 
    216     var div = document.createElement('div'); 
    217     var text = document.createTextNode(this); 
    218     div.appendChild(text); 
    219     return div.innerHTML; 
     276    var self = arguments.callee; 
     277    self.text.data = this; 
     278    return self.div.innerHTML; 
    220279  }, 
    221280 
    222281  unescapeHTML: function() { 
    223282    var div = document.createElement('div'); 
    224283    div.innerHTML = this.stripTags(); 
    225284    return div.childNodes[0] ? (div.childNodes.length > 1 ? 
    226       $A(div.childNodes).inject('',function(memo,node){ return memo+node.nodeValue }) : 
     285      $A(div.childNodes).inject('', function(memo, node) { return memo+node.nodeValue }) : 
    227286      div.childNodes[0].nodeValue) : ''; 
    228287  }, 
    229288 
     
    233292 
    234293    return match[1].split(separator || '&').inject({}, function(hash, pair) { 
    235294      if ((pair = pair.split('='))[0]) { 
    236         var name = decodeURIComponent(pair[0]); 
    237         var value = pair[1] ? decodeURIComponent(pair[1]) : undefined; 
     295        var key = decodeURIComponent(pair.shift()); 
     296        var value = pair.length > 1 ? pair.join('=') : pair[0]; 
     297        if (value != undefined) value = decodeURIComponent(value); 
    238298 
    239         if (hash[name] !== undefined) { 
    240           if (hash[name].constructor != Array) 
    241             hash[name] = [hash[name]]; 
    242           if (value) hash[name].push(value); 
     299        if (key in hash) { 
     300          if (hash[key].constructor != Array) hash[key] = [hash[key]]; 
     301          hash[key].push(value); 
    243302        } 
    244         else hash[name] = value; 
     303        else hash[key] = value; 
    245304      } 
    246305      return hash; 
    247306    }); 
     
    256315      String.fromCharCode(this.charCodeAt(this.length - 1) + 1); 
    257316  }, 
    258317 
     318  times: function(count) { 
     319    var result = ''; 
     320    for (var i = 0; i < count; i++) result += this; 
     321    return result; 
     322  }, 
     323 
    259324  camelize: function() { 
    260325    var parts = this.split('-'), len = parts.length; 
    261326    if (len == 1) return parts[0]; 
     
    270335    return camelized; 
    271336  }, 
    272337 
    273   capitalize: function(){ 
     338  capitalize: function() { 
    274339    return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase(); 
    275340  }, 
    276341 
     
    283348  }, 
    284349 
    285350  inspect: function(useDoubleQuotes) { 
    286     var escapedString = this.replace(/\\/g, '\\\\'); 
    287     if (useDoubleQuotes) 
    288       return '"' + escapedString.replace(/"/g, '\\"') + '"'; 
    289     else 
    290       return "'" + escapedString.replace(/'/g, '\\\'') + "'"; 
     351    var escapedString = this.gsub(/[\x00-\x1f\\]/, function(match) { 
     352      var character = String.specialChar[match[0]]; 
     353      return character ? character : '\\u00' + match[0].charCodeAt().toPaddedString(2, 16); 
     354    }); 
     355    if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"'; 
     356    return "'" + escapedString.replace(/'/g, '\\\'') + "'"; 
     357  }, 
     358 
     359  toJSON: function() { 
     360    return this.inspect(true); 
     361  }, 
     362 
     363  unfilterJSON: function(filter) { 
     364    return this.sub(filter || Prototype.JSONFilter, '#{1}'); 
     365  }, 
     366 
     367  evalJSON: function(sanitize) { 
     368    var json = this.unfilterJSON(); 
     369    try { 
     370      if (!sanitize || (/^("(\\.|[^"\\\n\r])*?"|[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t])+?$/.test(json))) 
     371        return eval('(' + json + ')'); 
     372    } catch (e) { } 
     373    throw new SyntaxError('Badly formed JSON string: ' + this.inspect()); 
     374  }, 
     375 
     376  include: function(pattern) { 
     377    return this.indexOf(pattern) > -1; 
     378  }, 
     379 
     380  startsWith: function(pattern) { 
     381    return this.indexOf(pattern) === 0; 
     382  }, 
     383 
     384  endsWith: function(pattern) { 
     385    var d = this.length - pattern.length; 
     386    return d >= 0 && this.lastIndexOf(pattern) === d; 
     387  }, 
     388 
     389  empty: function() { 
     390    return this == ''; 
     391  }, 
     392 
     393  blank: function() { 
     394    return /^\s*$/.test(this); 
    291395  } 
    292396}); 
    293397 
     398if (Prototype.Browser.WebKit || Prototype.Browser.IE) Object.extend(String.prototype, { 
     399  escapeHTML: function() { 
     400    return this.replace(/&/g,'&amp;').replace(/</g,'&lt;').replace(/>/g,'&gt;'); 
     401  }, 
     402  unescapeHTML: function() { 
     403    return this.replace(/&amp;/g,'&').replace(/&lt;/g,'<').replace(/&gt;/g,'>'); 
     404  } 
     405}); 
     406 
    294407String.prototype.gsub.prepareReplacement = function(replacement) { 
    295408  if (typeof replacement == 'function') return replacement; 
    296409  var template = new Template(replacement); 
     
    299412 
    300413String.prototype.parseQuery = String.prototype.toQueryParams; 
    301414 
     415Object.extend(String.prototype.escapeHTML, { 
     416  div:  document.createElement('div'), 
     417  text: document.createTextNode('') 
     418}); 
     419 
     420with (String.prototype.escapeHTML) div.appendChild(text); 
     421 
    302422var Template = Class.create(); 
    303423Template.Pattern = /(^|.|\r|\n)(#\{(.*?)\})/; 
    304424Template.prototype = { 
     
    316436  } 
    317437} 
    318438 
    319 var $break    = new Object(); 
    320 var $continue = new Object(); 
     439var $break = {}, $continue = new Error('"throw $continue" is deprecated, use "return" instead'); 
    321440 
    322441var Enumerable = { 
    323442  each: function(iterator) { 
    324443    var index = 0; 
    325444    try { 
    326445      this._each(function(value) { 
    327         try { 
    328           iterator(value, index++); 
    329         } catch (e) { 
    330           if (e != $continue) throw e; 
    331         } 
     446        iterator(value, index++); 
    332447      }); 
    333448    } catch (e) { 
    334449      if (e != $break) throw e; 
     
    530645  } 
    531646} 
    532647 
     648if (Prototype.Browser.WebKit) { 
     649  $A = Array.from = function(iterable) { 
     650    if (!iterable) return []; 
     651    if (!(typeof iterable == 'function' && iterable == '[object NodeList]') && 
     652      iterable.toArray) { 
     653      return iterable.toArray(); 
     654    } else { 
     655      var results = []; 
     656      for (var i = 0, length = iterable.length; i < length; i++) 
     657        results.push(iterable[i]); 
     658      return results; 
     659    } 
     660  } 
     661} 
     662 
    533663Object.extend(Array.prototype, Enumerable); 
    534664 
    535665if (!Array.prototype._reverse) 
     
    588718    return this.length > 1 ? this : this[0]; 
    589719  }, 
    590720 
    591   uniq: function() { 
    592     return this.inject([], function(array, value) { 
    593       return array.include(value) ? array : array.concat([value]); 
     721  uniq: function(sorted) { 
     722    return this.inject([], function(array, value, index) { 
     723      if (0 == index || (sorted ? array.last() != value : !array.include(value))) 
     724        array.push(value); 
     725      return array; 
    594726    }); 
    595727  }, 
    596728 
     
    604736 
    605737  inspect: function() { 
    606738    return '[' + this.map(Object.inspect).join(', ') + ']'; 
     739  }, 
     740 
     741  toJSON: function() { 
     742    var results = []; 
     743    this.each(function(object) { 
     744      var value = Object.toJSON(object); 
     745      if (value !== undefined) results.push(value); 
     746    }); 
     747    return '[' + results.join(', ') + ']'; 
    607748  } 
    608749}); 
    609750 
    610751Array.prototype.toArray = Array.prototype.clone; 
    611752 
    612 function $w(string){ 
     753function $w(string) { 
    613754  string = string.strip(); 
    614755  return string ? string.split(/\s+/) : []; 
    615756} 
    616757 
    617 if(window.opera){ 
    618   Array.prototype.concat = function(){ 
     758if (Prototype.Browser.Opera){ 
     759  Array.prototype.concat = function() { 
    619760    var array = []; 
    620     for(var i = 0, length = this.length; i < length; i++) array.push(this[i]); 
    621     for(var i = 0, length = arguments.length; i < length; i++) { 
    622       if(arguments[i].constructor == Array) { 
    623         for(var j = 0, arrayLength = arguments[i].length; j < arrayLength; j++) 
     761    for (var i = 0, length = this.length; i < length; i++) array.push(this[i]); 
     762    for (var i = 0, length = arguments.length; i < length; i++) { 
     763      if (arguments[i].constructor == Array) { 
     764        for (var j = 0, arrayLength = arguments[i].length; j < arrayLength; j++) 
    624765          array.push(arguments[i][j]); 
    625766      } else { 
    626767        array.push(arguments[i]); 
     
    629770    return array; 
    630771  } 
    631772} 
    632 var Hash = function(obj) { 
    633   Object.extend(this, obj || {}); 
     773var Hash = function(object) { 
     774  if (object instanceof Hash) this.merge(object); 
     775  else Object.extend(this, object || {}); 
    634776}; 
    635777 
    636778Object.extend(Hash, { 
    637779  toQueryString: function(obj) { 
    638780    var parts = []; 
     781    parts.add = arguments.callee.addPair; 
    639782 
    640           this.prototype._each.call(obj, function(pair) { 
     783    this.prototype._each.call(obj, function(pair) { 
    641784      if (!pair.key) return; 
     785      var value = pair.value; 
    642786 
    643       if (pair.value && pair.value.constructor == Array) { 
    644         var values = pair.value.compact(); 
    645         if (values.length < 2) pair.value = values.reduce(); 
    646         else { 
    647                 key = encodeURIComponent(pair.key); 
    648           values.each(function(value) { 
    649             value = value != undefined ? encodeURIComponent(value) : ''; 
    650             parts.push(key + '=' + encodeURIComponent(value)); 
    651           }); 
    652           return; 
    653         } 
     787      if (value && typeof value == 'object') { 
     788        if (value.constructor == Array) value.each(function(value) { 
     789          parts.add(pair.key, value); 
     790        }); 
     791        return; 
    654792      } 
    655       if (pair.value == undefined) pair[1] = ''; 
    656       parts.push(pair.map(encodeURIComponent).join('=')); 
    657           }); 
     793      parts.add(pair.key, value); 
     794    }); 
    658795 
    659796    return parts.join('&'); 
     797  }, 
     798 
     799  toJSON: function(object) { 
     800    var results = []; 
     801    this.prototype._each.call(object, function(pair) { 
     802      var value = Object.toJSON(pair.value); 
     803      if (value !== undefined) results.push(pair.key.toJSON() + ': ' + value); 
     804    }); 
     805    return '{' + results.join(', ') + '}'; 
    660806  } 
    661807}); 
    662808 
     809Hash.toQueryString.addPair = function(key, value, prefix) { 
     810  key = encodeURIComponent(key); 
     811  if (value === undefined) this.push(key); 
     812  else this.push(key + '=' + (value == null ? '' : encodeURIComponent(value))); 
     813} 
     814 
    663815Object.extend(Hash.prototype, Enumerable); 
    664816Object.extend(Hash.prototype, { 
    665817  _each: function(iterator) { 
     
    713865    return '#<Hash:{' + this.map(function(pair) { 
    714866      return pair.map(Object.inspect).join(': '); 
    715867    }).join(', ') + '}>'; 
     868  }, 
     869 
     870  toJSON: function() { 
     871    return Hash.toJSON(this); 
    716872  } 
    717873}); 
    718874 
    719875function $H(object) { 
    720   if (object && object.constructor == Hash) return object; 
     876  if (object instanceof Hash) return object; 
    721877  return new Hash(object); 
    722878}; 
     879 
     880// Safari iterates over shadowed properties 
     881if (function() { 
     882  var i = 0, Test = function(value) { this.key = value }; 
     883  Test.prototype.key = 'foo'; 
     884  for (var property in new Test('bar')) i++; 
     885  return i > 1; 
     886}()) Hash.prototype._each = function(iterator) { 
     887  var cache = []; 
     888  for (var key in this) { 
     889    var value = this[key]; 
     890    if ((value && value == Hash.prototype[key]) || cache.include(key)) continue; 
     891    cache.push(key); 
     892    var pair = [key, value]; 
     893    pair.key = key; 
     894    pair.value = value; 
     895    iterator(pair); 
     896  } 
     897}; 
    723898ObjectRange = Class.create(); 
    724899Object.extend(ObjectRange.prototype, Enumerable); 
    725900Object.extend(ObjectRange.prototype, { 
     
    8341009  request: function(url) { 
    8351010    this.url = url; 
    8361011    this.method = this.options.method; 
    837     var params = this.options.parameters; 
     1012    var params = Object.clone(this.options.parameters); 
    8381013 
    8391014    if (!['get', 'post'].include(this.method)) { 
    8401015      // simulate other verbs over post 
     
    8421017      this.method = 'post'; 
    8431018    } 
    8441019 
    845     params = Hash.toQueryString(params); 
    846     if (params && /Konqueror|Safari|KHTML/.test(navigator.userAgent)) params += '&_=' 
     1020    this.parameters = params; 
    8471021 
    848     // when GET, append parameters to URL 
    849     if (this.method == 'get' && params) 
    850       this.url += (this.url.indexOf('?') > -1 ? '&' : '?') + params; 
     1022    if (params = Hash.toQueryString(params)) { 
     1023      // when GET, append parameters to URL 
     1024      if (this.method == 'get') 
     1025        this.url += (this.url.include('?') ? '&' : '?') + params; 
     1026      else if (/Konqueror|Safari|KHTML/.test(navigator.userAgent)) 
     1027        params += '&_='; 
     1028    } 
    8511029 
    8521030    try { 
     1031      if (this.options.onCreate) this.options.onCreate(this.transport); 
    8531032      Ajax.Responders.dispatch('onCreate', this, this.transport); 
    8541033 
    8551034      this.transport.open(this.method.toUpperCase(), this.url, 
     
    8611040      this.transport.onreadystatechange = this.onStateChange.bind(this); 
    8621041      this.setRequestHeaders(); 
    8631042 
    864       var body = this.method == 'post' ? (this.options.postBody || params) : null; 
     1043      this.body = this.method == 'post' ? (this.options.postBody || params) : null; 
     1044      this.transport.send(this.body); 
    8651045 
    866       this.transport.send(body); 
    867  
    8681046      /* Force Firefox to handle ready state 4 for synchronous requests */ 
    8691047      if (!this.options.asynchronous && this.transport.overrideMimeType) 
    8701048        this.onStateChange(); 
     
    9351113        this.dispatchException(e); 
    9361114      } 
    9371115 
    938       if ((this.getHeader('Content-type') || 'text/javascript').strip(). 
     1116      var contentType = this.getHeader('Content-type'); 
     1117      if (contentType && contentType.strip(). 
    9391118        match(/^(text|application)\/(x-)?(java|ecma)script(;.*)?$/i)) 
    9401119          this.evalResponse(); 
    9411120    } 
     
    9621141  evalJSON: function() { 
    9631142    try { 
    9641143      var json = this.getHeader('X-JSON'); 
    965       return json ? eval('(' + json + ')') : null; 
     1144      return json ? json.evalJSON() : null; 
    9661145    } catch (e) { return null } 
    9671146  }, 
    9681147 
    9691148  evalResponse: function() { 
    9701149    try { 
    971       return eval(this.transport.responseText); 
     1150      return eval((this.transport.responseText || '').unfilterJSON()); 
    9721151    } catch (e) { 
    9731152      this.dispatchException(e); 
    9741153    } 
     
    10831262      results.push(query.snapshotItem(i)); 
    10841263    return results; 
    10851264  }; 
    1086 } 
    10871265 
    1088 document.getElementsByClassName = function(className, parentElement) { 
    1089   if (Prototype.BrowserFeatures.XPath) { 
     1266  document.getElementsByClassName = function(className, parentElement) { 
    10901267    var q = ".//*[contains(concat(' ', @class, ' '), ' " + className + " ')]"; 
    10911268    return document._getElementsByXPath(q, parentElement); 
    1092   } else { 
    1093     var children = ($(parentElement) || document.body).getElementsByTagName('*'); 
    1094     var elements = [], child; 
    1095     for (var i = 0, length = children.length; i < length; i++) { 
    1096       child = children[i]; 
    1097       if (Element.hasClassName(child, className)) 
    1098         elements.push(Element.extend(child)); 
    1099     } 
    1100     return elements; 
    11011269  } 
     1270 
     1271} else document.getElementsByClassName = function(className, parentElement) { 
     1272  var children = ($(parentElement) || document.body).getElementsByTagName('*'); 
     1273  var elements = [], child; 
     1274  for (var i = 0, length = children.length; i < length; i++) { 
     1275    child = children[i]; 
     1276    if (Element.hasClassName(child, className)) 
     1277      elements.push(Element.extend(child)); 
     1278  } 
     1279  return elements; 
    11021280}; 
    11031281 
    11041282/*--------------------------------------------------------------------------*/ 
    11051283 
    1106 if (!window.Element) 
    1107   var Element = new Object(); 
     1284if (!window.Element) var Element = {}; 
    11081285 
    11091286Element.extend = function(element) { 
    1110   if (!element || _nativeExtensions || element.nodeType == 3) return element; 
     1287  var F = Prototype.BrowserFeatures; 
     1288  if (!element || !element.tagName || element.nodeType == 3 || 
     1289   element._extended || F.SpecificElementExtensions || element == window) 
     1290    return element; 
    11111291 
    1112   if (!element._extended && element.tagName && element != window) { 
    1113     var methods = Object.clone(Element.Methods), cache = Element.extend.cache; 
     1292  var methods = {}, tagName = element.tagName, cache = Element.extend.cache, 
     1293   T = Element.Methods.ByTag; 
    11141294 
    1115     if (element.tagName == 'FORM') 
    1116       Object.extend(methods, Form.Methods); 
    1117     if (['INPUT', 'TEXTAREA', 'SELECT'].include(element.tagName)) 
    1118       Object.extend(methods, Form.Element.Methods); 
    1119  
     1295  // extend methods for all tags (Safari doesn't need this) 
     1296  if (!F.ElementExtensions) { 
     1297    Object.extend(methods, Element.Methods), 
    11201298    Object.extend(methods, Element.Methods.Simulated); 
     1299  } 
    11211300 
    1122     for (var property in methods) { 
    1123       var value = methods[property]; 
    1124       if (typeof value == 'function' && !(property in element)) 
    1125         element[property] = cache.findOrStore(value); 
    1126     } 
     1301  // extend methods for specific tags 
     1302  if (T[tagName]) Object.extend(methods, T[tagName]); 
     1303 
     1304  for (var property in methods) { 
     1305    var value = methods[property]; 
     1306    if (typeof value == 'function' && !(property in element)) 
     1307      element[property] = cache.findOrStore(value); 
    11271308  } 
    11281309 
    1129   element._extended = true; 
     1310  element._extended = Prototype.emptyFunction; 
    11301311  return element; 
    11311312}; 
    11321313 
     
    12121393  }, 
    12131394 
    12141395  descendants: function(element) { 
    1215     return $A($(element).getElementsByTagName('*')); 
     1396    return $A($(element).getElementsByTagName('*')).each(Element.extend); 
    12161397  }, 
    12171398 
     1399  firstDescendant: function(element) { 
     1400    element = $(element).firstChild; 
     1401    while (element && element.nodeType != 1) element = element.nextSibling; 
     1402    return $(element); 
     1403  }, 
     1404 
    12181405  immediateDescendants: function(element) { 
    12191406    if (!(element = $(element).firstChild)) return []; 
    12201407    while (element && element.nodeType != 1) element = element.nextSibling; 
     
    12421429  }, 
    12431430 
    12441431  up: function(element, expression, index) { 
    1245     return Selector.findElement($(element).ancestors(), expression, index); 
     1432    element = $(element); 
     1433    if (arguments.length == 1) return $(element.parentNode); 
     1434    var ancestors = element.ancestors(); 
     1435    return expression ? Selector.findElement(ancestors, expression, index) : 
     1436      ancestors[index || 0]; 
    12461437  }, 
    12471438 
    12481439  down: function(element, expression, index) { 
    1249     return Selector.findElement($(element).descendants(), expression, index); 
     1440    element = $(element); 
     1441    if (arguments.length == 1) return element.firstDescendant(); 
     1442    var descendants = element.descendants(); 
     1443    return expression ? Selector.findElement(descendants, expression, index) : 
     1444      descendants[index || 0]; 
    12501445  }, 
    12511446 
    12521447  previous: function(element, expression, index) { 
    1253     return Selector.findElement($(element).previousSiblings(), expression, index); 
     1448    element = $(element); 
     1449    if (arguments.length == 1) return $(Selector.handlers.previousElementSibling(element)); 
     1450    var previousSiblings = element.previousSiblings(); 
     1451    return expression ? Selector.findElement(previousSiblings, expression, index) : 
     1452      previousSiblings[index || 0]; 
    12541453  }, 
    12551454 
    12561455  next: function(element, expression, index) { 
    1257     return Selector.findElement($(element).nextSiblings(), expression, index); 
     1456    element = $(element); 
     1457    if (arguments.length == 1) return $(Selector.handlers.nextElementSibling(element)); 
     1458    var nextSiblings = element.nextSiblings(); 
     1459    return expression ? Selector.findElement(nextSiblings, expression, index) : 
     1460      nextSiblings[index || 0]; 
    12581461  }, 
    12591462 
    12601463  getElementsBySelector: function() { 
     
    12681471 
    12691472  readAttribute: function(element, name) { 
    12701473    element = $(element); 
    1271     if (document.all && !window.opera) { 
     1474    if (Prototype.Browser.IE) { 
     1475      if (!element.attributes) return null; 
    12721476      var t = Element._attributeTranslations; 
    12731477      if (t.values[name]) return t.values[name](element, name); 
    12741478      if (t.names[name])  name = t.names[name]; 
    12751479      var attribute = element.attributes[name]; 
    1276       if(attribute) return attribute.nodeValue; 
     1480      return attribute ? attribute.nodeValue : null; 
    12771481    } 
    12781482    return element.getAttribute(name); 
    12791483  }, 
     
    13421546  }, 
    13431547 
    13441548  empty: function(element) { 
    1345     return $(element).innerHTML.match(/^\s*$/); 
     1549    return $(element).innerHTML.blank(); 
    13461550  }, 
    13471551 
    13481552  descendantOf: function(element, ancestor) { 
     
    13611565 
    13621566  getStyle: function(element, style) { 
    13631567    element = $(element); 
    1364     if (['float','cssFloat'].include(style)) 
    1365       style = (typeof element.style.styleFloat != 'undefined' ? 'styleFloat' : 'cssFloat'); 
    1366     style = style.camelize(); 
     1568    style = style == 'float' ? 'cssFloat' : style.camelize(); 
    13671569    var value = element.style[style]; 
    13681570    if (!value) { 
    1369       if (document.defaultView && document.defaultView.getComputedStyle) { 
    1370         var css = document.defaultView.getComputedStyle(element, null); 
    1371         value = css ? css[style] : null; 
    1372       } else if (element.currentStyle) { 
    1373         value = element.currentStyle[style]; 
    1374       } 
     1571      var css = document.defaultView.getComputedStyle(element, null); 
     1572      value = css ? css[style] : null; 
    13751573    } 
     1574    if (style == 'opacity') return value ? parseFloat(value) : 1.0; 
     1575    return value == 'auto' ? null : value; 
     1576  }, 
    13761577 
    1377     if((value == 'auto') && ['width','height'].include(style) && (element.getStyle('display') != 'none')) 
    1378       value = element['offset'+style.capitalize()] + 'px'; 
     1578  getOpacity: function(element) { 
     1579    return $(element).getStyle('opacity'); 
     1580  }, 
    13791581 
    1380     if (window.opera && ['left', 'top', 'right', 'bottom'].include(style)) 
    1381       if (Element.getStyle(element, 'position') == 'static') value = 'auto'; 
    1382     if(style == 'opacity') { 
    1383       if(value) return parseFloat(value); 
    1384       if(value = (element.getStyle('filter') || '').match(/alpha\(opacity=(.*)\)/)) 
    1385         if(value[1]) return parseFloat(value[1]) / 100; 
    1386       return 1.0; 
    1387     } 
    1388     return value == 'auto' ? null : value; 
     1582  setStyle: function(element, styles, camelized) { 
     1583    element = $(element); 
     1584    var elementStyle = element.style; 
     1585 
     1586    for (var property in styles) 
     1587      if (property == 'opacity') element.setOpacity(styles[property]) 
     1588      else 
     1589        elementStyle[(property == 'float' || property == 'cssFloat') ? 
     1590          (elementStyle.styleFloat === undefined ? 'cssFloat' : 'styleFloat') : 
     1591          (camelized ? property : property.camelize())] = styles[property]; 
     1592 
     1593    return element; 
    13891594  }, 
    13901595 
    1391   setStyle: function(element, style) { 
     1596  setOpacity: function(element, value) { 
    13921597    element = $(element); 
    1393     for (var name in style) { 
    1394       var value = style[name]; 
    1395       if(name == 'opacity') { 
    1396         if (value == 1) { 
    1397           value = (/Gecko/.test(navigator.userAgent) && 
    1398             !/Konqueror|Safari|KHTML/.test(navigator.userAgent)) ? 0.999999 : 1.0; 
    1399           if(/MSIE/.test(navigator.userAgent) && !window.opera) 
    1400             element.style.filter = element.getStyle('filter').replace(/alpha\([^\)]*\)/gi,''); 
    1401         } else if(value == '') { 
    1402           if(/MSIE/.test(navigator.userAgent) && !window.opera) 
    1403             element.style.filter = element.getStyle('filter').replace(/alpha\([^\)]*\)/gi,''); 
    1404         } else { 
    1405           if(value < 0.00001) value = 0; 
    1406           if(/MSIE/.test(navigator.userAgent) && !window.opera) 
    1407             element.style.filter = element.getStyle('filter').replace(/alpha\([^\)]*\)/gi,'') + 
    1408               'alpha(opacity='+value*100+')'; 
    1409         } 
    1410       } else if(['float','cssFloat'].include(name)) name = (typeof element.style.styleFloat != 'undefined') ? 'styleFloat' : 'cssFloat'; 
    1411       element.style[name.camelize()] = value; 
    1412     } 
     1598    element.style.opacity = (value == 1 || value === '') ? '' : 
     1599      (value < 0.00001) ? 0 : value; 
    14131600    return element; 
    14141601  }, 
    14151602 
     
    14831670  } 
    14841671}; 
    14851672 
    1486 Object.extend(Element.Methods, {childOf: Element.Methods.descendantOf}); 
     1673Object.extend(Element.Methods, { 
     1674  childOf: Element.Methods.descendantOf, 
     1675  childElements: Element.Methods.immediateDescendants 
     1676}); 
    14871677 
    1488 Element._attributeTranslations = {}; 
     1678if (Prototype.Browser.Opera) { 
     1679  Element.Methods._getStyle = Element.Methods.getStyle; 
     1680  Element.Methods.getStyle = function(element, style) { 
     1681    switch(style) { 
     1682      case 'left': 
     1683      case 'top': 
     1684      case 'right': 
     1685      case 'bottom': 
     1686        if (Element._getStyle(element, 'position') == 'static') return null; 
     1687      default: return Element._getStyle(element, style); 
     1688    } 
     1689  }; 
     1690} 
     1691else if (Prototype.Browser.IE) { 
     1692  Element.Methods.getStyle = function(element, style) { 
     1693    element = $(element); 
     1694    style = (style == 'float' || style == 'cssFloat') ? 'styleFloat' : style.camelize(); 
     1695    var value = element.style[style]; 
     1696    if (!value && element.currentStyle) value = element.currentStyle[style]; 
    14891697 
    1490 Element._attributeTranslations.names = { 
    1491   colspan:   "colSpan", 
    1492   rowspan:   "rowSpan", 
    1493   valign:    "vAlign", 
    1494   datetime:  "dateTime", 
    1495   accesskey: "accessKey", 
    1496   tabindex:  "tabIndex", 
    1497   enctype:   "encType", 
    1498   maxlength: "maxLength", 
    1499   readonly:  "readOnly", 
    1500   longdesc:  "longDesc" 
    1501 }; 
     1698    if (style == 'opacity') { 
     1699      if (value = (element.getStyle('filter') || '').match(/alpha\(opacity=(.*)\)/)) 
     1700        if (value[1]) return parseFloat(value[1]) / 100; 
     1701      return 1.0; 
     1702    } 
    15021703 
    1503 Element._attributeTranslations.values = { 
    1504   _getAttr: function(element, attribute) { 
    1505     return element.getAttribute(attribute, 2); 
    1506   }, 
     1704    if (value == 'auto') { 
     1705      if ((style == 'width' || style == 'height') && (element.getStyle('display') != 'none')) 
     1706        return element['offset'+style.capitalize()] + 'px'; 
     1707      return null; 
     1708    } 
     1709    return value; 
     1710  }; 
    15071711 
    1508   _flag: function(element, attribute) { 
    1509     return $(element).hasAttribute(attribute) ? attribute : null; 
    1510   }, 
     1712  Element.Methods.setOpacity = function(element, value) { 
     1713    element = $(element); 
     1714    var filter = element.getStyle('filter'), style = element.style; 
     1715    if (value == 1 || value === '') { 
     1716      style.filter = filter.replace(/alpha\([^\)]*\)/gi,''); 
     1717      return element; 
     1718    } else if (value < 0.00001) value = 0; 
     1719    style.filter = filter.replace(/alpha\([^\)]*\)/gi, '') + 
     1720      'alpha(opacity=' + (value * 100) + ')'; 
     1721    return element; 
     1722  }; 
    15111723 
    1512   style: function(element) { 
    1513     return element.style.cssText.toLowerCase(); 
    1514   }, 
    1515  
    1516   title: function(element) { 
    1517     var node = element.getAttributeNode('title'); 
    1518     return node.specified ? node.nodeValue : null; 
    1519   } 
    1520 }; 
    1521  
    1522 Object.extend(Element._attributeTranslations.values, { 
    1523   href: Element._attributeTranslations.values._getAttr, 
    1524   src:  Element._attributeTranslations.values._getAttr, 
    1525   disabled: Element._attributeTranslations.values._flag, 
    1526   checked:  Element._attributeTranslations.values._flag, 
    1527   readonly: Element._attributeTranslations.values._flag, 
    1528   multiple: Element._attributeTranslations.values._flag 
    1529 }); 
    1530  
    1531 Element.Methods.Simulated = { 
    1532   hasAttribute: function(element, attribute) { 
    1533     var t = Element._attributeTranslations; 
    1534     attribute = t.names[attribute] || attribute; 
    1535     return $(element).getAttributeNode(attribute).specified; 
    1536   } 
    1537 }; 
    1538  
    1539 // IE is missing .innerHTML support for TABLE-related elements 
    1540 if (document.all && !window.opera){ 
     1724  // IE is missing .innerHTML support for TABLE-related elements 
    15411725  Element.Methods.update = function(element, html) { 
    15421726    element = $(element); 
    15431727    html = typeof html == 'undefined' ? '' : html.toString(); 
     
    15581742          div.innerHTML = '<table><tbody><tr><td>' +  html.stripScripts() + '</td></tr></tbody></table>'; 
    15591743          depth = 4; 
    15601744      } 
    1561       $A(element.childNodes).each(function(node){ 
    1562         element.removeChild(node) 
    1563       }); 
    1564       depth.times(function(){ div = div.firstChild }); 
    1565  
    1566       $A(div.childNodes).each( 
    1567         function(node){ element.appendChild(node) }); 
     1745      $A(element.childNodes).each(function(node) { element.removeChild(node) }); 
     1746      depth.times(function() { div = div.firstChild }); 
     1747      $A(div.childNodes).each(function(node) { element.appendChild(node) }); 
    15681748    } else { 
    15691749      element.innerHTML = html.stripScripts(); 
    15701750    } 
    1571     setTimeout(function() {html.evalScripts()}, 10); 
     1751    setTimeout(function() { html.evalScripts() }, 10); 
    15721752    return element; 
    15731753  } 
     1754} 
     1755else if (Prototype.Browser.Gecko) { 
     1756  Element.Methods.setOpacity = function(element, value) { 
     1757    element = $(element); 
     1758    element.style.opacity = (value == 1) ? 0.999999 : 
     1759      (value === '') ? '' : (value < 0.00001) ? 0 : value; 
     1760    return element; 
     1761  }; 
     1762} 
     1763 
     1764Element._attributeTranslations = { 
     1765  names: { 
     1766    colspan:   "colSpan", 
     1767    rowspan:   "rowSpan", 
     1768    valign:    "vAlign", 
     1769    datetime:  "dateTime", 
     1770    accesskey: "accessKey", 
     1771    tabindex:  "tabIndex", 
     1772    enctype:   "encType", 
     1773    maxlength: "maxLength", 
     1774    readonly:  "readOnly", 
     1775    longdesc:  "longDesc" 
     1776  }, 
     1777  values: { 
     1778    _getAttr: function(element, attribute) { 
     1779      return element.getAttribute(attribute, 2); 
     1780    }, 
     1781    _flag: function(element, attribute) { 
     1782      return $(element).hasAttribute(attribute) ? attribute : null; 
     1783    }, 
     1784    style: function(element) { 
     1785      return element.style.cssText.toLowerCase(); 
     1786    }, 
     1787    title: function(element) { 
     1788      var node = element.getAttributeNode('title'); 
     1789      return node.specified ? node.nodeValue : null; 
     1790    } 
     1791  } 
    15741792}; 
    15751793 
     1794(function() { 
     1795  Object.extend(this, { 
     1796    href: this._getAttr, 
     1797    src:  this._getAttr, 
     1798    type: this._getAttr, 
     1799    disabled: this._flag, 
     1800    checked:  this._flag, 
     1801    readonly: this._flag, 
     1802    multiple: this._flag 
     1803  }); 
     1804}).call(Element._attributeTranslations.values); 
     1805 
     1806Element.Methods.Simulated = { 
     1807  hasAttribute: function(element, attribute) { 
     1808    var t = Element._attributeTranslations, node; 
     1809    attribute = t.names[attribute] || attribute; 
     1810    node = $(element).getAttributeNode(attribute); 
     1811    return node && node.specified; 
     1812  } 
     1813}; 
     1814 
     1815Element.Methods.ByTag = {}; 
     1816 
    15761817Object.extend(Element, Element.Methods); 
    15771818 
    1578 var _nativeExtensions = false; 
     1819if (!Prototype.BrowserFeatures.ElementExtensions && 
     1820 document.createElement('div').__proto__) { 
     1821  window.HTMLElement = {}; 
     1822  window.HTMLElement.prototype = document.createElement('div').__proto__; 
     1823  Prototype.BrowserFeatures.ElementExtensions = true; 
     1824} 
    15791825 
    1580 if(/Konqueror|Safari|KHTML/.test(navigator.userAgent)) 
    1581   ['', 'Form', 'Input', 'TextArea', 'Select'].each(function(tag) { 
    1582     var className = 'HTML' + tag + 'Element'; 
    1583     if(window[className]) return; 
    1584     var klass = window[className] = {}; 
    1585     klass.prototype = document.createElement(tag ? tag.toLowerCase() : 'div').__proto__; 
    1586   }); 
     1826Element.hasAttribute = function(element, attribute) { 
     1827  if (element.hasAttribute) return element.hasAttribute(attribute); 
     1828  return Element.Methods.Simulated.hasAttribute(element, attribute); 
     1829}; 
    15871830 
    15881831Element.addMethods = function(methods) { 
    1589   Object.extend(Element.Methods, methods || {}); 
     1832  var F = Prototype.BrowserFeatures, T = Element.Methods.ByTag; 
    15901833 
     1834  if (!methods) { 
     1835    Object.extend(Form, Form.Methods); 
     1836    Object.extend(Form.Element, Form.Element.Methods); 
     1837    Object.extend(Element.Methods.ByTag, { 
     1838      "FORM":     Object.clone(Form.Methods), 
     1839      "INPUT":    Object.clone(Form.Element.Methods), 
     1840      "SELECT":   Object.clone(Form.Element.Methods), 
     1841      "TEXTAREA": Object.clone(Form.Element.Methods) 
     1842    }); 
     1843  } 
     1844 
     1845  if (arguments.length == 2) { 
     1846    var tagName = methods; 
     1847    methods = arguments[1]; 
     1848  } 
     1849 
     1850  if (!tagName) Object.extend(Element.Methods, methods || {}); 
     1851  else { 
     1852    if (tagName.constructor == Array) tagName.each(extend); 
     1853    else extend(tagName); 
     1854  } 
     1855 
     1856  function extend(tagName) { 
     1857    tagName = tagName.toUpperCase(); 
     1858    if (!Element.Methods.ByTag[tagName]) 
     1859      Element.Methods.ByTag[tagName] = {}; 
     1860    Object.extend(Element.Methods.ByTag[tagName], methods); 
     1861  } 
     1862 
    15911863  function copy(methods, destination, onlyIfAbsent) { 
    15921864    onlyIfAbsent = onlyIfAbsent || false; 
    15931865    var cache = Element.extend.cache; 
     
    15981870    } 
    15991871  } 
    16001872 
    1601   if (typeof HTMLElement != 'undefined') { 
     1873  function findDOMClass(tagName) { 
     1874    var klass; 
     1875    var trans = { 
     1876      "OPTGROUP": "OptGroup", "TEXTAREA": "TextArea", "P": "Paragraph", 
     1877      "FIELDSET": "FieldSet", "UL": "UList", "OL": "OList", "DL": "DList", 
     1878      "DIR": "Directory", "H1": "Heading", "H2": "Heading", "H3": "Heading", 
     1879      "H4": "Heading", "H5": "Heading", "H6": "Heading", "Q": "Quote", 
     1880      "INS": "Mod", "DEL": "Mod", "A": "Anchor", "IMG": "Image", "CAPTION": 
     1881      "TableCaption", "COL": "TableCol", "COLGROUP": "TableCol", "THEAD": 
     1882      "TableSection", "TFOOT": "TableSection", "TBODY": "TableSection", "TR": 
     1883      "TableRow", "TH": "TableCell", "TD": "TableCell", "FRAMESET": 
     1884      "FrameSet", "IFRAME": "IFrame" 
     1885    }; 
     1886    if (trans[tagName]) klass = 'HTML' + trans[tagName] + 'Element'; 
     1887    if (window[klass]) return window[klass]; 
     1888    klass = 'HTML' + tagName + 'Element'; 
     1889    if (window[klass]) return window[klass]; 
     1890    klass = 'HTML' + tagName.capitalize() + 'Element'; 
     1891    if (window[klass]) return window[klass]; 
     1892 
     1893    window[klass] = {}; 
     1894    window[klass].prototype = document.createElement(tagName).__proto__; 
     1895    return window[klass]; 
     1896  } 
     1897 
     1898  if (F.ElementExtensions) { 
    16021899    copy(Element.Methods, HTMLElement.prototype); 
    16031900    copy(Element.Methods.Simulated, HTMLElement.prototype, true); 
    1604     copy(Form.Methods, HTMLFormElement.prototype); 
    1605     [HTMLInputElement, HTMLTextAreaElement, HTMLSelectElement].each(function(klass) { 
    1606       copy(Form.Element.Methods, klass.prototype); 
    1607     }); 
    1608     _nativeExtensions = true; 
    16091901  } 
    1610 } 
    16111902 
    1612 var Toggle = new Object(); 
    1613 Toggle.display = Element.toggle; 
     1903  if (F.SpecificElementExtensions) { 
     1904    for (var tag in Element.Methods.ByTag) { 
     1905      var klass = findDOMClass(tag); 
     1906      if (typeof klass == "undefined") continue; 
     1907      copy(T[tag], klass.prototype); 
     1908    } 
     1909  } 
    16141910 
     1911  Object.extend(Element, Element.Methods); 
     1912  delete Element.ByTag; 
     1913}; 
     1914 
     1915var Toggle = { display: Element.toggle }; 
     1916 
    16151917/*--------------------------------------------------------------------------*/ 
    16161918 
    16171919Abstract.Insertion = function(adjacency) { 
     
    17412043}; 
    17422044 
    17432045Object.extend(Element.ClassNames.prototype, Enumerable); 
     2046/* Portions of the Selector class are derived from Jack Slocum’s DomQuery, 
     2047 * part of YUI-Ext version 0.40, distributed under the terms of an MIT-style 
     2048 * license.  Please see http://www.yui-ext.com/ for more information. */ 
     2049 
    17442050var Selector = Class.create(); 
     2051 
    17452052Selector.prototype = { 
    17462053  initialize: function(expression) { 
    1747     this.params = {classNames: []}; 
    1748     this.expression = expression.toString().strip(); 
    1749     this.parseExpression(); 
     2054    this.expression = expression.strip(); 
    17502055    this.compileMatcher(); 
    17512056  }, 
    17522057 
    1753   parseExpression: function() { 
    1754     function abort(message) { throw 'Parse error in selector: ' + message; } 
     2058  compileMatcher: function() { 
     2059    // Selectors with namespaced attributes can't use the XPath version 
     2060    if (Prototype.BrowserFeatures.XPath && !(/\[[\w-]*?:/).test(this.expression)) 
     2061      return this.compileXPathMatcher(); 
    17552062 
    1756     if (this.expression == '')  abort('empty expression'); 
     2063    var e = this.expression, ps = Selector.patterns, h = Selector.handlers, 
     2064        c = Selector.criteria, le, p, m; 
    17572065 
    1758     var params = this.params, expr = this.expression, match, modifier, clause, rest; 
    1759     while (match = expr.match(/^(.*)\[([a-z0-9_:-]+?)(?:([~\|!]?=)(?:"([^"]*)"|([^\]\s]*)))?\]$/i)) { 
    1760       params.attributes = params.attributes || []; 
    1761       params.attributes.push({name: match[2], operator: match[3], value: match[4] || match[5] || ''}); 
    1762       expr = match[1]; 
     2066    if (Selector._cache[e]) { 
     2067      this.matcher = Selector._cache[e]; return; 
    17632068    } 
     2069    this.matcher = ["this.matcher = function(root) {", 
     2070                    "var r = root, h = Selector.handlers, c = false, n;"]; 
    17642071 
    1765     if (expr == '*') return this.params.wildcard = true; 
    1766  
    1767     while (match = expr.match(/^([^a-z0-9_-])?([a-z0-9_-]+)(.*)/i)) { 
    1768       modifier = match[1], clause = match[2], rest = match[3]; 
    1769       switch (modifier) { 
    1770         case '#':       params.id = clause; break; 
    1771         case '.':       params.classNames.push(clause); break; 
    1772         case '': 
    1773         case undefined: params.tagName = clause.toUpperCase(); break; 
    1774         default:        abort(expr.inspect()); 
     2072    while (e && le != e && (/\S/).test(e)) { 
     2073      le = e; 
     2074      for (var i in ps) { 
     2075        p = ps[i]; 
     2076        if (m = e.match(p)) { 
     2077          this.matcher.push(typeof c[i] == 'function' ? c[i](m) : 
     2078              new Template(c[i]).evaluate(m)); 
     2079          e = e.replace(m[0], ''); 
     2080          break; 
     2081        } 
    17752082      } 
    1776       expr = rest; 
    17772083    } 
    17782084 
    1779     if (expr.length > 0) abort(expr.inspect()); 
     2085    this.matcher.push("return h.unique(n);\n}"); 
     2086    eval(this.matcher.join('\n')); 
     2087    Selector._cache[this.expression] = this.matcher; 
    17802088  }, 
    17812089 
    1782   buildMatchExpression: function() { 
    1783     var params = this.params, conditions = [], clause; 
     2090  compileXPathMatcher: function() { 
     2091    var e = this.expression, ps = Selector.patterns, 
     2092        x = Selector.xpath, le,  m; 
    17842093 
    1785     if (params.wildcard) 
    1786       conditions.push('true'); 
    1787     if (clause = params.id) 
    1788       conditions.push('element.readAttribute("id") == ' + clause.inspect()); 
    1789     if (clause = params.tagName) 
    1790       conditions.push('element.tagName.toUpperCase() == ' + clause.inspect()); 
    1791     if ((clause = params.classNames).length > 0) 
    1792       for (var i = 0, length = clause.length; i < length; i++) 
    1793         conditions.push('element.hasClassName(' + clause[i].inspect() + ')'); 
    1794     if (clause = params.attributes) { 
    1795       clause.each(function(attribute) { 
    1796         var value = 'element.readAttribute(' + attribute.name.inspect() + ')'; 
    1797         var splitValueBy = function(delimiter) { 
    1798           return value + ' && ' + value + '.split(' + delimiter.inspect() + ')'; 
     2094    if (Selector._cache[e]) { 
     2095      this.xpath = Selector._cache[e]; return; 
     2096    } 
     2097 
     2098    this.matcher = ['.//*']; 
     2099    while (e && le != e && (/\S/).test(e)) { 
     2100      le = e; 
     2101      for (var i in ps) { 
     2102        if (m = e.match(ps[i])) { 
     2103          this.matcher.push(typeof x[i] == 'function' ? x[i](m) : 
     2104            new Template(x[i]).evaluate(m)); 
     2105          e = e.replace(m[0], ''); 
     2106          break; 
    17992107        } 
     2108      } 
     2109    } 
    18002110 
    1801         switch (attribute.operator) { 
    1802           case '=':       conditions.push(value + ' == ' + attribute.value.inspect()); break; 
    1803           case '~=':      conditions.push(splitValueBy(' ') + '.include(' + attribute.value.inspect() + ')'); break; 
    1804           case '|=':      conditions.push( 
    1805                             splitValueBy('-') + '.first().toUpperCase() == ' + attribute.value.toUpperCase().inspect() 
    1806                           ); break; 
    1807           case '!=':      conditions.push(value + ' != ' + attribute.value.inspect()); break; 
    1808           case '': 
    1809           case undefined: conditions.push('element.hasAttribute(' + attribute.name.inspect() + ')'); break; 
    1810           default:        throw 'Unknown operator ' + attribute.operator + ' in selector'; 
     2111    this.xpath = this.matcher.join(''); 
     2112    Selector._cache[this.expression] = this.xpath; 
     2113  }, 
     2114 
     2115  findElements: function(root) { 
     2116    root = root || document; 
     2117    if (this.xpath) return document._getElementsByXPath(this.xpath, root); 
     2118    return this.matcher(root); 
     2119  }, 
     2120 
     2121  match: function(element) { 
     2122    return this.findElements(document).include(element); 
     2123  }, 
     2124 
     2125  toString: function() { 
     2126    return this.expression; 
     2127  }, 
     2128 
     2129  inspect: function() { 
     2130    return "#<Selector:" + this.expression.inspect() + ">"; 
     2131  } 
     2132}; 
     2133 
     2134Object.extend(Selector, { 
     2135  _cache: {}, 
     2136 
     2137  xpath: { 
     2138    descendant:   "//*", 
     2139    child:        "/*", 
     2140    adjacent:     "/following-sibling::*[1]", 
     2141    laterSibling: '/following-sibling::*', 
     2142    tagName:      function(m) { 
     2143      if (m[1] == '*') return ''; 
     2144      return "[local-name()='" + m[1].toLowerCase() + 
     2145             "' or local-name()='" + m[1].toUpperCase() + "']"; 
     2146    }, 
     2147    className:    "[contains(concat(' ', @class, ' '), ' #{1} ')]", 
     2148    id:           "[@id='#{1}']", 
     2149    attrPresence: "[@#{1}]", 
     2150    attr: function(m) { 
     2151      m[3] = m[5] || m[6]; 
     2152      return new Template(Selector.xpath.operators[m[2]]).evaluate(m); 
     2153    }, 
     2154    pseudo: function(m) { 
     2155      var h = Selector.xpath.pseudos[m[1]]; 
     2156      if (!h) return ''; 
     2157      if (typeof h === 'function') return h(m); 
     2158      return new Template(Selector.xpath.pseudos[m[1]]).evaluate(m); 
     2159    }, 
     2160    operators: { 
     2161      '=':  "[@#{1}='#{3}']", 
     2162      '!=': "[@#{1}!='#{3}']", 
     2163      '^=': "[starts-with(@#{1}, '#{3}')]", 
     2164      '$=': "[substring(@#{1}, (string-length(@#{1}) - string-length('#{3}') + 1))='#{3}']", 
     2165      '*=': "[contains(@#{1}, '#{3}')]", 
     2166      '~=': "[contains(concat(' ', @#{1}, ' '), ' #{3} ')]", 
     2167      '|=': "[contains(concat('-', @#{1}, '-'), '-#{3}-')]" 
     2168    }, 
     2169    pseudos: { 
     2170      'first-child': '[not(preceding-sibling::*)]', 
     2171      'last-child':  '[not(following-sibling::*)]', 
     2172      'only-child':  '[not(preceding-sibling::* or following-sibling::*)]', 
     2173      'empty':       "[count(*) = 0 and (count(text()) = 0 or translate(text(), ' \t\r\n', '') = '')]", 
     2174      'checked':     "[@checked]", 
     2175      'disabled':    "[@disabled]", 
     2176      'enabled':     "[not(@disabled)]", 
     2177      'not': function(m) { 
     2178        var e = m[6], p = Selector.patterns, 
     2179            x = Selector.xpath, le, m, v; 
     2180 
     2181        var exclusion = []; 
     2182        while (e && le != e && (/\S/).test(e)) { 
     2183          le = e; 
     2184          for (var i in p) { 
     2185            if (m = e.match(p[i])) { 
     2186              v = typeof x[i] == 'function' ? x[i](m) : new Template(x[i]).evaluate(m); 
     2187              exclusion.push("(" + v.substring(1, v.length - 1) + ")"); 
     2188              e = e.replace(m[0], ''); 
     2189              break; 
     2190            } 
     2191          } 
    18112192        } 
    1812       }); 
     2193        return "[not(" + exclusion.join(" and ") + ")]"; 
     2194      }, 
     2195      'nth-child':      function(m) { 
     2196        return Selector.xpath.pseudos.nth("(count(./preceding-sibling::*) + 1) ", m); 
     2197      }, 
     2198      'nth-last-child': function(m) { 
     2199        return Selector.xpath.pseudos.nth("(count(./following-sibling::*) + 1) ", m); 
     2200      }, 
     2201      'nth-of-type':    function(m) { 
     2202        return Selector.xpath.pseudos.nth("position() ", m); 
     2203      }, 
     2204      'nth-last-of-type': function(m) { 
     2205        return Selector.xpath.pseudos.nth("(last() + 1 - position()) ", m); 
     2206      }, 
     2207      'first-of-type':  function(m) { 
     2208        m[6] = "1"; return Selector.xpath.pseudos['nth-of-type'](m); 
     2209      }, 
     2210      'last-of-type':   function(m) { 
     2211        m[6] = "1"; return Selector.xpath.pseudos['nth-last-of-type'](m); 
     2212      }, 
     2213      'only-of-type':   function(m) { 
     2214        var p = Selector.xpath.pseudos; return p['first-of-type'](m) + p['last-of-type'](m); 
     2215      }, 
     2216      nth: function(fragment, m) { 
     2217        var mm, formula = m[6], predicate; 
     2218        if (formula == 'even') formula = '2n+0'; 
     2219        if (formula == 'odd')  formula = '2n+1'; 
     2220        if (mm = formula.match(/^(\d+)$/)) // digit only 
     2221          return '[' + fragment + "= " + mm[1] + ']'; 
     2222        if (mm = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b 
     2223          if (mm[1] == "-") mm[1] = -1; 
     2224          var a = mm[1] ? Number(mm[1]) : 1; 
     2225          var b = mm[2] ? Number(mm[2]) : 0; 
     2226          predicate = "[((#{fragment} - #{b}) mod #{a} = 0) and " + 
     2227          "((#{fragment} - #{b}) div #{a} >= 0)]"; 
     2228          return new Template(predicate).evaluate({ 
     2229            fragment: fragment, a: a, b: b }); 
     2230        } 
     2231      } 
    18132232    } 
     2233  }, 
    18142234 
    1815     return conditions.join(' && '); 
     2235  criteria: { 
     2236    tagName:      'n = h.tagName(n, r, "#{1}", c);   c = false;', 
     2237    className:    'n = h.className(n, r, "#{1}", c); c = false;', 
     2238    id:           'n = h.id(n, r, "#{1}", c);        c = false;', 
     2239    attrPresence: 'n = h.attrPresence(n, r, "#{1}"); c = false;', 
     2240    attr: function(m) { 
     2241      m[3] = (m[5] || m[6]); 
     2242      return new Template('n = h.attr(n, r, "#{1}", "#{3}", "#{2}"); c = false;').evaluate(m); 
     2243    }, 
     2244    pseudo:       function(m) { 
     2245      if (m[6]) m[6] = m[6].replace(/"/g, '\\"'); 
     2246      return new Template('n = h.pseudo(n, "#{1}", "#{6}", r, c); c = false;').evaluate(m); 
     2247    }, 
     2248    descendant:   'c = "descendant";', 
     2249    child:        'c = "child";', 
     2250    adjacent:     'c = "adjacent";', 
     2251    laterSibling: 'c = "laterSibling";' 
    18162252  }, 
    18172253 
    1818   compileMatcher: function() { 
    1819     this.match = new Function('element', 'if (!element.tagName) return false; \ 
    1820       element = $(element); \ 
    1821       return ' + this.buildMatchExpression()); 
     2254  patterns: { 
     2255    // combinators must be listed first 
     2256    // (and descendant needs to be last combinator) 
     2257    laterSibling: /^\s*~\s*/, 
     2258    child:        /^\s*>\s*/, 
     2259    adjacent:     /^\s*\+\s*/, 
     2260    descendant:   /^\s/, 
     2261 
     2262    // selectors follow 
     2263    tagName:      /^\s*(\*|[\w\-]+)(\b|$)?/, 
     2264    id:           /^#([\w\-\*]+)(\b|$)/, 
     2265    className:    /^\.([\w\-\*]+)(\b|$)/, 
     2266    pseudo:       /^:((first|last|nth|nth-last|only)(-child|-of-type)|empty|checked|(en|dis)abled|not)(\((.*?)\))?(\b|$|\s|(?=:))/, 
     2267    attrPresence: /^\[([\w]+)\]/, 
     2268    attr:         /\[((?:[\w-]*:)?[\w-]+)\s*(?:([!^$*~|]?=)\s*((['"])([^\]]*?)\4|([^'"][^\]]*?)))?\]/ 
    18222269  }, 
    18232270 
    1824   findElements: function(scope) { 
    1825     var element; 
     2271  handlers: { 
     2272    // UTILITY FUNCTIONS 
     2273    // joins two collections 
     2274    concat: function(a, b) { 
     2275      for (var i = 0, node; node = b[i]; i++) 
     2276        a.push(node); 
     2277      return a; 
     2278    }, 
    18262279 
    1827     if (element = $(this.params.id)) 
    1828       if (this.match(element)) 
    1829         if (!scope || Element.childOf(element, scope)) 
    1830           return [element]; 
     2280    // marks an array of nodes for counting 
     2281    mark: function(nodes) { 
     2282      for (var i = 0, node; node = nodes[i]; i++) 
     2283        node._counted = true; 
     2284      return nodes; 
     2285    }, 
    18312286 
    1832     scope = (scope || document).getElementsByTagName(this.params.tagName || '*'); 
     2287    unmark: function(nodes) { 
     2288      for (var i = 0, node; node = nodes[i]; i++) 
     2289        node._counted = undefined; 
     2290      return nodes; 
     2291    }, 
    18332292 
    1834     var results = []; 
    1835     for (var i = 0, length = scope.length; i < length; i++) 
    1836       if (this.match(element = scope[i])) 
    1837         results.push(Element.extend(element)); 
     2293    // mark each child node with its position (for nth calls) 
     2294    // "ofType" flag indicates whether we're indexing for nth-of-type 
     2295    // rather than nth-child 
     2296    index: function(parentNode, reverse, ofType) { 
     2297      parentNode._counted = true; 
     2298      if (reverse) { 
     2299        for (var nodes = parentNode.childNodes, i = nodes.length - 1, j = 1; i >= 0; i--) { 
     2300          node = nodes[i]; 
     2301          if (node.nodeType == 1 && (!ofType || node._counted)) node.nodeIndex = j++; 
     2302        } 
     2303      } else { 
     2304        for (var i = 0, j = 1, nodes = parentNode.childNodes; node = nodes[i]; i++) 
     2305          if (node.nodeType == 1 && (!ofType || node._counted)) node.nodeIndex = j++; 
     2306      } 
     2307    }, 
    18382308 
    1839     return results; 
     2309    // filters out duplicates and extends all nodes 
     2310    unique: function(nodes) { 
     2311      if (nodes.length == 0) return nodes; 
     2312      var results = [], n; 
     2313      for (var i = 0, l = nodes.length; i < l; i++) 
     2314        if (!(n = nodes[i])._counted) { 
     2315          n._counted = true; 
     2316          results.push(Element.extend(n)); 
     2317        } 
     2318      return Selector.handlers.unmark(results); 
     2319    }, 
     2320 
     2321    // COMBINATOR FUNCTIONS 
     2322    descendant: function(nodes) { 
     2323      var h = Selector.handlers; 
     2324      for (var i = 0, results = [], node; node = nodes[i]; i++) 
     2325        h.concat(results, node.getElementsByTagName('*')); 
     2326      return results; 
     2327    }, 
     2328 
     2329    child: function(nodes) { 
     2330      var h = Selector.handlers; 
     2331      for (var i = 0, results = [], node; node = nodes[i]; i++) { 
     2332        for (var j = 0, children = [], child; child = node.childNodes[j]; j++) 
     2333          if (child.nodeType == 1 && child.tagName != '!') results.push(child); 
     2334      } 
     2335      return results; 
     2336    }, 
     2337 
     2338    adjacent: function(nodes) { 
     2339      for (var i = 0, results = [], node; node = nodes[i]; i++) { 
     2340        var next = this.nextElementSibling(node); 
     2341        if (next) results.push(next); 
     2342      } 
     2343      return results; 
     2344    }, 
     2345 
     2346    laterSibling: function(nodes) { 
     2347      var h = Selector.handlers; 
     2348      for (var i = 0, results = [], node; node = nodes[i]; i++) 
     2349        h.concat(results, Element.nextSiblings(node)); 
     2350      return results; 
     2351    }, 
     2352 
     2353    nextElementSibling: function(node) { 
     2354      while (node = node.nextSibling) 
     2355              if (node.nodeType == 1) return node; 
     2356      return null; 
     2357    }, 
     2358 
     2359    previousElementSibling: function(node) { 
     2360      while (node = node.previousSibling) 
     2361        if (node.nodeType == 1) return node; 
     2362      return null; 
     2363    }, 
     2364 
     2365    // TOKEN FUNCTIONS 
     2366    tagName: function(nodes, root, tagName, combinator) { 
     2367      tagName = tagName.toUpperCase(); 
     2368      var results = [], h = Selector.handlers; 
     2369      if (nodes) { 
     2370        if (combinator) { 
     2371          // fastlane for ordinary descendant combinators 
     2372          if (combinator == "descendant") { 
     2373            for (var i = 0, node; node = nodes[i]; i++) 
     2374              h.concat(results, node.getElementsByTagName(tagName)); 
     2375            return results; 
     2376          } else nodes = this[combinator](nodes); 
     2377          if (tagName == "*") return nodes; 
     2378        } 
     2379        for (var i = 0, node; node = nodes[i]; i++) 
     2380          if (node.tagName.toUpperCase() == tagName) results.push(node); 
     2381        return results; 
     2382      } else return root.getElementsByTagName(tagName); 
     2383    }, 
     2384 
     2385    id: function(nodes, root, id, combinator) { 
     2386      var targetNode = $(id), h = Selector.handlers; 
     2387      if (!nodes && root == document) return targetNode ? [targetNode] : []; 
     2388      if (nodes) { 
     2389        if (combinator) { 
     2390          if (combinator == 'child') { 
     2391            for (var i = 0, node; node = nodes[i]; i++) 
     2392              if (targetNode.parentNode == node) return [targetNode]; 
     2393          } else if (combinator == 'descendant') { 
     2394            for (var i = 0, node; node = nodes[i]; i++) 
     2395              if (Element.descendantOf(targetNode, node)) return [targetNode]; 
     2396          } else if (combinator == 'adjacent') { 
     2397            for (var i = 0, node; node = nodes[i]; i++) 
     2398              if (Selector.handlers.previousElementSibling(targetNode) == node) 
     2399                return [targetNode]; 
     2400          } else nodes = h[combinator](nodes); 
     2401        } 
     2402        for (var i = 0, node; node = nodes[i]; i++) 
     2403          if (node == targetNode) return [targetNode]; 
     2404        return []; 
     2405      } 
     2406      return (targetNode && Element.descendantOf(targetNode, root)) ? [targetNode] : []; 
     2407    }, 
     2408 
     2409    className: function(nodes, root, className, combinator) { 
     2410      if (nodes && combinator) nodes = this[combinator](nodes); 
     2411      return Selector.handlers.byClassName(nodes, root, className); 
     2412    }, 
     2413 
     2414    byClassName: function(nodes, root, className) { 
     2415      if (!nodes) nodes = Selector.handlers.descendant([root]); 
     2416      var needle = ' ' + className + ' '; 
     2417      for (var i = 0, results = [], node, nodeClassName; node = nodes[i]; i++) { 
     2418        nodeClassName = node.className; 
     2419        if (nodeClassName.length == 0) continue; 
     2420        if (nodeClassName == className || (' ' + nodeClassName + ' ').include(needle)) 
     2421          results.push(node); 
     2422      } 
     2423      return results; 
     2424    }, 
     2425 
     2426    attrPresence: function(nodes, root, attr) { 
     2427      var results = []; 
     2428      for (var i = 0, node; node = nodes[i]; i++) 
     2429        if (Element.hasAttribute(node, attr)) results.push(node); 
     2430      return results; 
     2431    }, 
     2432 
     2433    attr: function(nodes, root, attr, value, operator) { 
     2434      if (!nodes) nodes = root.getElementsByTagName("*"); 
     2435      var handler = Selector.operators[operator], results = []; 
     2436      for (var i = 0, node; node = nodes[i]; i++) { 
     2437        var nodeValue = Element.readAttribute(node, attr); 
     2438        if (nodeValue === null) continue; 
     2439        if (handler(nodeValue, value)) results.push(node); 
     2440      } 
     2441      return results; 
     2442    }, 
     2443 
     2444    pseudo: function(nodes, name, value, root, combinator) { 
     2445      if (nodes && combinator) nodes = this[combinator](nodes); 
     2446      if (!nodes) nodes = root.getElementsByTagName("*"); 
     2447      return Selector.pseudos[name](nodes, value, root); 
     2448    } 
    18402449  }, 
    18412450 
    1842   toString: function() { 
    1843     return this.expression; 
    1844   } 
    1845 } 
     2451  pseudos: { 
     2452    'first-child': function(nodes, value, root) { 
     2453      for (var i = 0, results = [], node; node = nodes[i]; i++) { 
     2454        if (Selector.handlers.previousElementSibling(node)) continue; 
     2455          results.push(node); 
     2456      } 
     2457      return results; 
     2458    }, 
     2459    'last-child': function(nodes, value, root) { 
     2460      for (var i = 0, results = [], node; node = nodes[i]; i++) { 
     2461        if (Selector.handlers.nextElementSibling(node)) continue; 
     2462          results.push(node); 
     2463      } 
     2464      return results; 
     2465    }, 
     2466    'only-child': function(nodes, value, root) { 
     2467      var h = Selector.handlers; 
     2468      for (var i = 0, results = [], node; node = nodes[i]; i++) 
     2469        if (!h.previousElementSibling(node) && !h.nextElementSibling(node)) 
     2470          results.push(node); 
     2471      return results; 
     2472    }, 
     2473    'nth-child':        function(nodes, formula, root) { 
     2474      return Selector.pseudos.nth(nodes, formula, root); 
     2475    }, 
     2476    'nth-last-child':   function(nodes, formula, root) { 
     2477      return Selector.pseudos.nth(nodes, formula, root, true); 
     2478    }, 
     2479    'nth-of-type':      function(nodes, formula, root) { 
     2480      return Selector.pseudos.nth(nodes, formula, root, false, true); 
     2481    }, 
     2482    'nth-last-of-type': function(nodes, formula, root) { 
     2483      return Selector.pseudos.nth(nodes, formula, root, true, true); 
     2484    }, 
     2485    'first-of-type':    function(nodes, formula, root) { 
     2486      return Selector.pseudos.nth(nodes, "1", root, false, true); 
     2487    }, 
     2488    'last-of-type':     function(nodes, formula, root) { 
     2489      return Selector.pseudos.nth(nodes, "1", root, true, true); 
     2490    }, 
     2491    'only-of-type':     function(nodes, formula, root) { 
     2492      var p = Selector.pseudos; 
     2493      return p['last-of-type'](p['first-of-type'](nodes, formula, root), formula, root); 
     2494    }, 
    18462495 
    1847 Object.extend(Selector, { 
     2496    // handles the an+b logic 
     2497    getIndices: function(a, b, total) { 
     2498      if (a == 0) return b > 0 ? [b] : []; 
     2499      return $R(1, total).inject([], function(memo, i) { 
     2500        if (0 == (i - b) % a && (i - b) / a >= 0) memo.push(i); 
     2501        return memo; 
     2502      }); 
     2503    }, 
     2504 
     2505    // handles nth(-last)-child, nth(-last)-of-type, and (first|last)-of-type 
     2506    nth: function(nodes, formula, root, reverse, ofType) { 
     2507      if (nodes.length == 0) return []; 
     2508      if (formula == 'even') formula = '2n+0'; 
     2509      if (formula == 'odd')  formula = '2n+1'; 
     2510      var h = Selector.handlers, results = [], indexed = [], m; 
     2511      h.mark(nodes); 
     2512      for (var i = 0, node; node = nodes[i]; i++) { 
     2513        if (!node.parentNode._counted) { 
     2514          h.index(node.parentNode, reverse, ofType); 
     2515          indexed.push(node.parentNode); 
     2516        } 
     2517      } 
     2518      if (formula.match(/^\d+$/)) { // just a number 
     2519        formula = Number(formula); 
     2520        for (var i = 0, node; node = nodes[i]; i++) 
     2521          if (node.nodeIndex == formula) results.push(node); 
     2522      } else if (m = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b 
     2523        if (m[1] == "-") m[1] = -1; 
     2524        var a = m[1] ? Number(m[1]) : 1; 
     2525        var b = m[2] ? Number(m[2]) : 0; 
     2526        var indices = Selector.pseudos.getIndices(a, b, nodes.length); 
     2527        for (var i = 0, node, l = indices.length; node = nodes[i]; i++) { 
     2528          for (var j = 0; j < l; j++) 
     2529            if (node.nodeIndex == indices[j]) results.push(node); 
     2530        } 
     2531      } 
     2532      h.unmark(nodes); 
     2533      h.unmark(indexed); 
     2534      return results; 
     2535    }, 
     2536 
     2537    'empty': function(nodes, value, root) { 
     2538      for (var i = 0, results = [], node; node = nodes[i]; i++) { 
     2539        // IE treats comments as element nodes 
     2540        if (node.tagName == '!' || (node.firstChild && !node.innerHTML.match(/^\s*$/))) continue; 
     2541        results.push(node); 
     2542      } 
     2543      return results; 
     2544    }, 
     2545 
     2546    'not': function(nodes, selector, root) { 
     2547      var h = Selector.handlers, selectorType, m; 
     2548      var exclusions = new Selector(selector).findElements(root); 
     2549      h.mark(exclusions); 
     2550      for (var i = 0, results = [], node; node = nodes[i]; i++) 
     2551        if (!node._counted) results.push(node); 
     2552      h.unmark(exclusions); 
     2553      return results; 
     2554    }, 
     2555 
     2556    'enabled': function(nodes, value, root) { 
     2557      for (var i = 0, results = [], node; node = nodes[i]; i++) 
     2558        if (!node.disabled) results.push(node); 
     2559      return results; 
     2560    }, 
     2561 
     2562    'disabled': function(nodes, value, root) { 
     2563      for (var i = 0, results = [], node; node = nodes[i]; i++) 
     2564        if (node.disabled) results.push(node); 
     2565      return results; 
     2566    }, 
     2567 
     2568    'checked': function(nodes, value, root) { 
     2569      for (var i = 0, results = [], node; node = nodes[i]; i++) 
     2570        if (node.checked) results.push(node); 
     2571      return results; 
     2572    } 
     2573  }, 
     2574 
     2575  operators: { 
     2576    '=':  function(nv, v) { return nv == v; }, 
     2577    '!=': function(nv, v) { return nv != v; }, 
     2578    '^=': function(nv, v) { return nv.startsWith(v); }, 
     2579    '$=': function(nv, v) { return nv.endsWith(v); }, 
     2580    '*=': function(nv, v) { return nv.include(v); }, 
     2581    '~=': function(nv, v) { return (' ' + nv + ' ').include(' ' + v + ' '); }, 
     2582    '|=': function(nv, v) { return ('-' + nv.toUpperCase() + '-').include('-' + v.toUpperCase() + '-'); } 
     2583  }, 
     2584 
    18482585  matchElements: function(elements, expression) { 
    1849     var selector = new Selector(expression); 
    1850     return elements.select(selector.match.bind(selector)).map(Element.extend); 
     2586    var matches = new Selector(expression).findElements(), h = Selector.handlers; 
     2587    h.mark(matches); 
     2588    for (var i = 0, results = [], element; element = elements[i]; i++) 
     2589      if (element._counted) results.push(element); 
     2590    h.unmark(matches); 
     2591    return results; 
    18512592  }, 
    18522593 
    18532594  findElement: function(elements, expression, index) { 
    1854     if (typeof expression == 'number') index = expression, expression = false; 
     2595    if (typeof expression == 'number') { 
     2596      index = expression; expression = false; 
     2597    } 
    18552598    return Selector.matchElements(elements, expression || '*')[index || 0]; 
    18562599  }, 
    18572600 
    18582601  findChildElements: function(element, expressions) { 
    1859     return expressions.map(function(expression) { 
    1860       return expression.match(/[^\s"]+(?:"[^"]*"[^\s"]+)*/g).inject([null], function(results, expr) { 
    1861         var selector = new Selector(expr); 
    1862         return results.inject([], function(elements, result) { 
    1863           return elements.concat(selector.findElements(result || element)); 
    1864         }); 
    1865       }); 
    1866     }).flatten(); 
     2602    var exprs = expressions.join(','), expressions = []; 
     2603    exprs.scan(/(([\w#:.~>+()\s-]+|\*|\[.*?\])+)\s*(,|$)/, function(m) { 
     2604      expressions.push(m[1].strip()); 
     2605    }); 
     2606    var results = [], h = Selector.handlers; 
     2607    for (var i = 0, l = expressions.length, selector; i < l; i++) { 
     2608      selector = new Selector(expressions[i].strip()); 
     2609      h.concat(results, selector.findElements(element)); 
     2610    } 
     2611    return (l > 1) ? h.unique(results) : results; 
    18672612  } 
    18682613}); 
    18692614 
     
    18802625    var data = elements.inject({}, function(result, element) { 
    18812626      if (!element.disabled && element.name) { 
    18822627        var key = element.name, value = $(element).getValue(); 
    1883         if (value != undefined) { 
    1884           if (result[key]) { 
     2628        if (value != null) { 
     2629                if (key in result) { 
    18852630            if (result[key].constructor != Array) result[key] = [result[key]]; 
    18862631            result[key].push(value); 
    18872632          } 
     
    19282673 
    19292674  disable: function(form) { 
    19302675    form = $(form); 
    1931     form.getElements().each(function(element) { 
    1932       element.blur(); 
    1933       element.disabled = 'true'; 
    1934     }); 
     2676    Form.getElements(form).invoke('disable'); 
    19352677    return form; 
    19362678  }, 
    19372679 
    19382680  enable: function(form) { 
    19392681    form = $(form); 
    1940     form.getElements().each(function(element) { 
    1941       element.disabled = ''; 
    1942     }); 
     2682    Form.getElements(form).invoke('enable'); 
    19432683    return form; 
    19442684  }, 
    19452685 
     
    19542694    form = $(form); 
    19552695    form.findFirstElement().activate(); 
    19562696    return form; 
     2697  }, 
     2698 
     2699  request: function(form, options) { 
     2700    form = $(form), options = Object.clone(options || {}); 
     2701 
     2702    var params = options.parameters; 
     2703    options.parameters = form.serialize(true); 
     2704 
     2705    if (params) { 
     2706      if (typeof params == 'string') params = params.toQueryParams(); 
     2707      Object.extend(options.parameters, params); 
     2708    } 
     2709 
     2710    if (form.hasAttribute('method') && !options.method) 
     2711      options.method = form.method; 
     2712 
     2713    return new Ajax.Request(form.readAttribute('action'), options); 
    19572714  } 
    19582715} 
    19592716 
    1960 Object.extend(Form, Form.Methods); 
    1961  
    19622717/*--------------------------------------------------------------------------*/ 
    19632718 
    19642719Form.Element = { 
     
    20042759 
    20052760  activate: function(element) { 
    20062761    element = $(element); 
    2007     element.focus(); 
    2008     if (element.select && ( element.tagName.toLowerCase() != 'input' || 
    2009       !['button', 'reset', 'submit'].include(element.type) ) ) 
    2010       element.select(); 
     2762    try { 
     2763      element.focus(); 
     2764      if (element.select && (element.tagName.toLowerCase() != 'input' || 
     2765        !['button', 'reset', 'submit'].include(element.type))) 
     2766        element.select(); 
     2767    } catch (e) {} 
    20112768    return element; 
    20122769  }, 
    20132770 
    20142771  disable: function(element) { 
    20152772    element = $(element); 
     2773    element.blur(); 
    20162774    element.disabled = true; 
    20172775    return element; 
    20182776  }, 
    20192777 
    20202778  enable: function(element) { 
    20212779    element = $(element); 
    2022     element.blur(); 
    20232780    element.disabled = false; 
    20242781    return element; 
    20252782  } 
    20262783} 
    20272784 
    2028 Object.extend(Form.Element, Form.Element.Methods); 
     2785/*--------------------------------------------------------------------------*/ 
     2786 
    20292787var Field = Form.Element; 
    2030 var $F = Form.Element.getValue; 
     2788var $F = Form.Element.Methods.getValue; 
    20312789 
    20322790/*--------------------------------------------------------------------------*/ 
    20332791 
     
    21942952  KEY_PAGEDOWN: 34, 
    21952953 
    21962954  element: function(event) { 
    2197     return event.target || event.srcElement; 
     2955    return $(event.target || event.srcElement); 
    21982956  }, 
    21992957 
    22002958  isLeftClick: function(event) { 
     
    22593017    useCapture = useCapture || false; 
    22603018 
    22613019    if (name == 'keypress' && 
    2262         (navigator.appVersion.match(/Konqueror|Safari|KHTML/) 
    2263         || element.attachEvent)) 
     3020      (Prototype.Browser.WebKit || element.attachEvent)) 
    22643021      name = 'keydown'; 
    22653022 
    22663023    Event._observeAndCache(element, name, observer, useCapture); 
     
    22713028    useCapture = useCapture || false; 
    22723029 
    22733030    if (name == 'keypress' && 
    2274         (navigator.appVersion.match(/Konqueror|Safari|KHTML/) 
    2275         || element.detachEvent)) 
     3031        (Prototype.Browser.WebKit || element.attachEvent)) 
    22763032      name = 'keydown'; 
    22773033 
    22783034    if (element.removeEventListener) { 
     
    22863042}); 
    22873043 
    22883044/* prevent memory leaks in IE */ 
    2289 if (navigator.appVersion.match(/\bMSIE\b/)) 
     3045if (Prototype.Browser.IE) 
    22903046  Event.observe(window, 'unload', Event.unloadCache, false); 
    22913047var Position = { 
    22923048  // set to true if needed, warning: firefox performance problems 
     
    24003156      valueL += element.offsetLeft || 0; 
    24013157 
    24023158      // Safari fix 
    2403       if (element.offsetParent==document.body) 
     3159      if (element.offsetParent == document.body) 
    24043160        if (Element.getStyle(element,'position')=='absolute') break; 
    24053161 
    24063162    } while (element = element.offsetParent); 
     
    24963252// Safari returns margins on body which is incorrect if the child is absolutely 
    24973253// positioned.  For performance reasons, redefine Position.cumulativeOffset for 
    24983254// KHTML/WebKit only. 
    2499 if (/Konqueror|Safari|KHTML/.test(navigator.userAgent)) { 
     3255if (Prototype.Browser.WebKit) { 
    25003256  Position.cumulativeOffset = function(element) { 
    25013257    var valueT = 0, valueL = 0; 
    25023258    do {