thoughtbot-shoulda 2.10.2 → 2.11.1

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