ghazel-fiveruns_tuneup 0.8.22

Sign up to get free protection for your applications and to get access to all the features.
Files changed (60) hide show
  1. data/CHANGELOG +33 -0
  2. data/CONTRIBUTORS +5 -0
  3. data/README.rdoc +65 -0
  4. data/assets/images/arrows.gif +0 -0
  5. data/assets/images/edit.png +0 -0
  6. data/assets/images/fade.png +0 -0
  7. data/assets/images/fade_down.png +0 -0
  8. data/assets/images/head.gif +0 -0
  9. data/assets/images/logo.gif +0 -0
  10. data/assets/images/logo_clear.png +0 -0
  11. data/assets/images/magnify.png +0 -0
  12. data/assets/images/pin.png +0 -0
  13. data/assets/images/pip.gif +0 -0
  14. data/assets/images/pointer.gif +0 -0
  15. data/assets/images/pushed_pin.png +0 -0
  16. data/assets/images/schema.png +0 -0
  17. data/assets/images/signin.gif +0 -0
  18. data/assets/images/spinner.gif +0 -0
  19. data/assets/images/warning.gif +0 -0
  20. data/assets/javascripts/init.js +12 -0
  21. data/assets/javascripts/prototype.js +2515 -0
  22. data/assets/javascripts/tuneup.js +115 -0
  23. data/assets/stylesheets/tuneup.css +209 -0
  24. data/init.rb +2 -0
  25. data/install.rb +13 -0
  26. data/lib/bumpspark_helper.rb +52 -0
  27. data/lib/fiveruns/tuneup.rb +154 -0
  28. data/lib/fiveruns/tuneup/asset_tags.rb +54 -0
  29. data/lib/fiveruns/tuneup/configuration.rb +25 -0
  30. data/lib/fiveruns/tuneup/custom_methods.rb +8 -0
  31. data/lib/fiveruns/tuneup/environment.rb +29 -0
  32. data/lib/fiveruns/tuneup/instrumentation/action_controller/base.rb +59 -0
  33. data/lib/fiveruns/tuneup/instrumentation/action_view/base.rb +81 -0
  34. data/lib/fiveruns/tuneup/instrumentation/action_view/partial_template.rb +28 -0
  35. data/lib/fiveruns/tuneup/instrumentation/active_record/base.rb +126 -0
  36. data/lib/fiveruns/tuneup/instrumentation/cgi/session.rb +30 -0
  37. data/lib/fiveruns/tuneup/instrumentation/utilities.rb +187 -0
  38. data/lib/fiveruns/tuneup/multipart.rb +75 -0
  39. data/lib/fiveruns/tuneup/routing.rb +25 -0
  40. data/lib/fiveruns/tuneup/runs.rb +86 -0
  41. data/lib/fiveruns/tuneup/schema.rb +43 -0
  42. data/lib/fiveruns/tuneup/step.rb +221 -0
  43. data/lib/fiveruns/tuneup/version.rb +89 -0
  44. data/lib/fiveruns_tuneup.rb +11 -0
  45. data/lib/tuneup_controller.rb +46 -0
  46. data/lib/tuneup_helper.rb +181 -0
  47. data/rails/init.rb +14 -0
  48. data/tasks/assets.rake +32 -0
  49. data/test/test_helper.rb +3 -0
  50. data/test/tuneup_test.rb +0 -0
  51. data/uninstall.rb +6 -0
  52. data/views/tuneup/_data.html.erb +15 -0
  53. data/views/tuneup/_flash.html.erb +6 -0
  54. data/views/tuneup/_link.html.erb +1 -0
  55. data/views/tuneup/_schema.html.erb +17 -0
  56. data/views/tuneup/_sql.html.erb +23 -0
  57. data/views/tuneup/_step.html.erb +17 -0
  58. data/views/tuneup/panel/_show.html.erb +4 -0
  59. data/views/tuneup/sandbox.html.erb +6 -0
  60. metadata +124 -0
@@ -0,0 +1,33 @@
1
+ v0.8.20. Rails 2.3 support, remove old bin/fiveruns_tuneup executable.
2
+
3
+ v0.8.18. Fix deprecated use of Dependencies.
4
+
5
+ v0.8.17. Clean build to work around 0.8.16 build snafu (built from a non-clean clone)
6
+
7
+ v0.8.16. JRuby bug fix
8
+
9
+ v0.8.15. Removed use of all Rails JS helpers, this fixes an issue when jrails is used in the app (Thanks, Chris Zelenak)
10
+
11
+ v0.8.13. Improved unobtrusive branding.
12
+
13
+ v0.8.12. Added some inline branding.
14
+
15
+ v0.8.11. Signup and login for the TuneUp hosted service is now only required if the user wants to share a run (privately or publicly).
16
+
17
+ v0.8.10. Fixed TextMate links to be xhtml strict 1.0 compliant.
18
+
19
+ v0.8.9. Fixed broken rendering of TuneUp stack information when serving xhtml strict pages with a content-type of application/xhtml+xml.
20
+
21
+ v0.8.8. Fixed login broken when serving pages as xhtml strict. Filtered params logged to tuneup.log. TuneUp partials now rendered with exlicit extensions (.html.erb).
22
+
23
+ v0.8.7. Sandboxed javascript for non-Prototype apps (Thanks, Howard Rauscher). The tuneup.log now written explicitly to RAILS_ROOT/log. Removed History.rdoc in favor of CHANGELOG.
24
+
25
+ v0.8.6. Performance optimizations, CSS fixes that prevents some layout issues that occurred when system doesn't have tuneup's preferred fonts.
26
+
27
+ v0.8.5. Handle valid content types other than text/html (like strict XHTML with content type application/xhmtl+xml).
28
+
29
+ v0.8.4. Fixes for forgery protection breaking the panel, significant work on display issues (including CSS isolation concerns and alert notices pushing down absolute/fixed elements), and support for environmental configuration. Support for Rails 2.1 development mode as a gem dependency.
30
+
31
+ v0.8.3. Beta release.
32
+
33
+ v0.8.1. First gem package with basic functionality
@@ -0,0 +1,5 @@
1
+ * 2008-09-03: Chris Zelenak; pointed us in the right direction: a bug with tuneup + jrails
2
+
3
+ * 2008-07-21: Bryan Liles; merged fallback log dir code from git://github.com/bryanl/fiveruns_tuneup.git (master:e5838f47f)
4
+
5
+ * 2008-06-14: Howard Rauscher; merged javascript sandboxing code from git://github.com/howardr/fiveruns_tuneup.git (master:8cc558dbe)
@@ -0,0 +1,65 @@
1
+ = FiveRuns TuneUp Plugin (panel)
2
+
3
+ == Requirements
4
+
5
+ * Rails 2.0.1+
6
+ * Firefox 2.0+, Safari 3.1+. IE support is on the roadmap.
7
+
8
+ == Synopsis
9
+
10
+ Display call stack information on Rails requests.
11
+
12
+ == Installation
13
+
14
+ For Rails 2.1+, you can use it as a gem dependency (recommended).
15
+
16
+ Install the gem:
17
+
18
+ sudo gem install fiveruns_tuneup
19
+
20
+ In your +environment.rb+:
21
+
22
+ config.gem 'fiveruns_tuneup'
23
+
24
+ *Note*: There is an issue in Rails 2.1 that prevents the plugin running as an unpacked gem (via +rake gems:unpack+).
25
+
26
+ TuneUp can also be used from +vendor/plugins+.
27
+
28
+ == Configuration
29
+
30
+ By default, TuneUp only displays the panel and instruments your application
31
+ in the +development+ environment. If you'd like to configure it to run in the +production+
32
+ environment (for testing), you can create a +RAILS_ROOT/config/tuneup.rb+ file:
33
+
34
+ Fiveruns::Tuneup.config do |config|
35
+ config.environments << :production
36
+ end
37
+
38
+ == Known Issues
39
+
40
+ See the notes at http://github.com/fiveruns/fiveruns_tuneup/wikis/known-issues
41
+
42
+ == License
43
+
44
+ (The FiveRuns License)
45
+
46
+ Copyright (c) 2008 FiveRuns Corporation
47
+
48
+ Permission is hereby granted, free of charge, to any person obtaining
49
+ a copy of this software and associated documentation files (the
50
+ "Software"), to deal in the Software without restriction, including
51
+ without limitation the rights to use, copy, modify, merge, publish,
52
+ distribute, sublicense, and/or sell copies of the Software, and to
53
+ permit persons to whom the Software is furnished to do so, subject to
54
+ the following conditions:
55
+
56
+ The above copyright notice and this permission notice shall be
57
+ included in all copies or substantial portions of the Software.
58
+
59
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
60
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
61
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
62
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
63
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
64
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
65
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
Binary file
@@ -0,0 +1,12 @@
1
+ // setTimeout will not be called untilled the DOM is loaded
2
+ setTimeout(function() {
3
+ var ifr = document.createElement('iframe');
4
+ ifr.src = '/tuneup/sandbox';
5
+
6
+ var style = ifr.style;
7
+ style.visibility = 'hidden';
8
+ style.width = '0';
9
+ style.height = '0';
10
+
11
+ document.body.appendChild(ifr);
12
+ }, 50);
@@ -0,0 +1,2515 @@
1
+ /* Prototype JavaScript framework, version 1.5.0
2
+ * (c) 2005-2007 Sam Stephenson
3
+ *
4
+ * Prototype is freely distributable under the terms of an MIT-style license.
5
+ * For details, see the Prototype web site: http://prototype.conio.net/
6
+ *
7
+ /*--------------------------------------------------------------------------*/
8
+
9
+ var Prototype = {
10
+ Version: '1.5.0',
11
+ BrowserFeatures: {
12
+ XPath: !!document.evaluate
13
+ },
14
+
15
+ ScriptFragment: '(?:<script.*?>)((\n|\r|.)*?)(?:<\/script>)',
16
+ emptyFunction: function() {},
17
+ K: function(x) { return x }
18
+ }
19
+
20
+ var Class = {
21
+ create: function() {
22
+ return function() {
23
+ this.initialize.apply(this, arguments);
24
+ }
25
+ }
26
+ }
27
+
28
+ var Abstract = new Object();
29
+
30
+ Object.extend = function(destination, source) {
31
+ for (var property in source) {
32
+ destination[property] = source[property];
33
+ }
34
+ return destination;
35
+ }
36
+
37
+ Object.extend(Object, {
38
+ inspect: function(object) {
39
+ try {
40
+ if (object === undefined) return 'undefined';
41
+ if (object === null) return 'null';
42
+ return object.inspect ? object.inspect() : object.toString();
43
+ } catch (e) {
44
+ if (e instanceof RangeError) return '...';
45
+ throw e;
46
+ }
47
+ },
48
+
49
+ keys: function(object) {
50
+ var keys = [];
51
+ for (var property in object)
52
+ keys.push(property);
53
+ return keys;
54
+ },
55
+
56
+ values: function(object) {
57
+ var values = [];
58
+ for (var property in object)
59
+ values.push(object[property]);
60
+ return values;
61
+ },
62
+
63
+ clone: function(object) {
64
+ return Object.extend({}, object);
65
+ }
66
+ });
67
+
68
+ Function.prototype.bind = function() {
69
+ var __method = this, args = $A(arguments), object = args.shift();
70
+ return function() {
71
+ return __method.apply(object, args.concat($A(arguments)));
72
+ }
73
+ }
74
+
75
+ Function.prototype.bindAsEventListener = function(object) {
76
+ var __method = this, args = $A(arguments), object = args.shift();
77
+ return function(event) {
78
+ return __method.apply(object, [( event || window.event)].concat(args).concat($A(arguments)));
79
+ }
80
+ }
81
+
82
+ Object.extend(Number.prototype, {
83
+ toColorPart: function() {
84
+ var digits = this.toString(16);
85
+ if (this < 16) return '0' + digits;
86
+ return digits;
87
+ },
88
+
89
+ succ: function() {
90
+ return this + 1;
91
+ },
92
+
93
+ times: function(iterator) {
94
+ $R(0, this, true).each(iterator);
95
+ return this;
96
+ }
97
+ });
98
+
99
+ var Try = {
100
+ these: function() {
101
+ var returnValue;
102
+
103
+ for (var i = 0, length = arguments.length; i < length; i++) {
104
+ var lambda = arguments[i];
105
+ try {
106
+ returnValue = lambda();
107
+ break;
108
+ } catch (e) {}
109
+ }
110
+
111
+ return returnValue;
112
+ }
113
+ }
114
+
115
+ /*--------------------------------------------------------------------------*/
116
+
117
+ var PeriodicalExecuter = Class.create();
118
+ PeriodicalExecuter.prototype = {
119
+ initialize: function(callback, frequency) {
120
+ this.callback = callback;
121
+ this.frequency = frequency;
122
+ this.currentlyExecuting = false;
123
+
124
+ this.registerCallback();
125
+ },
126
+
127
+ registerCallback: function() {
128
+ this.timer = setInterval(this.onTimerEvent.bind(this), this.frequency * 1000);
129
+ },
130
+
131
+ stop: function() {
132
+ if (!this.timer) return;
133
+ clearInterval(this.timer);
134
+ this.timer = null;
135
+ },
136
+
137
+ onTimerEvent: function() {
138
+ if (!this.currentlyExecuting) {
139
+ try {
140
+ this.currentlyExecuting = true;
141
+ this.callback(this);
142
+ } finally {
143
+ this.currentlyExecuting = false;
144
+ }
145
+ }
146
+ }
147
+ }
148
+ String.interpret = function(value){
149
+ return value == null ? '' : String(value);
150
+ }
151
+
152
+ Object.extend(String.prototype, {
153
+ gsub: function(pattern, replacement) {
154
+ var result = '', source = this, match;
155
+ replacement = arguments.callee.prepareReplacement(replacement);
156
+
157
+ while (source.length > 0) {
158
+ if (match = source.match(pattern)) {
159
+ result += source.slice(0, match.index);
160
+ result += String.interpret(replacement(match));
161
+ source = source.slice(match.index + match[0].length);
162
+ } else {
163
+ result += source, source = '';
164
+ }
165
+ }
166
+ return result;
167
+ },
168
+
169
+ sub: function(pattern, replacement, count) {
170
+ replacement = this.gsub.prepareReplacement(replacement);
171
+ count = count === undefined ? 1 : count;
172
+
173
+ return this.gsub(pattern, function(match) {
174
+ if (--count < 0) return match[0];
175
+ return replacement(match);
176
+ });
177
+ },
178
+
179
+ scan: function(pattern, iterator) {
180
+ this.gsub(pattern, iterator);
181
+ return this;
182
+ },
183
+
184
+ truncate: function(length, truncation) {
185
+ length = length || 30;
186
+ truncation = truncation === undefined ? '...' : truncation;
187
+ return this.length > length ?
188
+ this.slice(0, length - truncation.length) + truncation : this;
189
+ },
190
+
191
+ strip: function() {
192
+ return this.replace(/^\s+/, '').replace(/\s+$/, '');
193
+ },
194
+
195
+ stripTags: function() {
196
+ return this.replace(/<\/?[^>]+>/gi, '');
197
+ },
198
+
199
+ stripScripts: function() {
200
+ return this.replace(new RegExp(Prototype.ScriptFragment, 'img'), '');
201
+ },
202
+
203
+ extractScripts: function() {
204
+ var matchAll = new RegExp(Prototype.ScriptFragment, 'img');
205
+ var matchOne = new RegExp(Prototype.ScriptFragment, 'im');
206
+ return (this.match(matchAll) || []).map(function(scriptTag) {
207
+ return (scriptTag.match(matchOne) || ['', ''])[1];
208
+ });
209
+ },
210
+
211
+ evalScripts: function() {
212
+ return this.extractScripts().map(function(script) { return eval(script) });
213
+ },
214
+
215
+ escapeHTML: function() {
216
+ var div = document.createElement('div');
217
+ var text = document.createTextNode(this);
218
+ div.appendChild(text);
219
+ return div.innerHTML;
220
+ },
221
+
222
+ unescapeHTML: function() {
223
+ var div = document.createElement('div');
224
+ div.innerHTML = this.stripTags();
225
+ return div.childNodes[0] ? (div.childNodes.length > 1 ?
226
+ $A(div.childNodes).inject('',function(memo,node){ return memo+node.nodeValue }) :
227
+ div.childNodes[0].nodeValue) : '';
228
+ },
229
+
230
+ toQueryParams: function(separator) {
231
+ var match = this.strip().match(/([^?#]*)(#.*)?$/);
232
+ if (!match) return {};
233
+
234
+ return match[1].split(separator || '&').inject({}, function(hash, pair) {
235
+ if ((pair = pair.split('='))[0]) {
236
+ var name = decodeURIComponent(pair[0]);
237
+ var value = pair[1] ? decodeURIComponent(pair[1]) : undefined;
238
+
239
+ if (hash[name] !== undefined) {
240
+ if (hash[name].constructor != Array)
241
+ hash[name] = [hash[name]];
242
+ if (value) hash[name].push(value);
243
+ }
244
+ else hash[name] = value;
245
+ }
246
+ return hash;
247
+ });
248
+ },
249
+
250
+ toArray: function() {
251
+ return this.split('');
252
+ },
253
+
254
+ succ: function() {
255
+ return this.slice(0, this.length - 1) +
256
+ String.fromCharCode(this.charCodeAt(this.length - 1) + 1);
257
+ },
258
+
259
+ camelize: function() {
260
+ var parts = this.split('-'), len = parts.length;
261
+ if (len == 1) return parts[0];
262
+
263
+ var camelized = this.charAt(0) == '-'
264
+ ? parts[0].charAt(0).toUpperCase() + parts[0].substring(1)
265
+ : parts[0];
266
+
267
+ for (var i = 1; i < len; i++)
268
+ camelized += parts[i].charAt(0).toUpperCase() + parts[i].substring(1);
269
+
270
+ return camelized;
271
+ },
272
+
273
+ capitalize: function(){
274
+ return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase();
275
+ },
276
+
277
+ underscore: function() {
278
+ return this.gsub(/::/, '/').gsub(/([A-Z]+)([A-Z][a-z])/,'#{1}_#{2}').gsub(/([a-z\d])([A-Z])/,'#{1}_#{2}').gsub(/-/,'_').toLowerCase();
279
+ },
280
+
281
+ dasherize: function() {
282
+ return this.gsub(/_/,'-');
283
+ },
284
+
285
+ inspect: function(useDoubleQuotes) {
286
+ var escapedString = this.replace(/\\/g, '\\\\');
287
+ if (useDoubleQuotes)
288
+ return '"' + escapedString.replace(/"/g, '\\"') + '"';
289
+ else
290
+ return "'" + escapedString.replace(/'/g, '\\\'') + "'";
291
+ }
292
+ });
293
+
294
+ String.prototype.gsub.prepareReplacement = function(replacement) {
295
+ if (typeof replacement == 'function') return replacement;
296
+ var template = new Template(replacement);
297
+ return function(match) { return template.evaluate(match) };
298
+ }
299
+
300
+ String.prototype.parseQuery = String.prototype.toQueryParams;
301
+
302
+ var Template = Class.create();
303
+ Template.Pattern = /(^|.|\r|\n)(#\{(.*?)\})/;
304
+ Template.prototype = {
305
+ initialize: function(template, pattern) {
306
+ this.template = template.toString();
307
+ this.pattern = pattern || Template.Pattern;
308
+ },
309
+
310
+ evaluate: function(object) {
311
+ return this.template.gsub(this.pattern, function(match) {
312
+ var before = match[1];
313
+ if (before == '\\') return match[2];
314
+ return before + String.interpret(object[match[3]]);
315
+ });
316
+ }
317
+ }
318
+
319
+ var $break = new Object();
320
+ var $continue = new Object();
321
+
322
+ var Enumerable = {
323
+ each: function(iterator) {
324
+ var index = 0;
325
+ try {
326
+ this._each(function(value) {
327
+ try {
328
+ iterator(value, index++);
329
+ } catch (e) {
330
+ if (e != $continue) throw e;
331
+ }
332
+ });
333
+ } catch (e) {
334
+ if (e != $break) throw e;
335
+ }
336
+ return this;
337
+ },
338
+
339
+ eachSlice: function(number, iterator) {
340
+ var index = -number, slices = [], array = this.toArray();
341
+ while ((index += number) < array.length)
342
+ slices.push(array.slice(index, index+number));
343
+ return slices.map(iterator);
344
+ },
345
+
346
+ all: function(iterator) {
347
+ var result = true;
348
+ this.each(function(value, index) {
349
+ result = result && !!(iterator || Prototype.K)(value, index);
350
+ if (!result) throw $break;
351
+ });
352
+ return result;
353
+ },
354
+
355
+ any: function(iterator) {
356
+ var result = false;
357
+ this.each(function(value, index) {
358
+ if (result = !!(iterator || Prototype.K)(value, index))
359
+ throw $break;
360
+ });
361
+ return result;
362
+ },
363
+
364
+ collect: function(iterator) {
365
+ var results = [];
366
+ this.each(function(value, index) {
367
+ results.push((iterator || Prototype.K)(value, index));
368
+ });
369
+ return results;
370
+ },
371
+
372
+ detect: function(iterator) {
373
+ var result;
374
+ this.each(function(value, index) {
375
+ if (iterator(value, index)) {
376
+ result = value;
377
+ throw $break;
378
+ }
379
+ });
380
+ return result;
381
+ },
382
+
383
+ findAll: function(iterator) {
384
+ var results = [];
385
+ this.each(function(value, index) {
386
+ if (iterator(value, index))
387
+ results.push(value);
388
+ });
389
+ return results;
390
+ },
391
+
392
+ grep: function(pattern, iterator) {
393
+ var results = [];
394
+ this.each(function(value, index) {
395
+ var stringValue = value.toString();
396
+ if (stringValue.match(pattern))
397
+ results.push((iterator || Prototype.K)(value, index));
398
+ })
399
+ return results;
400
+ },
401
+
402
+ include: function(object) {
403
+ var found = false;
404
+ this.each(function(value) {
405
+ if (value == object) {
406
+ found = true;
407
+ throw $break;
408
+ }
409
+ });
410
+ return found;
411
+ },
412
+
413
+ inGroupsOf: function(number, fillWith) {
414
+ fillWith = fillWith === undefined ? null : fillWith;
415
+ return this.eachSlice(number, function(slice) {
416
+ while(slice.length < number) slice.push(fillWith);
417
+ return slice;
418
+ });
419
+ },
420
+
421
+ inject: function(memo, iterator) {
422
+ this.each(function(value, index) {
423
+ memo = iterator(memo, value, index);
424
+ });
425
+ return memo;
426
+ },
427
+
428
+ invoke: function(method) {
429
+ var args = $A(arguments).slice(1);
430
+ return this.map(function(value) {
431
+ return value[method].apply(value, args);
432
+ });
433
+ },
434
+
435
+ max: function(iterator) {
436
+ var result;
437
+ this.each(function(value, index) {
438
+ value = (iterator || Prototype.K)(value, index);
439
+ if (result == undefined || value >= result)
440
+ result = value;
441
+ });
442
+ return result;
443
+ },
444
+
445
+ min: function(iterator) {
446
+ var result;
447
+ this.each(function(value, index) {
448
+ value = (iterator || Prototype.K)(value, index);
449
+ if (result == undefined || value < result)
450
+ result = value;
451
+ });
452
+ return result;
453
+ },
454
+
455
+ partition: function(iterator) {
456
+ var trues = [], falses = [];
457
+ this.each(function(value, index) {
458
+ ((iterator || Prototype.K)(value, index) ?
459
+ trues : falses).push(value);
460
+ });
461
+ return [trues, falses];
462
+ },
463
+
464
+ pluck: function(property) {
465
+ var results = [];
466
+ this.each(function(value, index) {
467
+ results.push(value[property]);
468
+ });
469
+ return results;
470
+ },
471
+
472
+ reject: function(iterator) {
473
+ var results = [];
474
+ this.each(function(value, index) {
475
+ if (!iterator(value, index))
476
+ results.push(value);
477
+ });
478
+ return results;
479
+ },
480
+
481
+ sortBy: function(iterator) {
482
+ return this.map(function(value, index) {
483
+ return {value: value, criteria: iterator(value, index)};
484
+ }).sort(function(left, right) {
485
+ var a = left.criteria, b = right.criteria;
486
+ return a < b ? -1 : a > b ? 1 : 0;
487
+ }).pluck('value');
488
+ },
489
+
490
+ toArray: function() {
491
+ return this.map();
492
+ },
493
+
494
+ zip: function() {
495
+ var iterator = Prototype.K, args = $A(arguments);
496
+ if (typeof args.last() == 'function')
497
+ iterator = args.pop();
498
+
499
+ var collections = [this].concat(args).map($A);
500
+ return this.map(function(value, index) {
501
+ return iterator(collections.pluck(index));
502
+ });
503
+ },
504
+
505
+ size: function() {
506
+ return this.toArray().length;
507
+ },
508
+
509
+ inspect: function() {
510
+ return '#<Enumerable:' + this.toArray().inspect() + '>';
511
+ }
512
+ }
513
+
514
+ Object.extend(Enumerable, {
515
+ map: Enumerable.collect,
516
+ find: Enumerable.detect,
517
+ select: Enumerable.findAll,
518
+ member: Enumerable.include,
519
+ entries: Enumerable.toArray
520
+ });
521
+ var $A = Array.from = function(iterable) {
522
+ if (!iterable) return [];
523
+ if (iterable.toArray) {
524
+ return iterable.toArray();
525
+ } else {
526
+ var results = [];
527
+ for (var i = 0, length = iterable.length; i < length; i++)
528
+ results.push(iterable[i]);
529
+ return results;
530
+ }
531
+ }
532
+
533
+ Object.extend(Array.prototype, Enumerable);
534
+
535
+ if (!Array.prototype._reverse)
536
+ Array.prototype._reverse = Array.prototype.reverse;
537
+
538
+ Object.extend(Array.prototype, {
539
+ _each: function(iterator) {
540
+ for (var i = 0, length = this.length; i < length; i++)
541
+ iterator(this[i]);
542
+ },
543
+
544
+ clear: function() {
545
+ this.length = 0;
546
+ return this;
547
+ },
548
+
549
+ first: function() {
550
+ return this[0];
551
+ },
552
+
553
+ last: function() {
554
+ return this[this.length - 1];
555
+ },
556
+
557
+ compact: function() {
558
+ return this.select(function(value) {
559
+ return value != null;
560
+ });
561
+ },
562
+
563
+ flatten: function() {
564
+ return this.inject([], function(array, value) {
565
+ return array.concat(value && value.constructor == Array ?
566
+ value.flatten() : [value]);
567
+ });
568
+ },
569
+
570
+ without: function() {
571
+ var values = $A(arguments);
572
+ return this.select(function(value) {
573
+ return !values.include(value);
574
+ });
575
+ },
576
+
577
+ indexOf: function(object) {
578
+ for (var i = 0, length = this.length; i < length; i++)
579
+ if (this[i] == object) return i;
580
+ return -1;
581
+ },
582
+
583
+ reverse: function(inline) {
584
+ return (inline !== false ? this : this.toArray())._reverse();
585
+ },
586
+
587
+ reduce: function() {
588
+ return this.length > 1 ? this : this[0];
589
+ },
590
+
591
+ uniq: function() {
592
+ return this.inject([], function(array, value) {
593
+ return array.include(value) ? array : array.concat([value]);
594
+ });
595
+ },
596
+
597
+ clone: function() {
598
+ return [].concat(this);
599
+ },
600
+
601
+ size: function() {
602
+ return this.length;
603
+ },
604
+
605
+ inspect: function() {
606
+ return '[' + this.map(Object.inspect).join(', ') + ']';
607
+ }
608
+ });
609
+
610
+ Array.prototype.toArray = Array.prototype.clone;
611
+
612
+ function $w(string){
613
+ string = string.strip();
614
+ return string ? string.split(/\s+/) : [];
615
+ }
616
+
617
+ if(window.opera){
618
+ Array.prototype.concat = function(){
619
+ 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++)
624
+ array.push(arguments[i][j]);
625
+ } else {
626
+ array.push(arguments[i]);
627
+ }
628
+ }
629
+ return array;
630
+ }
631
+ }
632
+ var Hash = function(obj) {
633
+ Object.extend(this, obj || {});
634
+ };
635
+
636
+ Object.extend(Hash, {
637
+ toQueryString: function(obj) {
638
+ var parts = [];
639
+
640
+ this.prototype._each.call(obj, function(pair) {
641
+ if (!pair.key) return;
642
+
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
+ }
654
+ }
655
+ if (pair.value == undefined) pair[1] = '';
656
+ parts.push(pair.map(encodeURIComponent).join('='));
657
+ });
658
+
659
+ return parts.join('&');
660
+ }
661
+ });
662
+
663
+ Object.extend(Hash.prototype, Enumerable);
664
+ Object.extend(Hash.prototype, {
665
+ _each: function(iterator) {
666
+ for (var key in this) {
667
+ var value = this[key];
668
+ if (value && value == Hash.prototype[key]) continue;
669
+
670
+ var pair = [key, value];
671
+ pair.key = key;
672
+ pair.value = value;
673
+ iterator(pair);
674
+ }
675
+ },
676
+
677
+ keys: function() {
678
+ return this.pluck('key');
679
+ },
680
+
681
+ values: function() {
682
+ return this.pluck('value');
683
+ },
684
+
685
+ merge: function(hash) {
686
+ return $H(hash).inject(this, function(mergedHash, pair) {
687
+ mergedHash[pair.key] = pair.value;
688
+ return mergedHash;
689
+ });
690
+ },
691
+
692
+ remove: function() {
693
+ var result;
694
+ for(var i = 0, length = arguments.length; i < length; i++) {
695
+ var value = this[arguments[i]];
696
+ if (value !== undefined){
697
+ if (result === undefined) result = value;
698
+ else {
699
+ if (result.constructor != Array) result = [result];
700
+ result.push(value)
701
+ }
702
+ }
703
+ delete this[arguments[i]];
704
+ }
705
+ return result;
706
+ },
707
+
708
+ toQueryString: function() {
709
+ return Hash.toQueryString(this);
710
+ },
711
+
712
+ inspect: function() {
713
+ return '#<Hash:{' + this.map(function(pair) {
714
+ return pair.map(Object.inspect).join(': ');
715
+ }).join(', ') + '}>';
716
+ }
717
+ });
718
+
719
+ function $H(object) {
720
+ if (object && object.constructor == Hash) return object;
721
+ return new Hash(object);
722
+ };
723
+ ObjectRange = Class.create();
724
+ Object.extend(ObjectRange.prototype, Enumerable);
725
+ Object.extend(ObjectRange.prototype, {
726
+ initialize: function(start, end, exclusive) {
727
+ this.start = start;
728
+ this.end = end;
729
+ this.exclusive = exclusive;
730
+ },
731
+
732
+ _each: function(iterator) {
733
+ var value = this.start;
734
+ while (this.include(value)) {
735
+ iterator(value);
736
+ value = value.succ();
737
+ }
738
+ },
739
+
740
+ include: function(value) {
741
+ if (value < this.start)
742
+ return false;
743
+ if (this.exclusive)
744
+ return value < this.end;
745
+ return value <= this.end;
746
+ }
747
+ });
748
+
749
+ var $R = function(start, end, exclusive) {
750
+ return new ObjectRange(start, end, exclusive);
751
+ }
752
+
753
+ var Ajax = {
754
+ getTransport: function() {
755
+ return Try.these(
756
+ function() {return new XMLHttpRequest()},
757
+ function() {return new ActiveXObject('Msxml2.XMLHTTP')},
758
+ function() {return new ActiveXObject('Microsoft.XMLHTTP')}
759
+ ) || false;
760
+ },
761
+
762
+ activeRequestCount: 0
763
+ }
764
+
765
+ Ajax.Responders = {
766
+ responders: [],
767
+
768
+ _each: function(iterator) {
769
+ this.responders._each(iterator);
770
+ },
771
+
772
+ register: function(responder) {
773
+ if (!this.include(responder))
774
+ this.responders.push(responder);
775
+ },
776
+
777
+ unregister: function(responder) {
778
+ this.responders = this.responders.without(responder);
779
+ },
780
+
781
+ dispatch: function(callback, request, transport, json) {
782
+ this.each(function(responder) {
783
+ if (typeof responder[callback] == 'function') {
784
+ try {
785
+ responder[callback].apply(responder, [request, transport, json]);
786
+ } catch (e) {}
787
+ }
788
+ });
789
+ }
790
+ };
791
+
792
+ Object.extend(Ajax.Responders, Enumerable);
793
+
794
+ Ajax.Responders.register({
795
+ onCreate: function() {
796
+ Ajax.activeRequestCount++;
797
+ },
798
+ onComplete: function() {
799
+ Ajax.activeRequestCount--;
800
+ }
801
+ });
802
+
803
+ Ajax.Base = function() {};
804
+ Ajax.Base.prototype = {
805
+ setOptions: function(options) {
806
+ this.options = {
807
+ method: 'post',
808
+ asynchronous: true,
809
+ contentType: 'application/x-www-form-urlencoded',
810
+ encoding: 'UTF-8',
811
+ parameters: ''
812
+ }
813
+ Object.extend(this.options, options || {});
814
+
815
+ this.options.method = this.options.method.toLowerCase();
816
+ if (typeof this.options.parameters == 'string')
817
+ this.options.parameters = this.options.parameters.toQueryParams();
818
+ }
819
+ }
820
+
821
+ Ajax.Request = Class.create();
822
+ Ajax.Request.Events =
823
+ ['Uninitialized', 'Loading', 'Loaded', 'Interactive', 'Complete'];
824
+
825
+ Ajax.Request.prototype = Object.extend(new Ajax.Base(), {
826
+ _complete: false,
827
+
828
+ initialize: function(url, options) {
829
+ this.transport = Ajax.getTransport();
830
+ this.setOptions(options);
831
+ this.request(url);
832
+ },
833
+
834
+ request: function(url) {
835
+ this.url = url;
836
+ this.method = this.options.method;
837
+ var params = this.options.parameters;
838
+
839
+ if (!['get', 'post'].include(this.method)) {
840
+ // simulate other verbs over post
841
+ params['_method'] = this.method;
842
+ this.method = 'post';
843
+ }
844
+
845
+ params = Hash.toQueryString(params);
846
+ if (params && /Konqueror|Safari|KHTML/.test(navigator.userAgent)) params += '&_='
847
+
848
+ // when GET, append parameters to URL
849
+ if (this.method == 'get' && params)
850
+ this.url += (this.url.indexOf('?') > -1 ? '&' : '?') + params;
851
+
852
+ try {
853
+ Ajax.Responders.dispatch('onCreate', this, this.transport);
854
+
855
+ this.transport.open(this.method.toUpperCase(), this.url,
856
+ this.options.asynchronous);
857
+
858
+ if (this.options.asynchronous)
859
+ setTimeout(function() { this.respondToReadyState(1) }.bind(this), 10);
860
+
861
+ this.transport.onreadystatechange = this.onStateChange.bind(this);
862
+ this.setRequestHeaders();
863
+
864
+ var body = this.method == 'post' ? (this.options.postBody || params) : null;
865
+
866
+ this.transport.send(body);
867
+
868
+ /* Force Firefox to handle ready state 4 for synchronous requests */
869
+ if (!this.options.asynchronous && this.transport.overrideMimeType)
870
+ this.onStateChange();
871
+
872
+ }
873
+ catch (e) {
874
+ this.dispatchException(e);
875
+ }
876
+ },
877
+
878
+ onStateChange: function() {
879
+ var readyState = this.transport.readyState;
880
+ if (readyState > 1 && !((readyState == 4) && this._complete))
881
+ this.respondToReadyState(this.transport.readyState);
882
+ },
883
+
884
+ setRequestHeaders: function() {
885
+ var headers = {
886
+ 'X-Requested-With': 'XMLHttpRequest',
887
+ 'X-Prototype-Version': Prototype.Version,
888
+ 'Accept': 'text/javascript, text/html, application/xml, text/xml, */*'
889
+ };
890
+
891
+ if (this.method == 'post') {
892
+ headers['Content-type'] = this.options.contentType +
893
+ (this.options.encoding ? '; charset=' + this.options.encoding : '');
894
+
895
+ /* Force "Connection: close" for older Mozilla browsers to work
896
+ * around a bug where XMLHttpRequest sends an incorrect
897
+ * Content-length header. See Mozilla Bugzilla #246651.
898
+ */
899
+ if (this.transport.overrideMimeType &&
900
+ (navigator.userAgent.match(/Gecko\/(\d{4})/) || [0,2005])[1] < 2005)
901
+ headers['Connection'] = 'close';
902
+ }
903
+
904
+ // user-defined headers
905
+ if (typeof this.options.requestHeaders == 'object') {
906
+ var extras = this.options.requestHeaders;
907
+
908
+ if (typeof extras.push == 'function')
909
+ for (var i = 0, length = extras.length; i < length; i += 2)
910
+ headers[extras[i]] = extras[i+1];
911
+ else
912
+ $H(extras).each(function(pair) { headers[pair.key] = pair.value });
913
+ }
914
+
915
+ for (var name in headers)
916
+ this.transport.setRequestHeader(name, headers[name]);
917
+ },
918
+
919
+ success: function() {
920
+ return !this.transport.status
921
+ || (this.transport.status >= 200 && this.transport.status < 300);
922
+ },
923
+
924
+ respondToReadyState: function(readyState) {
925
+ var state = Ajax.Request.Events[readyState];
926
+ var transport = this.transport, json = this.evalJSON();
927
+
928
+ if (state == 'Complete') {
929
+ try {
930
+ this._complete = true;
931
+ (this.options['on' + this.transport.status]
932
+ || this.options['on' + (this.success() ? 'Success' : 'Failure')]
933
+ || Prototype.emptyFunction)(transport, json);
934
+ } catch (e) {
935
+ this.dispatchException(e);
936
+ }
937
+
938
+ if ((this.getHeader('Content-type') || 'text/javascript').strip().
939
+ match(/^(text|application)\/(x-)?(java|ecma)script(;.*)?$/i))
940
+ this.evalResponse();
941
+ }
942
+
943
+ try {
944
+ (this.options['on' + state] || Prototype.emptyFunction)(transport, json);
945
+ Ajax.Responders.dispatch('on' + state, this, transport, json);
946
+ } catch (e) {
947
+ this.dispatchException(e);
948
+ }
949
+
950
+ if (state == 'Complete') {
951
+ // avoid memory leak in MSIE: clean up
952
+ this.transport.onreadystatechange = Prototype.emptyFunction;
953
+ }
954
+ },
955
+
956
+ getHeader: function(name) {
957
+ try {
958
+ return this.transport.getResponseHeader(name);
959
+ } catch (e) { return null }
960
+ },
961
+
962
+ evalJSON: function() {
963
+ try {
964
+ var json = this.getHeader('X-JSON');
965
+ return json ? eval('(' + json + ')') : null;
966
+ } catch (e) { return null }
967
+ },
968
+
969
+ evalResponse: function() {
970
+ try {
971
+ return eval(this.transport.responseText);
972
+ } catch (e) {
973
+ this.dispatchException(e);
974
+ }
975
+ },
976
+
977
+ dispatchException: function(exception) {
978
+ (this.options.onException || Prototype.emptyFunction)(this, exception);
979
+ Ajax.Responders.dispatch('onException', this, exception);
980
+ }
981
+ });
982
+
983
+ Ajax.Updater = Class.create();
984
+
985
+ Object.extend(Object.extend(Ajax.Updater.prototype, Ajax.Request.prototype), {
986
+ initialize: function(container, url, options) {
987
+ this.container = {
988
+ success: (container.success || container),
989
+ failure: (container.failure || (container.success ? null : container))
990
+ }
991
+
992
+ this.transport = Ajax.getTransport();
993
+ this.setOptions(options);
994
+
995
+ var onComplete = this.options.onComplete || Prototype.emptyFunction;
996
+ this.options.onComplete = (function(transport, param) {
997
+ this.updateContent();
998
+ onComplete(transport, param);
999
+ }).bind(this);
1000
+
1001
+ this.request(url);
1002
+ },
1003
+
1004
+ updateContent: function() {
1005
+ var receiver = this.container[this.success() ? 'success' : 'failure'];
1006
+ var response = this.transport.responseText;
1007
+
1008
+ if (!this.options.evalScripts) response = response.stripScripts();
1009
+
1010
+ if (receiver = $(receiver)) {
1011
+ if (this.options.insertion)
1012
+ new this.options.insertion(receiver, response);
1013
+ else
1014
+ receiver.update(response);
1015
+ }
1016
+
1017
+ if (this.success()) {
1018
+ if (this.onComplete)
1019
+ setTimeout(this.onComplete.bind(this), 10);
1020
+ }
1021
+ }
1022
+ });
1023
+
1024
+ Ajax.PeriodicalUpdater = Class.create();
1025
+ Ajax.PeriodicalUpdater.prototype = Object.extend(new Ajax.Base(), {
1026
+ initialize: function(container, url, options) {
1027
+ this.setOptions(options);
1028
+ this.onComplete = this.options.onComplete;
1029
+
1030
+ this.frequency = (this.options.frequency || 2);
1031
+ this.decay = (this.options.decay || 1);
1032
+
1033
+ this.updater = {};
1034
+ this.container = container;
1035
+ this.url = url;
1036
+
1037
+ this.start();
1038
+ },
1039
+
1040
+ start: function() {
1041
+ this.options.onComplete = this.updateComplete.bind(this);
1042
+ this.onTimerEvent();
1043
+ },
1044
+
1045
+ stop: function() {
1046
+ this.updater.options.onComplete = undefined;
1047
+ clearTimeout(this.timer);
1048
+ (this.onComplete || Prototype.emptyFunction).apply(this, arguments);
1049
+ },
1050
+
1051
+ updateComplete: function(request) {
1052
+ if (this.options.decay) {
1053
+ this.decay = (request.responseText == this.lastText ?
1054
+ this.decay * this.options.decay : 1);
1055
+
1056
+ this.lastText = request.responseText;
1057
+ }
1058
+ this.timer = setTimeout(this.onTimerEvent.bind(this),
1059
+ this.decay * this.frequency * 1000);
1060
+ },
1061
+
1062
+ onTimerEvent: function() {
1063
+ this.updater = new Ajax.Updater(this.container, this.url, this.options);
1064
+ }
1065
+ });
1066
+ function $(element) {
1067
+ if (arguments.length > 1) {
1068
+ for (var i = 0, elements = [], length = arguments.length; i < length; i++)
1069
+ elements.push($(arguments[i]));
1070
+ return elements;
1071
+ }
1072
+ if (typeof element == 'string')
1073
+ element = document.getElementById(element);
1074
+ return Element.extend(element);
1075
+ }
1076
+
1077
+ if (Prototype.BrowserFeatures.XPath) {
1078
+ document._getElementsByXPath = function(expression, parentElement) {
1079
+ var results = [];
1080
+ var query = document.evaluate(expression, $(parentElement) || document,
1081
+ null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null);
1082
+ for (var i = 0, length = query.snapshotLength; i < length; i++)
1083
+ results.push(query.snapshotItem(i));
1084
+ return results;
1085
+ };
1086
+ }
1087
+
1088
+ document.getElementsByClassName = function(className, parentElement) {
1089
+ if (Prototype.BrowserFeatures.XPath) {
1090
+ var q = ".//*[contains(concat(' ', @class, ' '), ' " + className + " ')]";
1091
+ 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;
1101
+ }
1102
+ };
1103
+
1104
+ /*--------------------------------------------------------------------------*/
1105
+
1106
+ if (!window.Element)
1107
+ var Element = new Object();
1108
+
1109
+ Element.extend = function(element) {
1110
+ if (!element || _nativeExtensions || element.nodeType == 3) return element;
1111
+
1112
+ if (!element._extended && element.tagName && element != window) {
1113
+ var methods = Object.clone(Element.Methods), cache = Element.extend.cache;
1114
+
1115
+ if (element.tagName.toUpperCase() == 'FORM')
1116
+ Object.extend(methods, Form.Methods);
1117
+ if (['INPUT', 'TEXTAREA', 'SELECT'].include(element.tagName.toUpperCase()))
1118
+ Object.extend(methods, Form.Element.Methods);
1119
+
1120
+ Object.extend(methods, Element.Methods.Simulated);
1121
+
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
+ }
1127
+ }
1128
+
1129
+ element._extended = true;
1130
+ return element;
1131
+ };
1132
+
1133
+ Element.extend.cache = {
1134
+ findOrStore: function(value) {
1135
+ return this[value] = this[value] || function() {
1136
+ return value.apply(null, [this].concat($A(arguments)));
1137
+ }
1138
+ }
1139
+ };
1140
+
1141
+ Element.Methods = {
1142
+ visible: function(element) {
1143
+ return $(element).style.display != 'none';
1144
+ },
1145
+
1146
+ toggle: function(element) {
1147
+ element = $(element);
1148
+ Element[Element.visible(element) ? 'hide' : 'show'](element);
1149
+ return element;
1150
+ },
1151
+
1152
+ hide: function(element) {
1153
+ $(element).style.display = 'none';
1154
+ return element;
1155
+ },
1156
+
1157
+ show: function(element) {
1158
+ $(element).style.display = '';
1159
+ return element;
1160
+ },
1161
+
1162
+ remove: function(element) {
1163
+ element = $(element);
1164
+ element.parentNode.removeChild(element);
1165
+ return element;
1166
+ },
1167
+
1168
+ update: function(element, html) {
1169
+ html = typeof html == 'undefined' ? '' : html.toString();
1170
+ $(element).innerHTML = html.stripScripts();
1171
+ setTimeout(function() {html.evalScripts()}, 10);
1172
+ return element;
1173
+ },
1174
+
1175
+ replace: function(element, html) {
1176
+ element = $(element);
1177
+ html = typeof html == 'undefined' ? '' : html.toString();
1178
+ if (element.outerHTML) {
1179
+ element.outerHTML = html.stripScripts();
1180
+ } else {
1181
+ var range = element.ownerDocument.createRange();
1182
+ range.selectNodeContents(element);
1183
+ element.parentNode.replaceChild(
1184
+ range.createContextualFragment(html.stripScripts()), element);
1185
+ }
1186
+ setTimeout(function() {html.evalScripts()}, 10);
1187
+ return element;
1188
+ },
1189
+
1190
+ inspect: function(element) {
1191
+ element = $(element);
1192
+ var result = '<' + element.tagName.toLowerCase();
1193
+ $H({'id': 'id', 'className': 'class'}).each(function(pair) {
1194
+ var property = pair.first(), attribute = pair.last();
1195
+ var value = (element[property] || '').toString();
1196
+ if (value) result += ' ' + attribute + '=' + value.inspect(true);
1197
+ });
1198
+ return result + '>';
1199
+ },
1200
+
1201
+ recursivelyCollect: function(element, property) {
1202
+ element = $(element);
1203
+ var elements = [];
1204
+ while (element = element[property])
1205
+ if (element.nodeType == 1)
1206
+ elements.push(Element.extend(element));
1207
+ return elements;
1208
+ },
1209
+
1210
+ ancestors: function(element) {
1211
+ return $(element).recursivelyCollect('parentNode');
1212
+ },
1213
+
1214
+ descendants: function(element) {
1215
+ return $A($(element).getElementsByTagName('*'));
1216
+ },
1217
+
1218
+ immediateDescendants: function(element) {
1219
+ if (!(element = $(element).firstChild)) return [];
1220
+ while (element && element.nodeType != 1) element = element.nextSibling;
1221
+ if (element) return [element].concat($(element).nextSiblings());
1222
+ return [];
1223
+ },
1224
+
1225
+ previousSiblings: function(element) {
1226
+ return $(element).recursivelyCollect('previousSibling');
1227
+ },
1228
+
1229
+ nextSiblings: function(element) {
1230
+ return $(element).recursivelyCollect('nextSibling');
1231
+ },
1232
+
1233
+ siblings: function(element) {
1234
+ element = $(element);
1235
+ return element.previousSiblings().reverse().concat(element.nextSiblings());
1236
+ },
1237
+
1238
+ match: function(element, selector) {
1239
+ if (typeof selector == 'string')
1240
+ selector = new Selector(selector);
1241
+ return selector.match($(element));
1242
+ },
1243
+
1244
+ up: function(element, expression, index) {
1245
+ return Selector.findElement($(element).ancestors(), expression, index);
1246
+ },
1247
+
1248
+ down: function(element, expression, index) {
1249
+ return Selector.findElement($(element).descendants(), expression, index);
1250
+ },
1251
+
1252
+ previous: function(element, expression, index) {
1253
+ return Selector.findElement($(element).previousSiblings(), expression, index);
1254
+ },
1255
+
1256
+ next: function(element, expression, index) {
1257
+ return Selector.findElement($(element).nextSiblings(), expression, index);
1258
+ },
1259
+
1260
+ getElementsBySelector: function() {
1261
+ var args = $A(arguments), element = $(args.shift());
1262
+ return Selector.findChildElements(element, args);
1263
+ },
1264
+
1265
+ getElementsByClassName: function(element, className) {
1266
+ return document.getElementsByClassName(className, element);
1267
+ },
1268
+
1269
+ readAttribute: function(element, name) {
1270
+ element = $(element);
1271
+ if (document.all && !window.opera) {
1272
+ var t = Element._attributeTranslations;
1273
+ if (t.values[name]) return t.values[name](element, name);
1274
+ if (t.names[name]) name = t.names[name];
1275
+ var attribute = element.attributes[name];
1276
+ if(attribute) return attribute.nodeValue;
1277
+ }
1278
+ return element.getAttribute(name);
1279
+ },
1280
+
1281
+ getHeight: function(element) {
1282
+ return $(element).getDimensions().height;
1283
+ },
1284
+
1285
+ getWidth: function(element) {
1286
+ return $(element).getDimensions().width;
1287
+ },
1288
+
1289
+ classNames: function(element) {
1290
+ return new Element.ClassNames(element);
1291
+ },
1292
+
1293
+ hasClassName: function(element, className) {
1294
+ if (!(element = $(element))) return;
1295
+ var elementClassName = element.className;
1296
+ if (elementClassName.length == 0) return false;
1297
+ if (elementClassName == className ||
1298
+ elementClassName.match(new RegExp("(^|\\s)" + className + "(\\s|$)")))
1299
+ return true;
1300
+ return false;
1301
+ },
1302
+
1303
+ addClassName: function(element, className) {
1304
+ if (!(element = $(element))) return;
1305
+ Element.classNames(element).add(className);
1306
+ return element;
1307
+ },
1308
+
1309
+ removeClassName: function(element, className) {
1310
+ if (!(element = $(element))) return;
1311
+ Element.classNames(element).remove(className);
1312
+ return element;
1313
+ },
1314
+
1315
+ toggleClassName: function(element, className) {
1316
+ if (!(element = $(element))) return;
1317
+ Element.classNames(element)[element.hasClassName(className) ? 'remove' : 'add'](className);
1318
+ return element;
1319
+ },
1320
+
1321
+ observe: function() {
1322
+ Event.observe.apply(Event, arguments);
1323
+ return $A(arguments).first();
1324
+ },
1325
+
1326
+ stopObserving: function() {
1327
+ Event.stopObserving.apply(Event, arguments);
1328
+ return $A(arguments).first();
1329
+ },
1330
+
1331
+ // removes whitespace-only text node children
1332
+ cleanWhitespace: function(element) {
1333
+ element = $(element);
1334
+ var node = element.firstChild;
1335
+ while (node) {
1336
+ var nextNode = node.nextSibling;
1337
+ if (node.nodeType == 3 && !/\S/.test(node.nodeValue))
1338
+ element.removeChild(node);
1339
+ node = nextNode;
1340
+ }
1341
+ return element;
1342
+ },
1343
+
1344
+ empty: function(element) {
1345
+ return $(element).innerHTML.match(/^\s*$/);
1346
+ },
1347
+
1348
+ descendantOf: function(element, ancestor) {
1349
+ element = $(element), ancestor = $(ancestor);
1350
+ while (element = element.parentNode)
1351
+ if (element == ancestor) return true;
1352
+ return false;
1353
+ },
1354
+
1355
+ scrollTo: function(element) {
1356
+ element = $(element);
1357
+ var pos = Position.cumulativeOffset(element);
1358
+ window.scrollTo(pos[0], pos[1]);
1359
+ return element;
1360
+ },
1361
+
1362
+ getStyle: function(element, style) {
1363
+ element = $(element);
1364
+ if (['float','cssFloat'].include(style))
1365
+ style = (typeof element.style.styleFloat != 'undefined' ? 'styleFloat' : 'cssFloat');
1366
+ style = style.camelize();
1367
+ var value = element.style[style];
1368
+ 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
+ }
1375
+ }
1376
+
1377
+ if((value == 'auto') && ['width','height'].include(style) && (element.getStyle('display') != 'none'))
1378
+ value = element['offset'+style.capitalize()] + 'px';
1379
+
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;
1389
+ },
1390
+
1391
+ setStyle: function(element, style) {
1392
+ 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
+ }
1413
+ return element;
1414
+ },
1415
+
1416
+ getDimensions: function(element) {
1417
+ element = $(element);
1418
+ var display = $(element).getStyle('display');
1419
+ if (display != 'none' && display != null) // Safari bug
1420
+ return {width: element.offsetWidth, height: element.offsetHeight};
1421
+
1422
+ // All *Width and *Height properties give 0 on elements with display none,
1423
+ // so enable the element temporarily
1424
+ var els = element.style;
1425
+ var originalVisibility = els.visibility;
1426
+ var originalPosition = els.position;
1427
+ var originalDisplay = els.display;
1428
+ els.visibility = 'hidden';
1429
+ els.position = 'absolute';
1430
+ els.display = 'block';
1431
+ var originalWidth = element.clientWidth;
1432
+ var originalHeight = element.clientHeight;
1433
+ els.display = originalDisplay;
1434
+ els.position = originalPosition;
1435
+ els.visibility = originalVisibility;
1436
+ return {width: originalWidth, height: originalHeight};
1437
+ },
1438
+
1439
+ makePositioned: function(element) {
1440
+ element = $(element);
1441
+ var pos = Element.getStyle(element, 'position');
1442
+ if (pos == 'static' || !pos) {
1443
+ element._madePositioned = true;
1444
+ element.style.position = 'relative';
1445
+ // Opera returns the offset relative to the positioning context, when an
1446
+ // element is position relative but top and left have not been defined
1447
+ if (window.opera) {
1448
+ element.style.top = 0;
1449
+ element.style.left = 0;
1450
+ }
1451
+ }
1452
+ return element;
1453
+ },
1454
+
1455
+ undoPositioned: function(element) {
1456
+ element = $(element);
1457
+ if (element._madePositioned) {
1458
+ element._madePositioned = undefined;
1459
+ element.style.position =
1460
+ element.style.top =
1461
+ element.style.left =
1462
+ element.style.bottom =
1463
+ element.style.right = '';
1464
+ }
1465
+ return element;
1466
+ },
1467
+
1468
+ makeClipping: function(element) {
1469
+ element = $(element);
1470
+ if (element._overflow) return element;
1471
+ element._overflow = element.style.overflow || 'auto';
1472
+ if ((Element.getStyle(element, 'overflow') || 'visible') != 'hidden')
1473
+ element.style.overflow = 'hidden';
1474
+ return element;
1475
+ },
1476
+
1477
+ undoClipping: function(element) {
1478
+ element = $(element);
1479
+ if (!element._overflow) return element;
1480
+ element.style.overflow = element._overflow == 'auto' ? '' : element._overflow;
1481
+ element._overflow = null;
1482
+ return element;
1483
+ }
1484
+ };
1485
+
1486
+ Object.extend(Element.Methods, {childOf: Element.Methods.descendantOf});
1487
+
1488
+ Element._attributeTranslations = {};
1489
+
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
+ };
1502
+
1503
+ Element._attributeTranslations.values = {
1504
+ _getAttr: function(element, attribute) {
1505
+ return element.getAttribute(attribute, 2);
1506
+ },
1507
+
1508
+ _flag: function(element, attribute) {
1509
+ return $(element).hasAttribute(attribute) ? attribute : null;
1510
+ },
1511
+
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){
1541
+ Element.Methods.update = function(element, html) {
1542
+ element = $(element);
1543
+ html = typeof html == 'undefined' ? '' : html.toString();
1544
+ var tagName = element.tagName.toUpperCase();
1545
+ if (['THEAD','TBODY','TR','TD'].include(tagName)) {
1546
+ var div = document.createElement('div');
1547
+ switch (tagName) {
1548
+ case 'THEAD':
1549
+ case 'TBODY':
1550
+ div.innerHTML = '<table><tbody>' + html.stripScripts() + '</tbody></table>';
1551
+ depth = 2;
1552
+ break;
1553
+ case 'TR':
1554
+ div.innerHTML = '<table><tbody><tr>' + html.stripScripts() + '</tr></tbody></table>';
1555
+ depth = 3;
1556
+ break;
1557
+ case 'TD':
1558
+ div.innerHTML = '<table><tbody><tr><td>' + html.stripScripts() + '</td></tr></tbody></table>';
1559
+ depth = 4;
1560
+ }
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) });
1568
+ } else {
1569
+ element.innerHTML = html.stripScripts();
1570
+ }
1571
+ setTimeout(function() {html.evalScripts()}, 10);
1572
+ return element;
1573
+ }
1574
+ };
1575
+
1576
+ Object.extend(Element, Element.Methods);
1577
+
1578
+ var _nativeExtensions = false;
1579
+
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
+ });
1587
+
1588
+ Element.addMethods = function(methods) {
1589
+ Object.extend(Element.Methods, methods || {});
1590
+
1591
+ function copy(methods, destination, onlyIfAbsent) {
1592
+ onlyIfAbsent = onlyIfAbsent || false;
1593
+ var cache = Element.extend.cache;
1594
+ for (var property in methods) {
1595
+ var value = methods[property];
1596
+ if (!onlyIfAbsent || !(property in destination))
1597
+ destination[property] = cache.findOrStore(value);
1598
+ }
1599
+ }
1600
+
1601
+ if (typeof HTMLElement != 'undefined') {
1602
+ copy(Element.Methods, HTMLElement.prototype);
1603
+ 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;
1609
+ }
1610
+ }
1611
+
1612
+ var Toggle = new Object();
1613
+ Toggle.display = Element.toggle;
1614
+
1615
+ /*--------------------------------------------------------------------------*/
1616
+
1617
+ Abstract.Insertion = function(adjacency) {
1618
+ this.adjacency = adjacency;
1619
+ }
1620
+
1621
+ Abstract.Insertion.prototype = {
1622
+ initialize: function(element, content) {
1623
+ this.element = $(element);
1624
+ this.content = content.stripScripts();
1625
+
1626
+ if (this.adjacency && this.element.insertAdjacentHTML) {
1627
+ try {
1628
+ this.element.insertAdjacentHTML(this.adjacency, this.content);
1629
+ } catch (e) {
1630
+ var tagName = this.element.tagName.toUpperCase();
1631
+ if (['TBODY', 'TR'].include(tagName)) {
1632
+ this.insertContent(this.contentFromAnonymousTable());
1633
+ } else {
1634
+ throw e;
1635
+ }
1636
+ }
1637
+ } else {
1638
+ this.range = this.element.ownerDocument.createRange();
1639
+ if (this.initializeRange) this.initializeRange();
1640
+ this.insertContent([this.range.createContextualFragment(this.content)]);
1641
+ }
1642
+
1643
+ setTimeout(function() {content.evalScripts()}, 10);
1644
+ },
1645
+
1646
+ contentFromAnonymousTable: function() {
1647
+ var div = document.createElement('div');
1648
+ div.innerHTML = '<table><tbody>' + this.content + '</tbody></table>';
1649
+ return $A(div.childNodes[0].childNodes[0].childNodes);
1650
+ }
1651
+ }
1652
+
1653
+ var Insertion = new Object();
1654
+
1655
+ Insertion.Before = Class.create();
1656
+ Insertion.Before.prototype = Object.extend(new Abstract.Insertion('beforeBegin'), {
1657
+ initializeRange: function() {
1658
+ this.range.setStartBefore(this.element);
1659
+ },
1660
+
1661
+ insertContent: function(fragments) {
1662
+ fragments.each((function(fragment) {
1663
+ this.element.parentNode.insertBefore(fragment, this.element);
1664
+ }).bind(this));
1665
+ }
1666
+ });
1667
+
1668
+ Insertion.Top = Class.create();
1669
+ Insertion.Top.prototype = Object.extend(new Abstract.Insertion('afterBegin'), {
1670
+ initializeRange: function() {
1671
+ this.range.selectNodeContents(this.element);
1672
+ this.range.collapse(true);
1673
+ },
1674
+
1675
+ insertContent: function(fragments) {
1676
+ fragments.reverse(false).each((function(fragment) {
1677
+ this.element.insertBefore(fragment, this.element.firstChild);
1678
+ }).bind(this));
1679
+ }
1680
+ });
1681
+
1682
+ Insertion.Bottom = Class.create();
1683
+ Insertion.Bottom.prototype = Object.extend(new Abstract.Insertion('beforeEnd'), {
1684
+ initializeRange: function() {
1685
+ this.range.selectNodeContents(this.element);
1686
+ this.range.collapse(this.element);
1687
+ },
1688
+
1689
+ insertContent: function(fragments) {
1690
+ fragments.each((function(fragment) {
1691
+ this.element.appendChild(fragment);
1692
+ }).bind(this));
1693
+ }
1694
+ });
1695
+
1696
+ Insertion.After = Class.create();
1697
+ Insertion.After.prototype = Object.extend(new Abstract.Insertion('afterEnd'), {
1698
+ initializeRange: function() {
1699
+ this.range.setStartAfter(this.element);
1700
+ },
1701
+
1702
+ insertContent: function(fragments) {
1703
+ fragments.each((function(fragment) {
1704
+ this.element.parentNode.insertBefore(fragment,
1705
+ this.element.nextSibling);
1706
+ }).bind(this));
1707
+ }
1708
+ });
1709
+
1710
+ /*--------------------------------------------------------------------------*/
1711
+
1712
+ Element.ClassNames = Class.create();
1713
+ Element.ClassNames.prototype = {
1714
+ initialize: function(element) {
1715
+ this.element = $(element);
1716
+ },
1717
+
1718
+ _each: function(iterator) {
1719
+ this.element.className.split(/\s+/).select(function(name) {
1720
+ return name.length > 0;
1721
+ })._each(iterator);
1722
+ },
1723
+
1724
+ set: function(className) {
1725
+ this.element.className = className;
1726
+ },
1727
+
1728
+ add: function(classNameToAdd) {
1729
+ if (this.include(classNameToAdd)) return;
1730
+ this.set($A(this).concat(classNameToAdd).join(' '));
1731
+ },
1732
+
1733
+ remove: function(classNameToRemove) {
1734
+ if (!this.include(classNameToRemove)) return;
1735
+ this.set($A(this).without(classNameToRemove).join(' '));
1736
+ },
1737
+
1738
+ toString: function() {
1739
+ return $A(this).join(' ');
1740
+ }
1741
+ };
1742
+
1743
+ Object.extend(Element.ClassNames.prototype, Enumerable);
1744
+ var Selector = Class.create();
1745
+ Selector.prototype = {
1746
+ initialize: function(expression) {
1747
+ this.params = {classNames: []};
1748
+ this.expression = expression.toString().strip();
1749
+ this.parseExpression();
1750
+ this.compileMatcher();
1751
+ },
1752
+
1753
+ parseExpression: function() {
1754
+ function abort(message) { throw 'Parse error in selector: ' + message; }
1755
+
1756
+ if (this.expression == '') abort('empty expression');
1757
+
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];
1763
+ }
1764
+
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());
1775
+ }
1776
+ expr = rest;
1777
+ }
1778
+
1779
+ if (expr.length > 0) abort(expr.inspect());
1780
+ },
1781
+
1782
+ buildMatchExpression: function() {
1783
+ var params = this.params, conditions = [], clause;
1784
+
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() + ')';
1799
+ }
1800
+
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';
1811
+ }
1812
+ });
1813
+ }
1814
+
1815
+ return conditions.join(' && ');
1816
+ },
1817
+
1818
+ compileMatcher: function() {
1819
+ this.match = new Function('element', 'if (!element.tagName) return false; \
1820
+ element = $(element); \
1821
+ return ' + this.buildMatchExpression());
1822
+ },
1823
+
1824
+ findElements: function(scope) {
1825
+ var element;
1826
+
1827
+ if (element = $(this.params.id))
1828
+ if (this.match(element))
1829
+ if (!scope || Element.childOf(element, scope))
1830
+ return [element];
1831
+
1832
+ scope = (scope || document).getElementsByTagName(this.params.tagName || '*');
1833
+
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));
1838
+
1839
+ return results;
1840
+ },
1841
+
1842
+ toString: function() {
1843
+ return this.expression;
1844
+ }
1845
+ }
1846
+
1847
+ Object.extend(Selector, {
1848
+ matchElements: function(elements, expression) {
1849
+ var selector = new Selector(expression);
1850
+ return elements.select(selector.match.bind(selector)).map(Element.extend);
1851
+ },
1852
+
1853
+ findElement: function(elements, expression, index) {
1854
+ if (typeof expression == 'number') index = expression, expression = false;
1855
+ return Selector.matchElements(elements, expression || '*')[index || 0];
1856
+ },
1857
+
1858
+ 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();
1867
+ }
1868
+ });
1869
+
1870
+ function $$() {
1871
+ return Selector.findChildElements(document, $A(arguments));
1872
+ }
1873
+ var Form = {
1874
+ reset: function(form) {
1875
+ $(form).reset();
1876
+ return form;
1877
+ },
1878
+
1879
+ serializeElements: function(elements, getHash) {
1880
+ var data = elements.inject({}, function(result, element) {
1881
+ if (!element.disabled && element.name) {
1882
+ var key = element.name, value = $(element).getValue();
1883
+ if (value != undefined) {
1884
+ if (result[key]) {
1885
+ if (result[key].constructor != Array) result[key] = [result[key]];
1886
+ result[key].push(value);
1887
+ }
1888
+ else result[key] = value;
1889
+ }
1890
+ }
1891
+ return result;
1892
+ });
1893
+
1894
+ return getHash ? data : Hash.toQueryString(data);
1895
+ }
1896
+ };
1897
+
1898
+ Form.Methods = {
1899
+ serialize: function(form, getHash) {
1900
+ return Form.serializeElements(Form.getElements(form), getHash);
1901
+ },
1902
+
1903
+ getElements: function(form) {
1904
+ return $A($(form).getElementsByTagName('*')).inject([],
1905
+ function(elements, child) {
1906
+ if (Form.Element.Serializers[child.tagName.toLowerCase()])
1907
+ elements.push(Element.extend(child));
1908
+ return elements;
1909
+ }
1910
+ );
1911
+ },
1912
+
1913
+ getInputs: function(form, typeName, name) {
1914
+ form = $(form);
1915
+ var inputs = form.getElementsByTagName('input');
1916
+
1917
+ if (!typeName && !name) return $A(inputs).map(Element.extend);
1918
+
1919
+ for (var i = 0, matchingInputs = [], length = inputs.length; i < length; i++) {
1920
+ var input = inputs[i];
1921
+ if ((typeName && input.type != typeName) || (name && input.name != name))
1922
+ continue;
1923
+ matchingInputs.push(Element.extend(input));
1924
+ }
1925
+
1926
+ return matchingInputs;
1927
+ },
1928
+
1929
+ disable: function(form) {
1930
+ form = $(form);
1931
+ form.getElements().each(function(element) {
1932
+ element.blur();
1933
+ element.disabled = 'true';
1934
+ });
1935
+ return form;
1936
+ },
1937
+
1938
+ enable: function(form) {
1939
+ form = $(form);
1940
+ form.getElements().each(function(element) {
1941
+ element.disabled = '';
1942
+ });
1943
+ return form;
1944
+ },
1945
+
1946
+ findFirstElement: function(form) {
1947
+ return $(form).getElements().find(function(element) {
1948
+ return element.type != 'hidden' && !element.disabled &&
1949
+ ['input', 'select', 'textarea'].include(element.tagName.toLowerCase());
1950
+ });
1951
+ },
1952
+
1953
+ focusFirstElement: function(form) {
1954
+ form = $(form);
1955
+ form.findFirstElement().activate();
1956
+ return form;
1957
+ }
1958
+ }
1959
+
1960
+ Object.extend(Form, Form.Methods);
1961
+
1962
+ /*--------------------------------------------------------------------------*/
1963
+
1964
+ Form.Element = {
1965
+ focus: function(element) {
1966
+ $(element).focus();
1967
+ return element;
1968
+ },
1969
+
1970
+ select: function(element) {
1971
+ $(element).select();
1972
+ return element;
1973
+ }
1974
+ }
1975
+
1976
+ Form.Element.Methods = {
1977
+ serialize: function(element) {
1978
+ element = $(element);
1979
+ if (!element.disabled && element.name) {
1980
+ var value = element.getValue();
1981
+ if (value != undefined) {
1982
+ var pair = {};
1983
+ pair[element.name] = value;
1984
+ return Hash.toQueryString(pair);
1985
+ }
1986
+ }
1987
+ return '';
1988
+ },
1989
+
1990
+ getValue: function(element) {
1991
+ element = $(element);
1992
+ var method = element.tagName.toLowerCase();
1993
+ return Form.Element.Serializers[method](element);
1994
+ },
1995
+
1996
+ clear: function(element) {
1997
+ $(element).value = '';
1998
+ return element;
1999
+ },
2000
+
2001
+ present: function(element) {
2002
+ return $(element).value != '';
2003
+ },
2004
+
2005
+ activate: function(element) {
2006
+ element = $(element);
2007
+ element.focus();
2008
+ if (element.select && ( element.tagName.toLowerCase() != 'input' ||
2009
+ !['button', 'reset', 'submit'].include(element.type) ) )
2010
+ element.select();
2011
+ return element;
2012
+ },
2013
+
2014
+ disable: function(element) {
2015
+ element = $(element);
2016
+ element.disabled = true;
2017
+ return element;
2018
+ },
2019
+
2020
+ enable: function(element) {
2021
+ element = $(element);
2022
+ element.blur();
2023
+ element.disabled = false;
2024
+ return element;
2025
+ }
2026
+ }
2027
+
2028
+ Object.extend(Form.Element, Form.Element.Methods);
2029
+ var Field = Form.Element;
2030
+ var $F = Form.Element.getValue;
2031
+
2032
+ /*--------------------------------------------------------------------------*/
2033
+
2034
+ Form.Element.Serializers = {
2035
+ input: function(element) {
2036
+ switch (element.type.toLowerCase()) {
2037
+ case 'checkbox':
2038
+ case 'radio':
2039
+ return Form.Element.Serializers.inputSelector(element);
2040
+ default:
2041
+ return Form.Element.Serializers.textarea(element);
2042
+ }
2043
+ },
2044
+
2045
+ inputSelector: function(element) {
2046
+ return element.checked ? element.value : null;
2047
+ },
2048
+
2049
+ textarea: function(element) {
2050
+ return element.value;
2051
+ },
2052
+
2053
+ select: function(element) {
2054
+ return this[element.type == 'select-one' ?
2055
+ 'selectOne' : 'selectMany'](element);
2056
+ },
2057
+
2058
+ selectOne: function(element) {
2059
+ var index = element.selectedIndex;
2060
+ return index >= 0 ? this.optionValue(element.options[index]) : null;
2061
+ },
2062
+
2063
+ selectMany: function(element) {
2064
+ var values, length = element.length;
2065
+ if (!length) return null;
2066
+
2067
+ for (var i = 0, values = []; i < length; i++) {
2068
+ var opt = element.options[i];
2069
+ if (opt.selected) values.push(this.optionValue(opt));
2070
+ }
2071
+ return values;
2072
+ },
2073
+
2074
+ optionValue: function(opt) {
2075
+ // extend element because hasAttribute may not be native
2076
+ return Element.extend(opt).hasAttribute('value') ? opt.value : opt.text;
2077
+ }
2078
+ }
2079
+
2080
+ /*--------------------------------------------------------------------------*/
2081
+
2082
+ Abstract.TimedObserver = function() {}
2083
+ Abstract.TimedObserver.prototype = {
2084
+ initialize: function(element, frequency, callback) {
2085
+ this.frequency = frequency;
2086
+ this.element = $(element);
2087
+ this.callback = callback;
2088
+
2089
+ this.lastValue = this.getValue();
2090
+ this.registerCallback();
2091
+ },
2092
+
2093
+ registerCallback: function() {
2094
+ setInterval(this.onTimerEvent.bind(this), this.frequency * 1000);
2095
+ },
2096
+
2097
+ onTimerEvent: function() {
2098
+ var value = this.getValue();
2099
+ var changed = ('string' == typeof this.lastValue && 'string' == typeof value
2100
+ ? this.lastValue != value : String(this.lastValue) != String(value));
2101
+ if (changed) {
2102
+ this.callback(this.element, value);
2103
+ this.lastValue = value;
2104
+ }
2105
+ }
2106
+ }
2107
+
2108
+ Form.Element.Observer = Class.create();
2109
+ Form.Element.Observer.prototype = Object.extend(new Abstract.TimedObserver(), {
2110
+ getValue: function() {
2111
+ return Form.Element.getValue(this.element);
2112
+ }
2113
+ });
2114
+
2115
+ Form.Observer = Class.create();
2116
+ Form.Observer.prototype = Object.extend(new Abstract.TimedObserver(), {
2117
+ getValue: function() {
2118
+ return Form.serialize(this.element);
2119
+ }
2120
+ });
2121
+
2122
+ /*--------------------------------------------------------------------------*/
2123
+
2124
+ Abstract.EventObserver = function() {}
2125
+ Abstract.EventObserver.prototype = {
2126
+ initialize: function(element, callback) {
2127
+ this.element = $(element);
2128
+ this.callback = callback;
2129
+
2130
+ this.lastValue = this.getValue();
2131
+ if (this.element.tagName.toLowerCase() == 'form')
2132
+ this.registerFormCallbacks();
2133
+ else
2134
+ this.registerCallback(this.element);
2135
+ },
2136
+
2137
+ onElementEvent: function() {
2138
+ var value = this.getValue();
2139
+ if (this.lastValue != value) {
2140
+ this.callback(this.element, value);
2141
+ this.lastValue = value;
2142
+ }
2143
+ },
2144
+
2145
+ registerFormCallbacks: function() {
2146
+ Form.getElements(this.element).each(this.registerCallback.bind(this));
2147
+ },
2148
+
2149
+ registerCallback: function(element) {
2150
+ if (element.type) {
2151
+ switch (element.type.toLowerCase()) {
2152
+ case 'checkbox':
2153
+ case 'radio':
2154
+ Event.observe(element, 'click', this.onElementEvent.bind(this));
2155
+ break;
2156
+ default:
2157
+ Event.observe(element, 'change', this.onElementEvent.bind(this));
2158
+ break;
2159
+ }
2160
+ }
2161
+ }
2162
+ }
2163
+
2164
+ Form.Element.EventObserver = Class.create();
2165
+ Form.Element.EventObserver.prototype = Object.extend(new Abstract.EventObserver(), {
2166
+ getValue: function() {
2167
+ return Form.Element.getValue(this.element);
2168
+ }
2169
+ });
2170
+
2171
+ Form.EventObserver = Class.create();
2172
+ Form.EventObserver.prototype = Object.extend(new Abstract.EventObserver(), {
2173
+ getValue: function() {
2174
+ return Form.serialize(this.element);
2175
+ }
2176
+ });
2177
+ if (!window.Event) {
2178
+ var Event = new Object();
2179
+ }
2180
+
2181
+ Object.extend(Event, {
2182
+ KEY_BACKSPACE: 8,
2183
+ KEY_TAB: 9,
2184
+ KEY_RETURN: 13,
2185
+ KEY_ESC: 27,
2186
+ KEY_LEFT: 37,
2187
+ KEY_UP: 38,
2188
+ KEY_RIGHT: 39,
2189
+ KEY_DOWN: 40,
2190
+ KEY_DELETE: 46,
2191
+ KEY_HOME: 36,
2192
+ KEY_END: 35,
2193
+ KEY_PAGEUP: 33,
2194
+ KEY_PAGEDOWN: 34,
2195
+
2196
+ element: function(event) {
2197
+ return event.target || event.srcElement;
2198
+ },
2199
+
2200
+ isLeftClick: function(event) {
2201
+ return (((event.which) && (event.which == 1)) ||
2202
+ ((event.button) && (event.button == 1)));
2203
+ },
2204
+
2205
+ pointerX: function(event) {
2206
+ return event.pageX || (event.clientX +
2207
+ (document.documentElement.scrollLeft || document.body.scrollLeft));
2208
+ },
2209
+
2210
+ pointerY: function(event) {
2211
+ return event.pageY || (event.clientY +
2212
+ (document.documentElement.scrollTop || document.body.scrollTop));
2213
+ },
2214
+
2215
+ stop: function(event) {
2216
+ if (event.preventDefault) {
2217
+ event.preventDefault();
2218
+ event.stopPropagation();
2219
+ } else {
2220
+ event.returnValue = false;
2221
+ event.cancelBubble = true;
2222
+ }
2223
+ },
2224
+
2225
+ // find the first node with the given tagName, starting from the
2226
+ // node the event was triggered on; traverses the DOM upwards
2227
+ findElement: function(event, tagName) {
2228
+ var element = Event.element(event);
2229
+ while (element.parentNode && (!element.tagName ||
2230
+ (element.tagName.toUpperCase() != tagName.toUpperCase())))
2231
+ element = element.parentNode;
2232
+ return element;
2233
+ },
2234
+
2235
+ observers: false,
2236
+
2237
+ _observeAndCache: function(element, name, observer, useCapture) {
2238
+ if (!this.observers) this.observers = [];
2239
+ if (element.addEventListener) {
2240
+ this.observers.push([element, name, observer, useCapture]);
2241
+ element.addEventListener(name, observer, useCapture);
2242
+ } else if (element.attachEvent) {
2243
+ this.observers.push([element, name, observer, useCapture]);
2244
+ element.attachEvent('on' + name, observer);
2245
+ }
2246
+ },
2247
+
2248
+ unloadCache: function() {
2249
+ if (!Event.observers) return;
2250
+ for (var i = 0, length = Event.observers.length; i < length; i++) {
2251
+ Event.stopObserving.apply(this, Event.observers[i]);
2252
+ Event.observers[i][0] = null;
2253
+ }
2254
+ Event.observers = false;
2255
+ },
2256
+
2257
+ observe: function(element, name, observer, useCapture) {
2258
+ element = $(element);
2259
+ useCapture = useCapture || false;
2260
+
2261
+ if (name == 'keypress' &&
2262
+ (navigator.appVersion.match(/Konqueror|Safari|KHTML/)
2263
+ || element.attachEvent))
2264
+ name = 'keydown';
2265
+
2266
+ Event._observeAndCache(element, name, observer, useCapture);
2267
+ },
2268
+
2269
+ stopObserving: function(element, name, observer, useCapture) {
2270
+ element = $(element);
2271
+ useCapture = useCapture || false;
2272
+
2273
+ if (name == 'keypress' &&
2274
+ (navigator.appVersion.match(/Konqueror|Safari|KHTML/)
2275
+ || element.detachEvent))
2276
+ name = 'keydown';
2277
+
2278
+ if (element.removeEventListener) {
2279
+ element.removeEventListener(name, observer, useCapture);
2280
+ } else if (element.detachEvent) {
2281
+ try {
2282
+ element.detachEvent('on' + name, observer);
2283
+ } catch (e) {}
2284
+ }
2285
+ }
2286
+ });
2287
+
2288
+ /* prevent memory leaks in IE */
2289
+ if (navigator.appVersion.match(/\bMSIE\b/))
2290
+ Event.observe(window, 'unload', Event.unloadCache, false);
2291
+ var Position = {
2292
+ // set to true if needed, warning: firefox performance problems
2293
+ // NOT neeeded for page scrolling, only if draggable contained in
2294
+ // scrollable elements
2295
+ includeScrollOffsets: false,
2296
+
2297
+ // must be called before calling withinIncludingScrolloffset, every time the
2298
+ // page is scrolled
2299
+ prepare: function() {
2300
+ this.deltaX = window.pageXOffset
2301
+ || document.documentElement.scrollLeft
2302
+ || document.body.scrollLeft
2303
+ || 0;
2304
+ this.deltaY = window.pageYOffset
2305
+ || document.documentElement.scrollTop
2306
+ || document.body.scrollTop
2307
+ || 0;
2308
+ },
2309
+
2310
+ realOffset: function(element) {
2311
+ var valueT = 0, valueL = 0;
2312
+ do {
2313
+ valueT += element.scrollTop || 0;
2314
+ valueL += element.scrollLeft || 0;
2315
+ element = element.parentNode;
2316
+ } while (element);
2317
+ return [valueL, valueT];
2318
+ },
2319
+
2320
+ cumulativeOffset: function(element) {
2321
+ var valueT = 0, valueL = 0;
2322
+ do {
2323
+ valueT += element.offsetTop || 0;
2324
+ valueL += element.offsetLeft || 0;
2325
+ element = element.offsetParent;
2326
+ } while (element);
2327
+ return [valueL, valueT];
2328
+ },
2329
+
2330
+ positionedOffset: function(element) {
2331
+ var valueT = 0, valueL = 0;
2332
+ do {
2333
+ valueT += element.offsetTop || 0;
2334
+ valueL += element.offsetLeft || 0;
2335
+ element = element.offsetParent;
2336
+ if (element) {
2337
+ if(element.tagName=='BODY') break;
2338
+ var p = Element.getStyle(element, 'position');
2339
+ if (p == 'relative' || p == 'absolute') break;
2340
+ }
2341
+ } while (element);
2342
+ return [valueL, valueT];
2343
+ },
2344
+
2345
+ offsetParent: function(element) {
2346
+ if (element.offsetParent) return element.offsetParent;
2347
+ if (element == document.body) return element;
2348
+
2349
+ while ((element = element.parentNode) && element != document.body)
2350
+ if (Element.getStyle(element, 'position') != 'static')
2351
+ return element;
2352
+
2353
+ return document.body;
2354
+ },
2355
+
2356
+ // caches x/y coordinate pair to use with overlap
2357
+ within: function(element, x, y) {
2358
+ if (this.includeScrollOffsets)
2359
+ return this.withinIncludingScrolloffsets(element, x, y);
2360
+ this.xcomp = x;
2361
+ this.ycomp = y;
2362
+ this.offset = this.cumulativeOffset(element);
2363
+
2364
+ return (y >= this.offset[1] &&
2365
+ y < this.offset[1] + element.offsetHeight &&
2366
+ x >= this.offset[0] &&
2367
+ x < this.offset[0] + element.offsetWidth);
2368
+ },
2369
+
2370
+ withinIncludingScrolloffsets: function(element, x, y) {
2371
+ var offsetcache = this.realOffset(element);
2372
+
2373
+ this.xcomp = x + offsetcache[0] - this.deltaX;
2374
+ this.ycomp = y + offsetcache[1] - this.deltaY;
2375
+ this.offset = this.cumulativeOffset(element);
2376
+
2377
+ return (this.ycomp >= this.offset[1] &&
2378
+ this.ycomp < this.offset[1] + element.offsetHeight &&
2379
+ this.xcomp >= this.offset[0] &&
2380
+ this.xcomp < this.offset[0] + element.offsetWidth);
2381
+ },
2382
+
2383
+ // within must be called directly before
2384
+ overlap: function(mode, element) {
2385
+ if (!mode) return 0;
2386
+ if (mode == 'vertical')
2387
+ return ((this.offset[1] + element.offsetHeight) - this.ycomp) /
2388
+ element.offsetHeight;
2389
+ if (mode == 'horizontal')
2390
+ return ((this.offset[0] + element.offsetWidth) - this.xcomp) /
2391
+ element.offsetWidth;
2392
+ },
2393
+
2394
+ page: function(forElement) {
2395
+ var valueT = 0, valueL = 0;
2396
+
2397
+ var element = forElement;
2398
+ do {
2399
+ valueT += element.offsetTop || 0;
2400
+ valueL += element.offsetLeft || 0;
2401
+
2402
+ // Safari fix
2403
+ if (element.offsetParent==document.body)
2404
+ if (Element.getStyle(element,'position')=='absolute') break;
2405
+
2406
+ } while (element = element.offsetParent);
2407
+
2408
+ element = forElement;
2409
+ do {
2410
+ if (!window.opera || element.tagName=='BODY') {
2411
+ valueT -= element.scrollTop || 0;
2412
+ valueL -= element.scrollLeft || 0;
2413
+ }
2414
+ } while (element = element.parentNode);
2415
+
2416
+ return [valueL, valueT];
2417
+ },
2418
+
2419
+ clone: function(source, target) {
2420
+ var options = Object.extend({
2421
+ setLeft: true,
2422
+ setTop: true,
2423
+ setWidth: true,
2424
+ setHeight: true,
2425
+ offsetTop: 0,
2426
+ offsetLeft: 0
2427
+ }, arguments[2] || {})
2428
+
2429
+ // find page position of source
2430
+ source = $(source);
2431
+ var p = Position.page(source);
2432
+
2433
+ // find coordinate system to use
2434
+ target = $(target);
2435
+ var delta = [0, 0];
2436
+ var parent = null;
2437
+ // delta [0,0] will do fine with position: fixed elements,
2438
+ // position:absolute needs offsetParent deltas
2439
+ if (Element.getStyle(target,'position') == 'absolute') {
2440
+ parent = Position.offsetParent(target);
2441
+ delta = Position.page(parent);
2442
+ }
2443
+
2444
+ // correct by body offsets (fixes Safari)
2445
+ if (parent == document.body) {
2446
+ delta[0] -= document.body.offsetLeft;
2447
+ delta[1] -= document.body.offsetTop;
2448
+ }
2449
+
2450
+ // set position
2451
+ if(options.setLeft) target.style.left = (p[0] - delta[0] + options.offsetLeft) + 'px';
2452
+ if(options.setTop) target.style.top = (p[1] - delta[1] + options.offsetTop) + 'px';
2453
+ if(options.setWidth) target.style.width = source.offsetWidth + 'px';
2454
+ if(options.setHeight) target.style.height = source.offsetHeight + 'px';
2455
+ },
2456
+
2457
+ absolutize: function(element) {
2458
+ element = $(element);
2459
+ if (element.style.position == 'absolute') return;
2460
+ Position.prepare();
2461
+
2462
+ var offsets = Position.positionedOffset(element);
2463
+ var top = offsets[1];
2464
+ var left = offsets[0];
2465
+ var width = element.clientWidth;
2466
+ var height = element.clientHeight;
2467
+
2468
+ element._originalLeft = left - parseFloat(element.style.left || 0);
2469
+ element._originalTop = top - parseFloat(element.style.top || 0);
2470
+ element._originalWidth = element.style.width;
2471
+ element._originalHeight = element.style.height;
2472
+
2473
+ element.style.position = 'absolute';
2474
+ element.style.top = top + 'px';
2475
+ element.style.left = left + 'px';
2476
+ element.style.width = width + 'px';
2477
+ element.style.height = height + 'px';
2478
+ },
2479
+
2480
+ relativize: function(element) {
2481
+ element = $(element);
2482
+ if (element.style.position == 'relative') return;
2483
+ Position.prepare();
2484
+
2485
+ element.style.position = 'relative';
2486
+ var top = parseFloat(element.style.top || 0) - (element._originalTop || 0);
2487
+ var left = parseFloat(element.style.left || 0) - (element._originalLeft || 0);
2488
+
2489
+ element.style.top = top + 'px';
2490
+ element.style.left = left + 'px';
2491
+ element.style.height = element._originalHeight;
2492
+ element.style.width = element._originalWidth;
2493
+ }
2494
+ }
2495
+
2496
+ // Safari returns margins on body which is incorrect if the child is absolutely
2497
+ // positioned. For performance reasons, redefine Position.cumulativeOffset for
2498
+ // KHTML/WebKit only.
2499
+ if (/Konqueror|Safari|KHTML/.test(navigator.userAgent)) {
2500
+ Position.cumulativeOffset = function(element) {
2501
+ var valueT = 0, valueL = 0;
2502
+ do {
2503
+ valueT += element.offsetTop || 0;
2504
+ valueL += element.offsetLeft || 0;
2505
+ if (element.offsetParent == document.body)
2506
+ if (Element.getStyle(element, 'position') == 'absolute') break;
2507
+
2508
+ element = element.offsetParent;
2509
+ } while (element);
2510
+
2511
+ return [valueL, valueT];
2512
+ }
2513
+ }
2514
+
2515
+ Element.addMethods();