active_scaffold 3.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (272) hide show
  1. data/.autotest +27 -0
  2. data/CHANGELOG +152 -0
  3. data/Gemfile +4 -0
  4. data/MIT-LICENSE +20 -0
  5. data/README +51 -0
  6. data/Rakefile +24 -0
  7. data/active_scaffold.gemspec +24 -0
  8. data/environment.rb +22 -0
  9. data/frontends/default/images/add.gif +0 -0
  10. data/frontends/default/images/arrow_down.gif +0 -0
  11. data/frontends/default/images/arrow_up.gif +0 -0
  12. data/frontends/default/images/close.gif +0 -0
  13. data/frontends/default/images/cross.png +0 -0
  14. data/frontends/default/images/indicator-small.gif +0 -0
  15. data/frontends/default/images/indicator.gif +0 -0
  16. data/frontends/default/images/magnifier.png +0 -0
  17. data/frontends/default/javascripts/jquery/active_scaffold.js +957 -0
  18. data/frontends/default/javascripts/jquery/jquery.editinplace.js +726 -0
  19. data/frontends/default/javascripts/prototype/active_scaffold.js +954 -0
  20. data/frontends/default/javascripts/prototype/dhtml_history.js +867 -0
  21. data/frontends/default/javascripts/prototype/form_enhancements.js +117 -0
  22. data/frontends/default/javascripts/prototype/rico_corner.js +370 -0
  23. data/frontends/default/stylesheets/stylesheet-ie.css +35 -0
  24. data/frontends/default/stylesheets/stylesheet.css +858 -0
  25. data/frontends/default/views/_add_existing_form.html.erb +30 -0
  26. data/frontends/default/views/_base_form.html.erb +41 -0
  27. data/frontends/default/views/_create_form.html.erb +6 -0
  28. data/frontends/default/views/_create_form_on_list.html.erb +5 -0
  29. data/frontends/default/views/_field_search.html.erb +32 -0
  30. data/frontends/default/views/_form.html.erb +24 -0
  31. data/frontends/default/views/_form_association.html.erb +14 -0
  32. data/frontends/default/views/_form_association_footer.html.erb +40 -0
  33. data/frontends/default/views/_form_attribute.html.erb +15 -0
  34. data/frontends/default/views/_form_hidden_attribute.html.erb +2 -0
  35. data/frontends/default/views/_form_messages.html.erb +5 -0
  36. data/frontends/default/views/_horizontal_subform.html.erb +19 -0
  37. data/frontends/default/views/_horizontal_subform_header.html.erb +10 -0
  38. data/frontends/default/views/_horizontal_subform_record.html.erb +37 -0
  39. data/frontends/default/views/_human_conditions.html.erb +1 -0
  40. data/frontends/default/views/_list.html.erb +18 -0
  41. data/frontends/default/views/_list_actions.html.erb +16 -0
  42. data/frontends/default/views/_list_calculations.html.erb +16 -0
  43. data/frontends/default/views/_list_column_headings.html.erb +12 -0
  44. data/frontends/default/views/_list_header.html.erb +12 -0
  45. data/frontends/default/views/_list_inline_adapter.html.erb +10 -0
  46. data/frontends/default/views/_list_messages.html.erb +32 -0
  47. data/frontends/default/views/_list_pagination.html.erb +11 -0
  48. data/frontends/default/views/_list_pagination_links.html.erb +9 -0
  49. data/frontends/default/views/_list_record.html.erb +14 -0
  50. data/frontends/default/views/_list_record_columns.html.erb +8 -0
  51. data/frontends/default/views/_list_with_header.html.erb +32 -0
  52. data/frontends/default/views/_messages.html.erb +10 -0
  53. data/frontends/default/views/_render_field.js.rjs +13 -0
  54. data/frontends/default/views/_row.html.erb +12 -0
  55. data/frontends/default/views/_search.html.erb +34 -0
  56. data/frontends/default/views/_search_attribute.html.erb +10 -0
  57. data/frontends/default/views/_show.html.erb +8 -0
  58. data/frontends/default/views/_show_columns.html.erb +12 -0
  59. data/frontends/default/views/_update_actions.html.erb +9 -0
  60. data/frontends/default/views/_update_form.html.erb +5 -0
  61. data/frontends/default/views/_vertical_subform.html.erb +12 -0
  62. data/frontends/default/views/_vertical_subform_record.html.erb +38 -0
  63. data/frontends/default/views/add_existing.js.rjs +17 -0
  64. data/frontends/default/views/add_existing_form.html.erb +5 -0
  65. data/frontends/default/views/create.html.erb +5 -0
  66. data/frontends/default/views/delete.html.erb +13 -0
  67. data/frontends/default/views/destroy.js.rjs +5 -0
  68. data/frontends/default/views/edit_associated.js.rjs +11 -0
  69. data/frontends/default/views/field_search.html.erb +5 -0
  70. data/frontends/default/views/form_messages.js.rjs +1 -0
  71. data/frontends/default/views/list.html.erb +1 -0
  72. data/frontends/default/views/list.js.rjs +1 -0
  73. data/frontends/default/views/on_action_update.js.rjs +8 -0
  74. data/frontends/default/views/on_create.js.rjs +24 -0
  75. data/frontends/default/views/on_update.js.rjs +15 -0
  76. data/frontends/default/views/search.html.erb +5 -0
  77. data/frontends/default/views/show.html.erb +5 -0
  78. data/frontends/default/views/update.html.erb +8 -0
  79. data/frontends/default/views/update_column.js.rjs +13 -0
  80. data/frontends/default/views/update_row.js.rjs +1 -0
  81. data/init.rb +1 -0
  82. data/install_assets.rb +44 -0
  83. data/lib/active_record_permissions.rb +134 -0
  84. data/lib/active_scaffold.rb +279 -0
  85. data/lib/active_scaffold/actions/common_search.rb +22 -0
  86. data/lib/active_scaffold/actions/core.rb +150 -0
  87. data/lib/active_scaffold/actions/create.rb +152 -0
  88. data/lib/active_scaffold/actions/delete.rb +72 -0
  89. data/lib/active_scaffold/actions/field_search.rb +82 -0
  90. data/lib/active_scaffold/actions/list.rb +128 -0
  91. data/lib/active_scaffold/actions/mark.rb +50 -0
  92. data/lib/active_scaffold/actions/nested.rb +241 -0
  93. data/lib/active_scaffold/actions/search.rb +47 -0
  94. data/lib/active_scaffold/actions/show.rb +54 -0
  95. data/lib/active_scaffold/actions/subform.rb +17 -0
  96. data/lib/active_scaffold/actions/update.rb +134 -0
  97. data/lib/active_scaffold/attribute_params.rb +207 -0
  98. data/lib/active_scaffold/bridges/ancestry/bridge.rb +5 -0
  99. data/lib/active_scaffold/bridges/ancestry/lib/ancestry_bridge.rb +38 -0
  100. data/lib/active_scaffold/bridges/bridge.rb +52 -0
  101. data/lib/active_scaffold/bridges/calendar_date_select/bridge.rb +16 -0
  102. data/lib/active_scaffold/bridges/calendar_date_select/lib/as_cds_bridge.rb +79 -0
  103. data/lib/active_scaffold/bridges/carrierwave/bridge.rb +7 -0
  104. data/lib/active_scaffold/bridges/carrierwave/lib/carrierwave_bridge.rb +38 -0
  105. data/lib/active_scaffold/bridges/carrierwave/lib/carrierwave_bridge_helpers.rb +26 -0
  106. data/lib/active_scaffold/bridges/carrierwave/lib/form_ui.rb +35 -0
  107. data/lib/active_scaffold/bridges/carrierwave/lib/list_ui.rb +17 -0
  108. data/lib/active_scaffold/bridges/date_picker/bridge.rb +22 -0
  109. data/lib/active_scaffold/bridges/date_picker/lib/datepicker_bridge.rb +225 -0
  110. data/lib/active_scaffold/bridges/date_picker/public/javascripts/date_picker_bridge.js +22 -0
  111. data/lib/active_scaffold/bridges/file_column/bridge.rb +11 -0
  112. data/lib/active_scaffold/bridges/file_column/lib/as_file_column_bridge.rb +46 -0
  113. data/lib/active_scaffold/bridges/file_column/lib/file_column_helpers.rb +59 -0
  114. data/lib/active_scaffold/bridges/file_column/lib/form_ui.rb +37 -0
  115. data/lib/active_scaffold/bridges/file_column/lib/list_ui.rb +26 -0
  116. data/lib/active_scaffold/bridges/file_column/test/functional/file_column_keep_test.rb +43 -0
  117. data/lib/active_scaffold/bridges/file_column/test/mock_model.rb +9 -0
  118. data/lib/active_scaffold/bridges/file_column/test/test_helper.rb +15 -0
  119. data/lib/active_scaffold/bridges/paperclip/bridge.rb +12 -0
  120. data/lib/active_scaffold/bridges/paperclip/lib/form_ui.rb +27 -0
  121. data/lib/active_scaffold/bridges/paperclip/lib/list_ui.rb +16 -0
  122. data/lib/active_scaffold/bridges/paperclip/lib/paperclip_bridge.rb +38 -0
  123. data/lib/active_scaffold/bridges/paperclip/lib/paperclip_bridge_helpers.rb +26 -0
  124. data/lib/active_scaffold/bridges/semantic_attributes/bridge.rb +5 -0
  125. data/lib/active_scaffold/bridges/semantic_attributes/lib/semantic_attributes_bridge.rb +20 -0
  126. data/lib/active_scaffold/bridges/shared/date_bridge.rb +187 -0
  127. data/lib/active_scaffold/bridges/tiny_mce/bridge.rb +5 -0
  128. data/lib/active_scaffold/bridges/tiny_mce/lib/tiny_mce_bridge.rb +45 -0
  129. data/lib/active_scaffold/bridges/validation_reflection/bridge.rb +8 -0
  130. data/lib/active_scaffold/bridges/validation_reflection/lib/validation_reflection_bridge.rb +21 -0
  131. data/lib/active_scaffold/config/base.rb +54 -0
  132. data/lib/active_scaffold/config/core.rb +229 -0
  133. data/lib/active_scaffold/config/create.rb +43 -0
  134. data/lib/active_scaffold/config/delete.rb +25 -0
  135. data/lib/active_scaffold/config/field_search.rb +74 -0
  136. data/lib/active_scaffold/config/form.rb +46 -0
  137. data/lib/active_scaffold/config/list.rb +174 -0
  138. data/lib/active_scaffold/config/mark.rb +22 -0
  139. data/lib/active_scaffold/config/nested.rb +43 -0
  140. data/lib/active_scaffold/config/search.rb +68 -0
  141. data/lib/active_scaffold/config/show.rb +34 -0
  142. data/lib/active_scaffold/config/subform.rb +35 -0
  143. data/lib/active_scaffold/config/update.rb +38 -0
  144. data/lib/active_scaffold/configurable.rb +29 -0
  145. data/lib/active_scaffold/constraints.rb +179 -0
  146. data/lib/active_scaffold/data_structures/action_columns.rb +133 -0
  147. data/lib/active_scaffold/data_structures/action_link.rb +162 -0
  148. data/lib/active_scaffold/data_structures/action_links.rb +59 -0
  149. data/lib/active_scaffold/data_structures/actions.rb +45 -0
  150. data/lib/active_scaffold/data_structures/column.rb +348 -0
  151. data/lib/active_scaffold/data_structures/columns.rb +75 -0
  152. data/lib/active_scaffold/data_structures/error_message.rb +24 -0
  153. data/lib/active_scaffold/data_structures/nested_info.rb +108 -0
  154. data/lib/active_scaffold/data_structures/set.rb +62 -0
  155. data/lib/active_scaffold/data_structures/sorting.rb +168 -0
  156. data/lib/active_scaffold/finder.rb +333 -0
  157. data/lib/active_scaffold/helpers/association_helpers.rb +40 -0
  158. data/lib/active_scaffold/helpers/controller_helpers.rb +40 -0
  159. data/lib/active_scaffold/helpers/country_helpers.rb +352 -0
  160. data/lib/active_scaffold/helpers/form_column_helpers.rb +343 -0
  161. data/lib/active_scaffold/helpers/human_condition_helpers.rb +59 -0
  162. data/lib/active_scaffold/helpers/id_helpers.rb +131 -0
  163. data/lib/active_scaffold/helpers/list_column_helpers.rb +363 -0
  164. data/lib/active_scaffold/helpers/pagination_helpers.rb +55 -0
  165. data/lib/active_scaffold/helpers/search_column_helpers.rb +238 -0
  166. data/lib/active_scaffold/helpers/show_column_helpers.rb +46 -0
  167. data/lib/active_scaffold/helpers/view_helpers.rb +315 -0
  168. data/lib/active_scaffold/locale/de.rb +113 -0
  169. data/lib/active_scaffold/locale/en.rb +118 -0
  170. data/lib/active_scaffold/locale/es.yml +112 -0
  171. data/lib/active_scaffold/locale/fr.rb +113 -0
  172. data/lib/active_scaffold/locale/hu.yml +63 -0
  173. data/lib/active_scaffold/locale/ja.yml +64 -0
  174. data/lib/active_scaffold/locale/ru.yml +62 -0
  175. data/lib/active_scaffold/marked_model.rb +38 -0
  176. data/lib/dhtml_confirm.rb +54 -0
  177. data/lib/extensions/action_controller_rendering.rb +20 -0
  178. data/lib/extensions/action_view_rendering.rb +113 -0
  179. data/lib/extensions/action_view_resolver.rb +7 -0
  180. data/lib/extensions/active_record_offset.rb +12 -0
  181. data/lib/extensions/array.rb +7 -0
  182. data/lib/extensions/localize.rb +10 -0
  183. data/lib/extensions/name_option_for_datetime.rb +12 -0
  184. data/lib/extensions/nil_id_in_url_params.rb +7 -0
  185. data/lib/extensions/paginator_extensions.rb +26 -0
  186. data/lib/extensions/reverse_associations.rb +62 -0
  187. data/lib/extensions/routing_mapper.rb +34 -0
  188. data/lib/extensions/to_label.rb +8 -0
  189. data/lib/extensions/unsaved_associated.rb +61 -0
  190. data/lib/extensions/unsaved_record.rb +20 -0
  191. data/lib/extensions/usa_state.rb +46 -0
  192. data/lib/generators/active_scaffold/USAGE +29 -0
  193. data/lib/generators/active_scaffold/active_scaffold_generator.rb +20 -0
  194. data/lib/generators/active_scaffold_controller/USAGE +19 -0
  195. data/lib/generators/active_scaffold_controller/active_scaffold_controller_generator.rb +28 -0
  196. data/lib/generators/active_scaffold_controller/templates/controller.rb +4 -0
  197. data/lib/generators/active_scaffold_setup/USAGE +10 -0
  198. data/lib/generators/active_scaffold_setup/active_scaffold_setup_generator.rb +53 -0
  199. data/lib/paginator.rb +136 -0
  200. data/lib/responds_to_parent.rb +70 -0
  201. data/public/blank.html +33 -0
  202. data/shoulda_macros/macros.rb +136 -0
  203. data/test/bridges/bridge_test.rb +47 -0
  204. data/test/config/base_test.rb +15 -0
  205. data/test/config/create_test.rb +55 -0
  206. data/test/config/list_test.rb +74 -0
  207. data/test/config/show_test.rb +43 -0
  208. data/test/config/update_test.rb +17 -0
  209. data/test/const_mocker.rb +36 -0
  210. data/test/data_structures/action_columns_test.rb +113 -0
  211. data/test/data_structures/action_link_test.rb +78 -0
  212. data/test/data_structures/action_links_test.rb +78 -0
  213. data/test/data_structures/actions_test.rb +25 -0
  214. data/test/data_structures/association_column_test.rb +42 -0
  215. data/test/data_structures/column_test.rb +185 -0
  216. data/test/data_structures/columns_test.rb +69 -0
  217. data/test/data_structures/error_message_test.rb +28 -0
  218. data/test/data_structures/set_test.rb +86 -0
  219. data/test/data_structures/sorting_test.rb +126 -0
  220. data/test/data_structures/standard_column_test.rb +24 -0
  221. data/test/data_structures/virtual_column_test.rb +23 -0
  222. data/test/extensions/active_record_test.rb +45 -0
  223. data/test/extensions/array_test.rb +12 -0
  224. data/test/helpers/form_column_helpers_test.rb +31 -0
  225. data/test/helpers/list_column_helpers_test.rb +31 -0
  226. data/test/helpers/pagination_helpers_test.rb +55 -0
  227. data/test/misc/active_record_permissions_test.rb +154 -0
  228. data/test/misc/attribute_params_test.rb +110 -0
  229. data/test/misc/configurable_test.rb +96 -0
  230. data/test/misc/constraints_test.rb +193 -0
  231. data/test/misc/finder_test.rb +93 -0
  232. data/test/misc/lang_test.rb +12 -0
  233. data/test/mock_app/.gitignore +2 -0
  234. data/test/mock_app/app/controllers/application_controller.rb +10 -0
  235. data/test/mock_app/app/helpers/application_helper.rb +3 -0
  236. data/test/mock_app/config/boot.rb +110 -0
  237. data/test/mock_app/config/database.yml +16 -0
  238. data/test/mock_app/config/environment.rb +43 -0
  239. data/test/mock_app/config/environments/development.rb +17 -0
  240. data/test/mock_app/config/environments/production.rb +28 -0
  241. data/test/mock_app/config/environments/test.rb +28 -0
  242. data/test/mock_app/config/initializers/backtrace_silencers.rb +7 -0
  243. data/test/mock_app/config/initializers/inflections.rb +10 -0
  244. data/test/mock_app/config/initializers/mime_types.rb +5 -0
  245. data/test/mock_app/config/initializers/new_rails_defaults.rb +19 -0
  246. data/test/mock_app/config/initializers/session_store.rb +15 -0
  247. data/test/mock_app/config/locales/en.yml +5 -0
  248. data/test/mock_app/config/routes.rb +43 -0
  249. data/test/mock_app/db/test.sqlite3 +1 -0
  250. data/test/mock_app/public/blank.html +33 -0
  251. data/test/mock_app/public/images/active_scaffold/DO_NOT_EDIT +2 -0
  252. data/test/mock_app/public/images/active_scaffold/default/add.gif +0 -0
  253. data/test/mock_app/public/images/active_scaffold/default/arrow_down.gif +0 -0
  254. data/test/mock_app/public/images/active_scaffold/default/arrow_up.gif +0 -0
  255. data/test/mock_app/public/images/active_scaffold/default/close.gif +0 -0
  256. data/test/mock_app/public/images/active_scaffold/default/cross.png +0 -0
  257. data/test/mock_app/public/images/active_scaffold/default/indicator-small.gif +0 -0
  258. data/test/mock_app/public/images/active_scaffold/default/indicator.gif +0 -0
  259. data/test/mock_app/public/images/active_scaffold/default/magnifier.png +0 -0
  260. data/test/mock_app/public/javascripts/active_scaffold/DO_NOT_EDIT +2 -0
  261. data/test/mock_app/public/javascripts/active_scaffold/default/active_scaffold.js +532 -0
  262. data/test/mock_app/public/javascripts/active_scaffold/default/dhtml_history.js +867 -0
  263. data/test/mock_app/public/javascripts/active_scaffold/default/form_enhancements.js +117 -0
  264. data/test/mock_app/public/javascripts/active_scaffold/default/rico_corner.js +370 -0
  265. data/test/mock_app/public/stylesheets/active_scaffold/DO_NOT_EDIT +2 -0
  266. data/test/mock_app/public/stylesheets/active_scaffold/default/stylesheet-ie.css +35 -0
  267. data/test/mock_app/public/stylesheets/active_scaffold/default/stylesheet.css +839 -0
  268. data/test/model_stub.rb +55 -0
  269. data/test/run_all.rb +8 -0
  270. data/test/test_helper.rb +39 -0
  271. data/uninstall.rb +13 -0
  272. metadata +478 -0
@@ -0,0 +1,867 @@
1
+ /*
2
+ Copyright (c) 2007 Brian Dillard and Brad Neuberg:
3
+ Brian Dillard | Project Lead | bdillard@pathf.com | http://blogs.pathf.com/agileajax/
4
+ Brad Neuberg | Original Project Creator | http://codinginparadise.org
5
+
6
+ SVN r113 from http://code.google.com/p/reallysimplehistory
7
+ + Changes by Ed Wildgoose - MailASail
8
+ + Changed EncodeURIComponent -> EncodeURI
9
+ + Changed DecodeURIComponent -> DecodeURI
10
+ + Changed 'blank.html?' -> '/blank.html?'
11
+
12
+ Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files
13
+ (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge,
14
+ publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do
15
+ so, subject to the following conditions:
16
+
17
+ The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
18
+
19
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
20
+ MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE
21
+ FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
22
+ WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
23
+ */
24
+
25
+ /*
26
+ dhtmlHistory: An object that provides history, history data, and bookmarking for DHTML and Ajax applications.
27
+
28
+ dependencies:
29
+ * the historyStorage object included in this file.
30
+
31
+ */
32
+ window.dhtmlHistory = {
33
+
34
+ /*Public: User-agent booleans*/
35
+ isIE: false,
36
+ isOpera: false,
37
+ isSafari: false,
38
+ isKonquerer: false,
39
+ isGecko: false,
40
+ isSupported: false,
41
+
42
+ /*Public: Create the DHTML history infrastructure*/
43
+ create: function(options) {
44
+
45
+ /*
46
+ options - object to store initialization parameters
47
+ options.blankURL - string to override the default location of blank.html. Must end in "?"
48
+ options.debugMode - boolean that causes hidden form fields to be shown for development purposes.
49
+ options.toJSON - function to override default JSON stringifier
50
+ options.fromJSON - function to override default JSON parser
51
+ options.baseTitle - pattern for title changes; example: "Armchair DJ [@@@]" - @@@ will be replaced
52
+ */
53
+
54
+ var that = this;
55
+
56
+ /*Set up the historyStorage object; pass in options bundle*/
57
+ window.historyStorage.setup(options);
58
+
59
+ /*Set up our base title if one is passed in*/
60
+ if (options && options.baseTitle) {
61
+ if (options.baseTitle.indexOf("@@@") < 0 && historyStorage.debugMode) {
62
+ throw new Error("Programmer error: options.baseTitle must contain the replacement parameter"
63
+ + " '@@@' to be useful.");
64
+ }
65
+ this.baseTitle = options.baseTitle;
66
+ }
67
+
68
+ /*set user-agent flags*/
69
+ var UA = navigator.userAgent.toLowerCase();
70
+ var platform = navigator.platform.toLowerCase();
71
+ var vendor = navigator.vendor || "";
72
+ if (vendor === "KDE") {
73
+ this.isKonqueror = true;
74
+ this.isSupported = false;
75
+ } else if (typeof window.opera !== "undefined") {
76
+ this.isOpera = true;
77
+ this.isSupported = true;
78
+ } else if (typeof document.all !== "undefined") {
79
+ this.isIE = true;
80
+ this.isSupported = true;
81
+ } else if (vendor.indexOf("Apple Computer, Inc.") > -1) {
82
+ this.isSafari = true;
83
+ this.isSupported = (platform.indexOf("mac") > -1);
84
+ } else if (UA.indexOf("gecko") != -1) {
85
+ this.isGecko = true;
86
+ this.isSupported = true;
87
+ }
88
+
89
+ /*Create Safari/Opera-specific code*/
90
+ if (this.isSafari) {
91
+ this.createSafari();
92
+ } else if (this.isOpera) {
93
+ this.createOpera();
94
+ }
95
+
96
+ /*Get our initial location*/
97
+ var initialHash = this.getCurrentLocation();
98
+
99
+ /*Save it as our current location*/
100
+ this.currentLocation = initialHash;
101
+
102
+ /*Now that we have a hash, create IE-specific code*/
103
+ if (this.isIE) {
104
+ /*Optionally override the URL of IE's blank HTML file*/
105
+ if (options && options.blankURL) {
106
+ var u = options.blankURL;
107
+ /*assign the value, adding the trailing ? if it's not passed in*/
108
+ this.blankURL = (u.indexOf("?") != u.length - 1
109
+ ? u + "?"
110
+ : u
111
+ );
112
+ }
113
+ this.createIE(initialHash);
114
+ }
115
+
116
+ /*Add an unload listener for the page; this is needed for FF 1.5+ because this browser caches all dynamic updates to the
117
+ page, which can break some of our logic related to testing whether this is the first instance a page has loaded or whether
118
+ it is being pulled from the cache*/
119
+
120
+ var unloadHandler = function() {
121
+ that.firstLoad = null;
122
+ };
123
+
124
+ this.addEventListener(window,'unload',unloadHandler);
125
+
126
+ /*Determine if this is our first page load; for IE, we do this in this.iframeLoaded(), which is fired on pageload. We do it
127
+ there because we have no historyStorage at this point, which only exists after the page is finished loading in IE*/
128
+ if (this.isIE) {
129
+ /*The iframe will get loaded on page load, and we want to ignore this fact*/
130
+ this.ignoreLocationChange = true;
131
+ } else {
132
+ if (!historyStorage.hasKey(this.PAGELOADEDSTRING)) {
133
+ /*This is our first page load, so ignore the location change and add our special history entry*/
134
+ this.ignoreLocationChange = true;
135
+ this.firstLoad = true;
136
+ historyStorage.put(this.PAGELOADEDSTRING, true);
137
+ } else {
138
+ /*This isn't our first page load, so indicate that we want to pay attention to this location change*/
139
+ this.ignoreLocationChange = false;
140
+ this.firstLoad = false;
141
+ /*For browsers other than IE, fire a history change event; on IE, the event will be thrown automatically when its
142
+ hidden iframe reloads on page load. Unfortunately, we don't have any listeners yet; indicate that we want to fire
143
+ an event when a listener is added.*/
144
+ this.fireOnNewListener = true;
145
+ }
146
+ }
147
+
148
+ /*Other browsers can use a location handler that checks at regular intervals as their primary mechanism; we use it for IE as
149
+ well to handle an important edge case; see checkLocation() for details*/
150
+ var locationHandler = function() {
151
+ that.checkLocation();
152
+ };
153
+ setInterval(locationHandler, 100);
154
+ },
155
+
156
+ /*Public: Initialize our DHTML history. You must call this after the page is finished loading. Optionally, you can pass your listener in
157
+ here so you don't need to make a separate call to addListener*/
158
+ initialize: function(listener) {
159
+
160
+ /*save original document title to plug in when we hit a null-key history point*/
161
+ this.originalTitle = document.title;
162
+
163
+ /*IE needs to be explicitly initialized. IE doesn't autofill form data until the page is finished loading, so we have to wait*/
164
+ if (this.isIE) {
165
+ /*If this is the first time this page has loaded*/
166
+ if (!historyStorage.hasKey(this.PAGELOADEDSTRING)) {
167
+ /*For IE, we do this in initialize(); for other browsers, we do it in create()*/
168
+ this.fireOnNewListener = false;
169
+ this.firstLoad = true;
170
+ historyStorage.put(this.PAGELOADEDSTRING, true);
171
+ }
172
+ /*Else if this is a fake onload event*/
173
+ else {
174
+ this.fireOnNewListener = true;
175
+ this.firstLoad = false;
176
+ }
177
+ }
178
+ /*optional convenience to save a separate call to addListener*/
179
+ if (listener) {
180
+ this.addListener(listener);
181
+ }
182
+ },
183
+
184
+ /*Public: Adds a history change listener. Only one listener is supported at this time.*/
185
+ addListener: function(listener) {
186
+ this.listener = listener;
187
+ /*If the page was just loaded and we should not ignore it, fire an event to our new listener now*/
188
+ if (this.fireOnNewListener) {
189
+ this.fireHistoryEvent(this.currentLocation);
190
+ this.fireOnNewListener = false;
191
+ }
192
+ },
193
+
194
+ /*Public: Change the current HTML title*/
195
+ changeTitle: function(historyData) {
196
+ var winTitle = (historyData && historyData.newTitle
197
+ /*Plug the new title into the pattern*/
198
+ ? this.baseTitle.replace('@@@', historyData.newTitle)
199
+ /*Otherwise, if there is no new title, use the original document title. This is useful when some
200
+ history changes have title changes and some don't; we can automatically return to the original
201
+ title rather than leaving a misleading title in the title bar. The same goes for our "virgin"
202
+ (hashless) page state.*/
203
+ : this.originalTitle
204
+ );
205
+ /*No need to do anything if the title isn't changing*/
206
+ if (document.title == winTitle) {
207
+ return;
208
+ }
209
+
210
+
211
+ /*Now change the DOM*/
212
+ document.title = winTitle;
213
+ /*Change it in the iframe, too, for IE*/
214
+ if (this.isIE) {
215
+ this.iframe.contentWindow.document.title = winTitle;
216
+ }
217
+
218
+ /*If non-IE, reload the hash so the new title "sticks" in the browser history object*/
219
+ if (!this.isIE && !this.isOpera) {
220
+ var hash = decodeURI(document.location.hash);
221
+ if (hash != "") {
222
+ var encodedHash = encodeURI(this.removeHash(hash));
223
+ document.location.hash = encodedHash;
224
+ } else {
225
+ //document.location.hash = "#";
226
+ }
227
+ }
228
+ },
229
+
230
+ /*Public: Add a history point. Parameters available:
231
+ * newLocation (required):
232
+ This will be the #hash value in the URL. Users can bookmark it. It will persist across sessions, so
233
+ your application should be able to restore itself to a specific state based on just this value. It
234
+ should be either a simple keyword for a viewstate or else a pseudo-querystring.
235
+ * historyData (optional):
236
+ This is for complex data that is relevant only to the current browsing session. It will be available
237
+ to your application until the browser is closed. If the user comes back to a bookmarked history point
238
+ during a later session, this data will no longer be available. Don't rely on it for application
239
+ re-initialization from a bookmark.
240
+ * historyData.newTitle (optional):
241
+ This will swap out the html <title> attribute with a new value. If you have set a baseTitle using the
242
+ options bundle, the value will be plugged into the baseTitle by swapping out the @@@ replacement param.
243
+ */
244
+ add: function(newLocation, historyData) {
245
+
246
+ var that = this;
247
+
248
+ /*Escape the location and remove any leading hash symbols*/
249
+ var encodedLocation = encodeURI(this.removeHash(newLocation));
250
+
251
+ if (this.isSafari) {
252
+
253
+ /*Store the history data into history storage - pass in unencoded newLocation since
254
+ historyStorage does its own encoding*/
255
+ historyStorage.put(newLocation, historyData);
256
+
257
+ /*Save this as our current location*/
258
+ this.currentLocation = encodedLocation;
259
+
260
+ /*Change the browser location*/
261
+ window.location.hash = encodedLocation;
262
+
263
+ /*Save this to the Safari form field*/
264
+ this.putSafariState(encodedLocation);
265
+
266
+ this.changeTitle(historyData);
267
+
268
+ } else {
269
+
270
+ /*Most browsers require that we wait a certain amount of time before changing the location, such
271
+ as 200 MS; rather than forcing external callers to use window.setTimeout to account for this,
272
+ we internally handle it by putting requests in a queue.*/
273
+ var addImpl = function() {
274
+
275
+ /*Indicate that the current wait time is now less*/
276
+ if (that.currentWaitTime > 0) {
277
+ that.currentWaitTime = that.currentWaitTime - that.waitTime;
278
+ }
279
+
280
+ /*IE has a strange bug; if the encodedLocation is the same as _any_ preexisting id in the
281
+ document, then the history action gets recorded twice; throw a programmer exception if
282
+ there is an element with this ID*/
283
+ if (document.getElementById(encodedLocation) && that.debugMode) {
284
+ var e = "Exception: History locations can not have the same value as _any_ IDs that might be in the document,"
285
+ + " due to a bug in IE; please ask the developer to choose a history location that does not match any HTML"
286
+ + " IDs in this document. The following ID is already taken and cannot be a location: " + newLocation;
287
+ throw new Error(e);
288
+ }
289
+
290
+ /*Store the history data into history storage - pass in unencoded newLocation since
291
+ historyStorage does its own encoding*/
292
+ historyStorage.put(newLocation, historyData);
293
+
294
+ /*Indicate to the browser to ignore this upcomming location change since we're making it programmatically*/
295
+ that.ignoreLocationChange = true;
296
+
297
+ /*Indicate to IE that this is an atomic location change block*/
298
+ that.ieAtomicLocationChange = true;
299
+
300
+ /*Save this as our current location*/
301
+ that.currentLocation = encodedLocation;
302
+
303
+ /*Change the browser location*/
304
+ window.location.hash = encodedLocation;
305
+
306
+ /*Change the hidden iframe's location if on IE*/
307
+ if (that.isIE) {
308
+ that.iframe.src = that.blankURL + encodedLocation;
309
+ }
310
+
311
+ /*End of atomic location change block for IE*/
312
+ that.ieAtomicLocationChange = false;
313
+
314
+ that.changeTitle(historyData);
315
+
316
+ };
317
+
318
+ /*Now queue up this add request*/
319
+ window.setTimeout(addImpl, this.currentWaitTime);
320
+
321
+ /*Indicate that the next request will have to wait for awhile*/
322
+ this.currentWaitTime = this.currentWaitTime + this.waitTime;
323
+ }
324
+ },
325
+
326
+ /*Public*/
327
+ isFirstLoad: function() {
328
+ return this.firstLoad;
329
+ },
330
+
331
+ /*Public*/
332
+ getVersion: function() {
333
+ return this.VERSIONNUMBER;
334
+ },
335
+
336
+ /*- - - - - - - - - - - -*/
337
+
338
+ /*Private: Constant for our own internal history event called when the page is loaded*/
339
+ PAGELOADEDSTRING: "DhtmlHistory_pageLoaded",
340
+
341
+ VERSIONNUMBER: "0.8",
342
+
343
+ /*
344
+ Private: Pattern for title changes. Example: "Armchair DJ [@@@]" where @@@ will be relaced by values passed to add();
345
+ Default is just the title itself, hence "@@@"
346
+ */
347
+ baseTitle: "@@@",
348
+
349
+ /*Private: Placeholder variable for the original document title; will be set in ititialize()*/
350
+ originalTitle: null,
351
+
352
+ /*Private: URL for the blank html file we use for IE; can be overridden via the options bundle. Otherwise it must be served
353
+ in same directory as this library*/
354
+ blankURL: "/blank.html?",
355
+
356
+ /*Private: Our history change listener.*/
357
+ listener: null,
358
+
359
+ /*Private: MS to wait between add requests - will be reset for certain browsers*/
360
+ waitTime: 200,
361
+
362
+ /*Private: MS before an add request can execute*/
363
+ currentWaitTime: 0,
364
+
365
+ /*Private: Our current hash location, without the "#" symbol.*/
366
+ currentLocation: null,
367
+
368
+ /*Private: Hidden iframe used to IE to detect history changes*/
369
+ iframe: null,
370
+
371
+ /*Private: Flags and DOM references used only by Safari*/
372
+ safariHistoryStartPoint: null,
373
+ safariStack: null,
374
+ safariLength: null,
375
+
376
+ /*Private: Flag used to keep checkLocation() from doing anything when it discovers location changes we've made ourselves
377
+ programmatically with the add() method. Basically, add() sets this to true. When checkLocation() discovers it's true,
378
+ it refrains from firing our listener, then resets the flag to false for next cycle. That way, our listener only gets fired on
379
+ history change events triggered by the user via back/forward buttons and manual hash changes. This flag also helps us set up
380
+ IE's special iframe-based method of handling history changes.*/
381
+ ignoreLocationChange: null,
382
+
383
+ /*Private: A flag that indicates that we should fire a history change event when we are ready, i.e. after we are initialized and
384
+ we have a history change listener. This is needed due to an edge case in browsers other than IE; if you leave a page entirely
385
+ then return, we must fire this as a history change event. Unfortunately, we have lost all references to listeners from earlier,
386
+ because JavaScript clears out.*/
387
+ fireOnNewListener: null,
388
+
389
+ /*Private: A variable that indicates whether this is the first time this page has been loaded. If you go to a web page, leave it
390
+ for another one, and then return, the page's onload listener fires again. We need a way to differentiate between the first page
391
+ load and subsequent ones. This variable works hand in hand with the pageLoaded variable we store into historyStorage.*/
392
+ firstLoad: null,
393
+
394
+ /*Private: A variable to handle an important edge case in IE. In IE, if a user manually types an address into their browser's
395
+ location bar, we must intercept this by calling checkLocation() at regular intervals. However, if we are programmatically
396
+ changing the location bar ourselves using the add() method, we need to ignore these changes in checkLocation(). Unfortunately,
397
+ these changes take several lines of code to complete, so for the duration of those lines of code, we set this variable to true.
398
+ That signals to checkLocation() to ignore the change-in-progress. Once we're done with our chunk of location-change code in
399
+ add(), we set this back to false. We'll do the same thing when capturing user-entered address changes in checkLocation itself.*/
400
+ ieAtomicLocationChange: null,
401
+
402
+ /*Private: Generic utility function for attaching events*/
403
+ addEventListener: function(o,e,l) {
404
+ if (o.addEventListener) {
405
+ o.addEventListener(e,l,false);
406
+ } else if (o.attachEvent) {
407
+ o.attachEvent('on'+e,function() {
408
+ l(window.event);
409
+ });
410
+ }
411
+ },
412
+
413
+
414
+ /*Private: Create IE-specific DOM nodes and overrides*/
415
+ createIE: function(initialHash) {
416
+ /*write out a hidden iframe for IE and set the amount of time to wait between add() requests*/
417
+ this.waitTime = 400;/*IE needs longer between history updates*/
418
+ var styles = (historyStorage.debugMode
419
+ ? 'width: 800px;height:80px;border:1px solid black;'
420
+ : historyStorage.hideStyles
421
+ );
422
+ var iframeID = "rshHistoryFrame";
423
+ var iframeHTML = '<iframe frameborder="0" id="' + iframeID + '" style="' + styles + '" src="' + this.blankURL + initialHash + '"></iframe>';
424
+ document.write(iframeHTML);
425
+ this.iframe = document.getElementById(iframeID);
426
+ },
427
+
428
+ /*Private: Create Opera-specific DOM nodes and overrides*/
429
+ createOpera: function() {
430
+ this.waitTime = 400;/*Opera needs longer between history updates*/
431
+ var imgHTML = '<img src="javascript:location.href=\'javascript:dhtmlHistory.checkLocation();\';" style="' + historyStorage.hideStyles + '" />';
432
+ document.write(imgHTML);
433
+ },
434
+
435
+ /*Private: Create Safari-specific DOM nodes and overrides*/
436
+ createSafari: function() {
437
+ var formID = "rshSafariForm";
438
+ var stackID = "rshSafariStack";
439
+ var lengthID = "rshSafariLength";
440
+ var formStyles = historyStorage.debugMode ? historyStorage.showStyles : historyStorage.hideStyles;
441
+ var stackStyles = (historyStorage.debugMode
442
+ ? 'width: 800px;height:80px;border:1px solid black;'
443
+ : historyStorage.hideStyles
444
+ );
445
+ var lengthStyles = (historyStorage.debugMode
446
+ ? 'width:800px;height:20px;border:1px solid black;margin:0;padding:0;'
447
+ : historyStorage.hideStyles
448
+ );
449
+ var safariHTML = '<form id="' + formID + '" style="' + formStyles + '">'
450
+ + '<textarea style="' + stackStyles + '" id="' + stackID + '">[]</textarea>'
451
+ + '<input type="text" style="' + lengthStyles + '" id="' + lengthID + '" value=""/>'
452
+ + '</form>';
453
+ document.write(safariHTML);
454
+ this.safariStack = document.getElementById(stackID);
455
+ this.safariLength = document.getElementById(lengthID);
456
+ if (!historyStorage.hasKey(this.PAGELOADEDSTRING)) {
457
+ this.safariHistoryStartPoint = history.length;
458
+ this.safariLength.value = this.safariHistoryStartPoint;
459
+ } else {
460
+ this.safariHistoryStartPoint = this.safariLength.value;
461
+ }
462
+ },
463
+
464
+ /*TODO: make this public again?*/
465
+ /*Private: Get browser's current hash location; for Safari, read value from a hidden form field*/
466
+ getCurrentLocation: function() {
467
+ var r = (this.isSafari
468
+ ? this.getSafariState()
469
+ : this.getCurrentHash()
470
+ );
471
+ return r;
472
+ },
473
+
474
+ /*TODO: make this public again?*/
475
+ /*Private: Manually parse the current url for a hash; tip of the hat to YUI*/
476
+ getCurrentHash: function() {
477
+ var r = window.location.href;
478
+ var i = r.indexOf("#");
479
+ return (i >= 0
480
+ ? r.substr(i+1)
481
+ : ""
482
+ );
483
+ },
484
+
485
+ /*Private: Safari method to read the history stack from a hidden form field*/
486
+ getSafariStack: function() {
487
+ var r = this.safariStack.value;
488
+ return historyStorage.fromJSON(r);
489
+ },
490
+ /*Private: Safari method to read from the history stack*/
491
+ getSafariState: function() {
492
+ var stack = this.getSafariStack();
493
+ var state = stack[history.length - this.safariHistoryStartPoint - 1];
494
+ return state;
495
+ },
496
+ /*Private: Safari method to write the history stack to a hidden form field*/
497
+ putSafariState: function(newLocation) {
498
+ var stack = this.getSafariStack();
499
+ stack[history.length - this.safariHistoryStartPoint] = newLocation;
500
+ this.safariStack.value = historyStorage.toJSON(stack);
501
+ },
502
+
503
+ /*Private: Notify the listener of new history changes.*/
504
+ fireHistoryEvent: function(newHash) {
505
+ var decodedHash = decodeURI(newHash)
506
+ /*extract the value from our history storage for this hash*/
507
+ var historyData = historyStorage.get(decodedHash);
508
+ this.changeTitle(historyData);
509
+ /*call our listener*/
510
+ this.listener.call(null, decodedHash, historyData);
511
+ },
512
+
513
+ /*Private: See if the browser has changed location. This is the primary history mechanism for Firefox. For IE, we use this to
514
+ handle an important edge case: if a user manually types in a new hash value into their IE location bar and press enter, we want to
515
+ to intercept this and notify any history listener.*/
516
+ checkLocation: function() {
517
+
518
+ /*Ignore any location changes that we made ourselves for browsers other than IE*/
519
+ if (!this.isIE && this.ignoreLocationChange) {
520
+ this.ignoreLocationChange = false;
521
+ return;
522
+ }
523
+
524
+ /*If we are dealing with IE and we are in the middle of making a location change from an iframe, ignore it*/
525
+ if (!this.isIE && this.ieAtomicLocationChange) {
526
+ return;
527
+ }
528
+
529
+ /*Get hash location*/
530
+ var hash = this.getCurrentLocation();
531
+
532
+ /*Do nothing if there's been no change*/
533
+ if (hash == this.currentLocation) {
534
+ return;
535
+ }
536
+
537
+ /*In IE, users manually entering locations into the browser; we do this by comparing the browser's location against the
538
+ iframe's location; if they differ, we are dealing with a manual event and need to place it inside our history, otherwise
539
+ we can return*/
540
+ this.ieAtomicLocationChange = true;
541
+
542
+ if (this.isIE && this.getIframeHash() != hash) {
543
+ this.iframe.src = this.blankURL + hash;
544
+ }
545
+ else if (this.isIE) {
546
+ /*the iframe is unchanged*/
547
+ return;
548
+ }
549
+
550
+ /*Save this new location*/
551
+ this.currentLocation = hash;
552
+
553
+ this.ieAtomicLocationChange = false;
554
+
555
+ /*Notify listeners of the change*/
556
+ this.fireHistoryEvent(hash);
557
+ },
558
+
559
+ /*Private: Get the current location of IE's hidden iframe.*/
560
+ getIframeHash: function() {
561
+ var doc = this.iframe.contentWindow.document;
562
+ var hash = String(doc.location.search);
563
+ if (hash.length == 1 && hash.charAt(0) == "?") {
564
+ hash = "";
565
+ }
566
+ else if (hash.length >= 2 && hash.charAt(0) == "?") {
567
+ hash = hash.substring(1);
568
+ }
569
+ return hash;
570
+ },
571
+
572
+ /*Private: Remove any leading hash that might be on a location.*/
573
+ removeHash: function(hashValue) {
574
+ var r;
575
+ if (hashValue === null || hashValue === undefined) {
576
+ r = null;
577
+ }
578
+ else if (hashValue === "") {
579
+ r = "";
580
+ }
581
+ else if (hashValue.length == 1 && hashValue.charAt(0) == "#") {
582
+ r = "";
583
+ }
584
+ else if (hashValue.length > 1 && hashValue.charAt(0) == "#") {
585
+ r = hashValue.substring(1);
586
+ }
587
+ else {
588
+ r = hashValue;
589
+ }
590
+ return r;
591
+ },
592
+
593
+ /*Private: For IE, tell when the hidden iframe has finished loading.*/
594
+ iframeLoaded: function(newLocation) {
595
+ /*ignore any location changes that we made ourselves*/
596
+ if (this.ignoreLocationChange) {
597
+ this.ignoreLocationChange = false;
598
+ return;
599
+ }
600
+
601
+ /*Get the new location*/
602
+ var hash = String(newLocation.search);
603
+ if (hash.length == 1 && hash.charAt(0) == "?") {
604
+ hash = "";
605
+ }
606
+ else if (hash.length >= 2 && hash.charAt(0) == "?") {
607
+ hash = hash.substring(1);
608
+ }
609
+ /*Keep the browser location bar in sync with the iframe hash*/
610
+ window.location.hash = hash;
611
+
612
+ /*Notify listeners of the change*/
613
+ this.fireHistoryEvent(hash);
614
+ }
615
+
616
+
617
+ };
618
+
619
+ /*
620
+ historyStorage: An object that uses a hidden form to store history state across page loads. The mechanism for doing so relies on
621
+ the fact that browsers save the text in form data for the life of the browser session, which means the text is still there when
622
+ the user navigates back to the page. This object can be used independently of the dhtmlHistory object for caching of Ajax
623
+ session information.
624
+
625
+ dependencies:
626
+ * json2007.js (included in a separate file) or alternate JSON methods passed in through an options bundle.
627
+ */
628
+ window.historyStorage = {
629
+
630
+ /*Public: Set up our historyStorage object for use by dhtmlHistory or other objects*/
631
+ setup: function(options) {
632
+
633
+ /*
634
+ options - object to store initialization parameters - passed in from dhtmlHistory or directly into historyStorage
635
+ options.debugMode - boolean that causes hidden form fields to be shown for development purposes.
636
+ options.toJSON - function to override default JSON stringifier
637
+ options.fromJSON - function to override default JSON parser
638
+ */
639
+
640
+ /*process init parameters*/
641
+ if (typeof options !== "undefined") {
642
+ if (options.debugMode) {
643
+ this.debugMode = options.debugMode;
644
+ }
645
+ if (options.toJSON) {
646
+ this.toJSON = options.toJSON;
647
+ }
648
+ if (options.fromJSON) {
649
+ this.fromJSON = options.fromJSON;
650
+ }
651
+ }
652
+
653
+ /*write a hidden form and textarea into the page; we'll stow our history stack here*/
654
+ var formID = "rshStorageForm";
655
+ var textareaID = "rshStorageField";
656
+ var formStyles = this.debugMode ? historyStorage.showStyles : historyStorage.hideStyles;
657
+ var textareaStyles = (historyStorage.debugMode
658
+ ? 'width: 800px;height:80px;border:1px solid black;'
659
+ : historyStorage.hideStyles
660
+ );
661
+ var textareaHTML = '<form id="' + formID + '" style="' + formStyles + '">'
662
+ + '<textarea id="' + textareaID + '" style="' + textareaStyles + '"></textarea>'
663
+ + '</form>';
664
+ document.write(textareaHTML);
665
+ this.storageField = document.getElementById(textareaID);
666
+ if (typeof window.opera !== "undefined") {
667
+ this.storageField.focus();/*Opera needs to focus this element before persisting values in it*/
668
+ }
669
+ },
670
+
671
+ /*Public*/
672
+ put: function(key, value) {
673
+
674
+ var encodedKey = encodeURI(key);
675
+
676
+ this.assertValidKey(encodedKey);
677
+ /*if we already have a value for this, remove the value before adding the new one*/
678
+ if (this.hasKey(key)) {
679
+ this.remove(key);
680
+ }
681
+ /*store this new key*/
682
+ this.storageHash[encodedKey] = value;
683
+ /*save and serialize the hashtable into the form*/
684
+ this.saveHashTable();
685
+ },
686
+
687
+ /*Public*/
688
+ get: function(key) {
689
+
690
+ var encodedKey = encodeURI(key);
691
+
692
+ this.assertValidKey(encodedKey);
693
+ /*make sure the hash table has been loaded from the form*/
694
+ this.loadHashTable();
695
+ var value = this.storageHash[encodedKey];
696
+ if (value === undefined) {
697
+ value = null;
698
+ }
699
+ return value;
700
+ },
701
+
702
+ /*Public*/
703
+ remove: function(key) {
704
+
705
+ var encodedKey = encodeURI(key);
706
+
707
+ this.assertValidKey(encodedKey);
708
+ /*make sure the hash table has been loaded from the form*/
709
+ this.loadHashTable();
710
+ /*delete the value*/
711
+ delete this.storageHash[encodedKey];
712
+ /*serialize and save the hash table into the form*/
713
+ this.saveHashTable();
714
+ },
715
+
716
+ /*Public: Clears out all saved data.*/
717
+ reset: function() {
718
+ this.storageField.value = "";
719
+ this.storageHash = {};
720
+ },
721
+
722
+ /*Public*/
723
+ hasKey: function(key) {
724
+
725
+ var encodedKey = encodeURI(key);
726
+
727
+ this.assertValidKey(encodedKey);
728
+ /*make sure the hash table has been loaded from the form*/
729
+ this.loadHashTable();
730
+ return (typeof this.storageHash[encodedKey] !== "undefined");
731
+ },
732
+
733
+ /*Public*/
734
+ isValidKey: function(key) {
735
+ return (typeof key === "string");
736
+ //TODO - should we ban hash signs and other special characters?
737
+ },
738
+
739
+ /*- - - - - - - - - - - -*/
740
+
741
+ /*Private - CSS strings utilized by both objects to hide or show behind-the-scenes DOM elements*/
742
+ showStyles: 'border:0;margin:0;padding:0;',
743
+ hideStyles: 'left:-1000px;top:-1000px;width:1px;height:1px;border:0;position:absolute;',
744
+
745
+ /*Private - debug mode flag*/
746
+ debugMode: false,
747
+
748
+ /*Private: Our hash of key name/values.*/
749
+ storageHash: {},
750
+
751
+ /*Private: If true, we have loaded our hash table out of the storage form.*/
752
+ hashLoaded: false,
753
+
754
+ /*Private: DOM reference to our history field*/
755
+ storageField: null,
756
+
757
+ /*Private: Assert that a key is valid; throw an exception if it not.*/
758
+ assertValidKey: function(key) {
759
+ var isValid = this.isValidKey(key);
760
+ if (!isValid && this.debugMode) {
761
+ throw new Error("Please provide a valid key for window.historyStorage. Invalid key = " + key + ".");
762
+ }
763
+ },
764
+
765
+ /*Private: Load the hash table up from the form.*/
766
+ loadHashTable: function() {
767
+ if (!this.hashLoaded) {
768
+ var serializedHashTable = this.storageField.value;
769
+ if (serializedHashTable !== "" && serializedHashTable !== null) {
770
+ this.storageHash = this.fromJSON(serializedHashTable);
771
+ this.hashLoaded = true;
772
+ }
773
+ }
774
+ },
775
+ /*Private: Save the hash table into the form.*/
776
+ saveHashTable: function() {
777
+ this.loadHashTable();
778
+ var serializedHashTable = this.toJSON(this.storageHash);
779
+ this.storageField.value = serializedHashTable;
780
+ },
781
+ /*Private: Bridges for our JSON implementations - both rely on 2007 JSON.org library - can be overridden by options bundle*/
782
+ toJSON: function(o) {
783
+ return o.toJSONString();
784
+ },
785
+ fromJSON: function(s) {
786
+ return s.parseJSON();
787
+ }
788
+ };
789
+
790
+
791
+ /*******************************************************************/
792
+ /** QueryString Object from http://adamv.com/dev/javascript/querystring */
793
+ /* Client-side access to querystring name=value pairs
794
+ Version 1.3
795
+ 28 May 2008
796
+
797
+ License (Simplified BSD):
798
+ http://adamv.com/dev/javascript/qslicense.txt
799
+ */
800
+ function Querystring(qs) { // optionally pass a querystring to parse
801
+ this.params = {};
802
+
803
+ if (qs == null) qs = location.search.substring(1, location.search.length);
804
+ if (qs.length == 0) return;
805
+
806
+ // Turn <plus> back to <space>
807
+ // See: http://www.w3.org/TR/REC-html40/interact/forms.html#h-17.13.4.1
808
+ qs = qs.replace(/\+/g, ' ');
809
+ var args = qs.split('&'); // parse out name/value pairs separated via &
810
+
811
+ // split out each name=value pair
812
+ for (var i = 0; i < args.length; i++) {
813
+ var pair = args[i].split('=');
814
+ var name = decodeURI(pair[0]);
815
+
816
+ var value = (pair.length==2)
817
+ ? decodeURI(pair[1])
818
+ : name;
819
+
820
+ this.params[name] = value;
821
+ }
822
+ }
823
+
824
+ Querystring.prototype.get = function(key, default_) {
825
+ var value = this.params[key];
826
+ return (value != null) ? value : default_;
827
+ }
828
+
829
+ Querystring.prototype.contains = function(key) {
830
+ var value = this.params[key];
831
+ return (value != null);
832
+ }
833
+
834
+ /*******************************************************************/
835
+ /* Added by Ed Wildgoose - MailASail */
836
+ /* Initialise the library and add our history callback */
837
+ /*******************************************************************/
838
+ window.dhtmlHistory.create({
839
+ toJSON: function(o) {
840
+ return Object.toJSON(o);
841
+ }
842
+ , fromJSON: function(s) {
843
+ return s.evalJSON();
844
+ }
845
+
846
+ // Enable this to assist with debugging
847
+ // , debugMode: true
848
+
849
+ // dhtmlHistory has been modified not to need the next line
850
+ // But left in for robustness when updating dhtmlHistory
851
+ , blankURL: '/blank.html?'
852
+ });
853
+
854
+ /** Our callback to receive history
855
+ change events. */
856
+ var handleHistoryChange = function(pageId, pageData) {
857
+ if (!pageData) return;
858
+ var info = pageId.split(':');
859
+ var id = info[0];
860
+ pageData += '&_method=get';
861
+ new Ajax.Request(pageData, {asynchronous:true, evalScripts:true, method: 'get', onLoading:function(request){Element.show(id+'-pagination-loading-indicator');}});
862
+ }
863
+
864
+ window.onload = function() {
865
+ dhtmlHistory.initialize(handleHistoryChange);
866
+ };
867
+