fiveruns_tuneup 0.8.2

Sign up to get free protection for your applications and to get access to all the features.
Files changed (59) hide show
  1. data/History.rdoc +3 -0
  2. data/Manifest +57 -0
  3. data/README.rdoc +44 -0
  4. data/Rakefile +15 -0
  5. data/assets/images/arrows.gif +0 -0
  6. data/assets/images/edit.png +0 -0
  7. data/assets/images/fade.png +0 -0
  8. data/assets/images/fade_down.png +0 -0
  9. data/assets/images/head.gif +0 -0
  10. data/assets/images/logo.gif +0 -0
  11. data/assets/images/logo_clear.png +0 -0
  12. data/assets/images/magnify.png +0 -0
  13. data/assets/images/pip.gif +0 -0
  14. data/assets/images/pointer.gif +0 -0
  15. data/assets/images/schema.png +0 -0
  16. data/assets/images/signin.gif +0 -0
  17. data/assets/images/spinner.gif +0 -0
  18. data/assets/images/warning.gif +0 -0
  19. data/assets/javascripts/prototype.js +2515 -0
  20. data/assets/javascripts/tuneup.js +30 -0
  21. data/assets/stylesheets/tuneup.css +204 -0
  22. data/bin/fiveruns_tuneup +26 -0
  23. data/fiveruns_tuneup.gemspec +49 -0
  24. data/init.rb +2 -0
  25. data/install.rb +18 -0
  26. data/lib/bumpspark_helper.rb +52 -0
  27. data/lib/fiveruns/tuneup.rb +103 -0
  28. data/lib/fiveruns/tuneup/asset_tags.rb +39 -0
  29. data/lib/fiveruns/tuneup/custom_methods.rb +8 -0
  30. data/lib/fiveruns/tuneup/environment.rb +29 -0
  31. data/lib/fiveruns/tuneup/instrumentation/action_controller/base.rb +59 -0
  32. data/lib/fiveruns/tuneup/instrumentation/action_view/base.rb +77 -0
  33. data/lib/fiveruns/tuneup/instrumentation/active_record/base.rb +126 -0
  34. data/lib/fiveruns/tuneup/instrumentation/cgi/session.rb +30 -0
  35. data/lib/fiveruns/tuneup/instrumentation/utilities.rb +172 -0
  36. data/lib/fiveruns/tuneup/multipart.rb +75 -0
  37. data/lib/fiveruns/tuneup/runs.rb +86 -0
  38. data/lib/fiveruns/tuneup/schema.rb +43 -0
  39. data/lib/fiveruns/tuneup/step.rb +219 -0
  40. data/lib/fiveruns/tuneup/urls.rb +23 -0
  41. data/lib/fiveruns/tuneup/version.rb +80 -0
  42. data/lib/fiveruns_tuneup.rb +1 -0
  43. data/lib/tuneup_config.rb +29 -0
  44. data/lib/tuneup_controller.rb +140 -0
  45. data/lib/tuneup_helper.rb +185 -0
  46. data/rails/init.rb +20 -0
  47. data/tasks/assets.rake +32 -0
  48. data/test/test_helper.rb +3 -0
  49. data/test/tuneup_test.rb +0 -0
  50. data/uninstall.rb +6 -0
  51. data/views/tuneup/_data.html.erb +15 -0
  52. data/views/tuneup/_flash.html.erb +6 -0
  53. data/views/tuneup/_link.html.erb +1 -0
  54. data/views/tuneup/_schema.html.erb +17 -0
  55. data/views/tuneup/_sql.html.erb +23 -0
  56. data/views/tuneup/_step.html.erb +15 -0
  57. data/views/tuneup/panel/_registered.html.erb +4 -0
  58. data/views/tuneup/panel/_unregistered.html.erb +14 -0
  59. metadata +146 -0
data/History.rdoc ADDED
@@ -0,0 +1,3 @@
1
+ = History
2
+
3
+ * v0.8.1. First gem package with basic functionality
data/Manifest ADDED
@@ -0,0 +1,57 @@
1
+ assets/images/arrows.gif
2
+ assets/images/edit.png
3
+ assets/images/fade.png
4
+ assets/images/fade_down.png
5
+ assets/images/head.gif
6
+ assets/images/logo.gif
7
+ assets/images/logo_clear.png
8
+ assets/images/magnify.png
9
+ assets/images/pip.gif
10
+ assets/images/pointer.gif
11
+ assets/images/schema.png
12
+ assets/images/signin.gif
13
+ assets/images/spinner.gif
14
+ assets/images/warning.gif
15
+ assets/javascripts/prototype.js
16
+ assets/javascripts/tuneup.js
17
+ assets/stylesheets/tuneup.css
18
+ bin/fiveruns_tuneup
19
+ History.rdoc
20
+ init.rb
21
+ install.rb
22
+ lib/bumpspark_helper.rb
23
+ lib/fiveruns/tuneup/asset_tags.rb
24
+ lib/fiveruns/tuneup/custom_methods.rb
25
+ lib/fiveruns/tuneup/environment.rb
26
+ lib/fiveruns/tuneup/instrumentation/action_controller/base.rb
27
+ lib/fiveruns/tuneup/instrumentation/action_view/base.rb
28
+ lib/fiveruns/tuneup/instrumentation/active_record/base.rb
29
+ lib/fiveruns/tuneup/instrumentation/cgi/session.rb
30
+ lib/fiveruns/tuneup/instrumentation/utilities.rb
31
+ lib/fiveruns/tuneup/multipart.rb
32
+ lib/fiveruns/tuneup/runs.rb
33
+ lib/fiveruns/tuneup/schema.rb
34
+ lib/fiveruns/tuneup/step.rb
35
+ lib/fiveruns/tuneup/urls.rb
36
+ lib/fiveruns/tuneup/version.rb
37
+ lib/fiveruns/tuneup.rb
38
+ lib/fiveruns_tuneup.rb
39
+ lib/tuneup_config.rb
40
+ lib/tuneup_controller.rb
41
+ lib/tuneup_helper.rb
42
+ Manifest
43
+ rails/init.rb
44
+ Rakefile
45
+ README.rdoc
46
+ tasks/assets.rake
47
+ test/test_helper.rb
48
+ test/tuneup_test.rb
49
+ uninstall.rb
50
+ views/tuneup/_data.html.erb
51
+ views/tuneup/_flash.html.erb
52
+ views/tuneup/_link.html.erb
53
+ views/tuneup/_schema.html.erb
54
+ views/tuneup/_sql.html.erb
55
+ views/tuneup/_step.html.erb
56
+ views/tuneup/panel/_registered.html.erb
57
+ views/tuneup/panel/_unregistered.html.erb
data/README.rdoc ADDED
@@ -0,0 +1,44 @@
1
+ = FiveRuns TuneUp Plugin
2
+
3
+ == Requirements
4
+
5
+ * Rails 2.0.0+
6
+ * Firefox 3rc1+ (2 soon), Safari 3+. IE support on the roadmap.
7
+
8
+ == Synopsis
9
+
10
+ Display call stack information on Rails requests.
11
+
12
+ THIS IS A BETA RELEASE; the normal disclaimers apply.
13
+
14
+ == Known Issues
15
+
16
+ The following issues have been seen from time to time on some apps:
17
+ * Issues with fixed/absolute positioned elements overlapping the panel
18
+ and/or being pushed down too far.
19
+ * Panel not showing up on the `root' route.
20
+
21
+ == License
22
+
23
+ (The FiveRuns License)
24
+
25
+ Copyright (c) 2008 FiveRuns Corporation
26
+
27
+ Permission is hereby granted, free of charge, to any person obtaining
28
+ a copy of this software and associated documentation files (the
29
+ "Software"), to deal in the Software without restriction, including
30
+ without limitation the rights to use, copy, modify, merge, publish,
31
+ distribute, sublicense, and/or sell copies of the Software, and to
32
+ permit persons to whom the Software is furnished to do so, subject to
33
+ the following conditions:
34
+
35
+ The above copyright notice and this permission notice shall be
36
+ included in all copies or substantial portions of the Software.
37
+
38
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
39
+ EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
40
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
41
+ NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
42
+ LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
43
+ OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
44
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/Rakefile ADDED
@@ -0,0 +1,15 @@
1
+ require 'rubygems'
2
+ require 'echoe'
3
+
4
+ require File.dirname(__FILE__) << "/lib/fiveruns/tuneup/version"
5
+
6
+ Echoe.new 'fiveruns_tuneup' do |p|
7
+ p.version = Fiveruns::Tuneup::Version::STRING
8
+ p.author = "FiveRuns Development Team"
9
+ p.email = 'dev@fiveruns.com'
10
+ p.project = 'fiveruns'
11
+ p.summary = "Instrumentation for the FiveRuns TuneUp product."
12
+ p.url = "http://github.com/fiveruns/fiveruns_tuneup"
13
+ p.dependencies = %w(activesupport)
14
+ p.include_rakefile = true
15
+ end
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
Binary file
@@ -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 == 'FORM')
1116
+ Object.extend(methods, Form.Methods);
1117
+ if (['INPUT', 'TEXTAREA', 'SELECT'].include(element.tagName))
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();