actionpack_csi 2.3.5.p6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (429) hide show
  1. data/CHANGELOG +5184 -0
  2. data/MIT-LICENSE +21 -0
  3. data/README +409 -0
  4. data/RUNNING_UNIT_TESTS +24 -0
  5. data/Rakefile +160 -0
  6. data/install.rb +30 -0
  7. data/lib/action_controller/assertions/dom_assertions.rb +55 -0
  8. data/lib/action_controller/assertions/model_assertions.rb +21 -0
  9. data/lib/action_controller/assertions/response_assertions.rb +160 -0
  10. data/lib/action_controller/assertions/routing_assertions.rb +146 -0
  11. data/lib/action_controller/assertions/selector_assertions.rb +638 -0
  12. data/lib/action_controller/assertions/tag_assertions.rb +127 -0
  13. data/lib/action_controller/base.rb +1423 -0
  14. data/lib/action_controller/benchmarking.rb +107 -0
  15. data/lib/action_controller/caching/actions.rb +177 -0
  16. data/lib/action_controller/caching/fragments.rb +120 -0
  17. data/lib/action_controller/caching/pages.rb +152 -0
  18. data/lib/action_controller/caching/sweeper.rb +45 -0
  19. data/lib/action_controller/caching/sweeping.rb +55 -0
  20. data/lib/action_controller/caching.rb +71 -0
  21. data/lib/action_controller/cgi_ext/cookie.rb +112 -0
  22. data/lib/action_controller/cgi_ext/query_extension.rb +22 -0
  23. data/lib/action_controller/cgi_ext/stdinput.rb +24 -0
  24. data/lib/action_controller/cgi_ext.rb +15 -0
  25. data/lib/action_controller/cgi_process.rb +77 -0
  26. data/lib/action_controller/cookies.rb +95 -0
  27. data/lib/action_controller/dispatcher.rb +133 -0
  28. data/lib/action_controller/failsafe.rb +86 -0
  29. data/lib/action_controller/filters.rb +680 -0
  30. data/lib/action_controller/flash.rb +171 -0
  31. data/lib/action_controller/headers.rb +33 -0
  32. data/lib/action_controller/helpers.rb +225 -0
  33. data/lib/action_controller/http_authentication.rb +309 -0
  34. data/lib/action_controller/integration.rb +692 -0
  35. data/lib/action_controller/layout.rb +286 -0
  36. data/lib/action_controller/middleware_stack.rb +119 -0
  37. data/lib/action_controller/middlewares.rb +14 -0
  38. data/lib/action_controller/mime_responds.rb +193 -0
  39. data/lib/action_controller/mime_type.rb +212 -0
  40. data/lib/action_controller/mime_types.rb +21 -0
  41. data/lib/action_controller/params_parser.rb +77 -0
  42. data/lib/action_controller/performance_test.rb +15 -0
  43. data/lib/action_controller/polymorphic_routes.rb +189 -0
  44. data/lib/action_controller/rack_lint_patch.rb +36 -0
  45. data/lib/action_controller/record_identifier.rb +104 -0
  46. data/lib/action_controller/reloader.rb +54 -0
  47. data/lib/action_controller/request.rb +493 -0
  48. data/lib/action_controller/request_forgery_protection.rb +113 -0
  49. data/lib/action_controller/rescue.rb +183 -0
  50. data/lib/action_controller/resources.rb +682 -0
  51. data/lib/action_controller/response.rb +239 -0
  52. data/lib/action_controller/routing/builder.rb +197 -0
  53. data/lib/action_controller/routing/optimisations.rb +130 -0
  54. data/lib/action_controller/routing/recognition_optimisation.rb +167 -0
  55. data/lib/action_controller/routing/route.rb +265 -0
  56. data/lib/action_controller/routing/route_set.rb +502 -0
  57. data/lib/action_controller/routing/routing_ext.rb +49 -0
  58. data/lib/action_controller/routing/segments.rb +343 -0
  59. data/lib/action_controller/routing.rb +388 -0
  60. data/lib/action_controller/session/abstract_store.rb +181 -0
  61. data/lib/action_controller/session/cookie_store.rb +221 -0
  62. data/lib/action_controller/session/mem_cache_store.rb +51 -0
  63. data/lib/action_controller/session_management.rb +54 -0
  64. data/lib/action_controller/status_codes.rb +88 -0
  65. data/lib/action_controller/streaming.rb +181 -0
  66. data/lib/action_controller/string_coercion.rb +29 -0
  67. data/lib/action_controller/templates/rescues/_request_and_response.erb +24 -0
  68. data/lib/action_controller/templates/rescues/_trace.erb +26 -0
  69. data/lib/action_controller/templates/rescues/diagnostics.erb +11 -0
  70. data/lib/action_controller/templates/rescues/layout.erb +29 -0
  71. data/lib/action_controller/templates/rescues/missing_template.erb +2 -0
  72. data/lib/action_controller/templates/rescues/routing_error.erb +10 -0
  73. data/lib/action_controller/templates/rescues/template_error.erb +21 -0
  74. data/lib/action_controller/templates/rescues/unknown_action.erb +2 -0
  75. data/lib/action_controller/test_case.rb +209 -0
  76. data/lib/action_controller/test_process.rb +580 -0
  77. data/lib/action_controller/translation.rb +13 -0
  78. data/lib/action_controller/uploaded_file.rb +44 -0
  79. data/lib/action_controller/url_rewriter.rb +216 -0
  80. data/lib/action_controller/vendor/html-scanner/html/document.rb +68 -0
  81. data/lib/action_controller/vendor/html-scanner/html/node.rb +537 -0
  82. data/lib/action_controller/vendor/html-scanner/html/sanitizer.rb +173 -0
  83. data/lib/action_controller/vendor/html-scanner/html/selector.rb +828 -0
  84. data/lib/action_controller/vendor/html-scanner/html/tokenizer.rb +105 -0
  85. data/lib/action_controller/vendor/html-scanner/html/version.rb +11 -0
  86. data/lib/action_controller/vendor/html-scanner.rb +16 -0
  87. data/lib/action_controller/verification.rb +130 -0
  88. data/lib/action_controller.rb +113 -0
  89. data/lib/action_pack/version.rb +9 -0
  90. data/lib/action_pack.rb +24 -0
  91. data/lib/action_view/base.rb +362 -0
  92. data/lib/action_view/erb/util.rb +44 -0
  93. data/lib/action_view/helpers/active_record_helper.rb +305 -0
  94. data/lib/action_view/helpers/asset_tag_helper.rb +694 -0
  95. data/lib/action_view/helpers/atom_feed_helper.rb +198 -0
  96. data/lib/action_view/helpers/benchmark_helper.rb +54 -0
  97. data/lib/action_view/helpers/cache_helper.rb +39 -0
  98. data/lib/action_view/helpers/capture_helper.rb +136 -0
  99. data/lib/action_view/helpers/date_helper.rb +988 -0
  100. data/lib/action_view/helpers/debug_helper.rb +38 -0
  101. data/lib/action_view/helpers/form_helper.rb +1074 -0
  102. data/lib/action_view/helpers/form_options_helper.rb +600 -0
  103. data/lib/action_view/helpers/form_tag_helper.rb +487 -0
  104. data/lib/action_view/helpers/javascript_helper.rb +208 -0
  105. data/lib/action_view/helpers/number_helper.rb +308 -0
  106. data/lib/action_view/helpers/prototype_helper.rb +1305 -0
  107. data/lib/action_view/helpers/raw_output_helper.rb +9 -0
  108. data/lib/action_view/helpers/record_identification_helper.rb +20 -0
  109. data/lib/action_view/helpers/record_tag_helper.rb +58 -0
  110. data/lib/action_view/helpers/sanitize_helper.rb +259 -0
  111. data/lib/action_view/helpers/scriptaculous_helper.rb +226 -0
  112. data/lib/action_view/helpers/tag_helper.rb +150 -0
  113. data/lib/action_view/helpers/text_helper.rb +587 -0
  114. data/lib/action_view/helpers/translation_helper.rb +39 -0
  115. data/lib/action_view/helpers/url_helper.rb +639 -0
  116. data/lib/action_view/helpers.rb +59 -0
  117. data/lib/action_view/inline_template.rb +19 -0
  118. data/lib/action_view/locale/en.yml +117 -0
  119. data/lib/action_view/partials.rb +240 -0
  120. data/lib/action_view/paths.rb +69 -0
  121. data/lib/action_view/reloadable_template.rb +117 -0
  122. data/lib/action_view/renderable.rb +95 -0
  123. data/lib/action_view/renderable_partial.rb +47 -0
  124. data/lib/action_view/safe_buffer.rb +28 -0
  125. data/lib/action_view/template.rb +252 -0
  126. data/lib/action_view/template_error.rb +99 -0
  127. data/lib/action_view/template_handler.rb +34 -0
  128. data/lib/action_view/template_handlers/builder.rb +17 -0
  129. data/lib/action_view/template_handlers/erb.rb +22 -0
  130. data/lib/action_view/template_handlers/rjs.rb +13 -0
  131. data/lib/action_view/template_handlers.rb +48 -0
  132. data/lib/action_view/test_case.rb +162 -0
  133. data/lib/action_view.rb +58 -0
  134. data/lib/actionpack.rb +2 -0
  135. data/test/abstract_unit.rb +61 -0
  136. data/test/active_record_unit.rb +104 -0
  137. data/test/activerecord/active_record_store_test.rb +174 -0
  138. data/test/activerecord/render_partial_with_record_identification_test.rb +188 -0
  139. data/test/adv_attr_test.rb +20 -0
  140. data/test/controller/action_pack_assertions_test.rb +543 -0
  141. data/test/controller/addresses_render_test.rb +37 -0
  142. data/test/controller/assert_select_test.rb +734 -0
  143. data/test/controller/base_test.rb +217 -0
  144. data/test/controller/benchmark_test.rb +32 -0
  145. data/test/controller/caching_test.rb +729 -0
  146. data/test/controller/capture_test.rb +66 -0
  147. data/test/controller/content_type_test.rb +168 -0
  148. data/test/controller/controller_fixtures/app/controllers/admin/user_controller.rb +0 -0
  149. data/test/controller/controller_fixtures/app/controllers/user_controller.rb +0 -0
  150. data/test/controller/controller_fixtures/vendor/plugins/bad_plugin/lib/plugin_controller.rb +0 -0
  151. data/test/controller/cookie_test.rb +134 -0
  152. data/test/controller/deprecation/deprecated_base_methods_test.rb +32 -0
  153. data/test/controller/dispatcher_test.rb +144 -0
  154. data/test/controller/dom_assertions_test.rb +53 -0
  155. data/test/controller/failsafe_test.rb +60 -0
  156. data/test/controller/fake_controllers.rb +33 -0
  157. data/test/controller/fake_models.rb +19 -0
  158. data/test/controller/filter_params_test.rb +52 -0
  159. data/test/controller/filters_test.rb +885 -0
  160. data/test/controller/flash_test.rb +147 -0
  161. data/test/controller/header_test.rb +14 -0
  162. data/test/controller/helper_test.rb +224 -0
  163. data/test/controller/html-scanner/cdata_node_test.rb +15 -0
  164. data/test/controller/html-scanner/document_test.rb +148 -0
  165. data/test/controller/html-scanner/node_test.rb +89 -0
  166. data/test/controller/html-scanner/sanitizer_test.rb +274 -0
  167. data/test/controller/html-scanner/tag_node_test.rb +238 -0
  168. data/test/controller/html-scanner/text_node_test.rb +50 -0
  169. data/test/controller/html-scanner/tokenizer_test.rb +131 -0
  170. data/test/controller/http_basic_authentication_test.rb +113 -0
  171. data/test/controller/http_digest_authentication_test.rb +254 -0
  172. data/test/controller/integration_test.rb +483 -0
  173. data/test/controller/layout_test.rb +215 -0
  174. data/test/controller/logging_test.rb +46 -0
  175. data/test/controller/middleware_stack_test.rb +90 -0
  176. data/test/controller/mime_responds_test.rb +536 -0
  177. data/test/controller/mime_type_test.rb +93 -0
  178. data/test/controller/polymorphic_routes_test.rb +297 -0
  179. data/test/controller/rack_test.rb +311 -0
  180. data/test/controller/record_identifier_test.rb +139 -0
  181. data/test/controller/redirect_test.rb +285 -0
  182. data/test/controller/reloader_test.rb +124 -0
  183. data/test/controller/render_test.rb +1762 -0
  184. data/test/controller/request/json_params_parsing_test.rb +65 -0
  185. data/test/controller/request/multipart_params_parsing_test.rb +162 -0
  186. data/test/controller/request/query_string_parsing_test.rb +120 -0
  187. data/test/controller/request/test_request_test.rb +35 -0
  188. data/test/controller/request/url_encoded_params_parsing_test.rb +146 -0
  189. data/test/controller/request/xml_params_parsing_test.rb +103 -0
  190. data/test/controller/request_forgery_protection_test.rb +265 -0
  191. data/test/controller/request_test.rb +395 -0
  192. data/test/controller/rescue_test.rb +536 -0
  193. data/test/controller/resources_test.rb +1393 -0
  194. data/test/controller/routing_test.rb +2591 -0
  195. data/test/controller/selector_test.rb +628 -0
  196. data/test/controller/send_file_test.rb +171 -0
  197. data/test/controller/session/cookie_store_test.rb +216 -0
  198. data/test/controller/session/mem_cache_store_test.rb +127 -0
  199. data/test/controller/session/test_session_test.rb +58 -0
  200. data/test/controller/test_test.rb +700 -0
  201. data/test/controller/translation_test.rb +26 -0
  202. data/test/controller/url_rewriter_test.rb +385 -0
  203. data/test/controller/verification_test.rb +270 -0
  204. data/test/controller/view_paths_test.rb +141 -0
  205. data/test/controller/webservice_test.rb +273 -0
  206. data/test/fixtures/_top_level_partial.html.erb +1 -0
  207. data/test/fixtures/_top_level_partial_only.erb +1 -0
  208. data/test/fixtures/addresses/list.erb +1 -0
  209. data/test/fixtures/alternate_helpers/foo_helper.rb +3 -0
  210. data/test/fixtures/bad_customers/_bad_customer.html.erb +1 -0
  211. data/test/fixtures/companies.yml +24 -0
  212. data/test/fixtures/company.rb +10 -0
  213. data/test/fixtures/content_type/render_default_content_types_for_respond_to.rhtml +1 -0
  214. data/test/fixtures/content_type/render_default_for_rhtml.rhtml +1 -0
  215. data/test/fixtures/content_type/render_default_for_rjs.rjs +1 -0
  216. data/test/fixtures/content_type/render_default_for_rxml.rxml +1 -0
  217. data/test/fixtures/customers/_customer.html.erb +1 -0
  218. data/test/fixtures/db_definitions/sqlite.sql +49 -0
  219. data/test/fixtures/developer.rb +9 -0
  220. data/test/fixtures/developers/_developer.erb +1 -0
  221. data/test/fixtures/developers.yml +21 -0
  222. data/test/fixtures/developers_projects.yml +13 -0
  223. data/test/fixtures/failsafe/500.html +1 -0
  224. data/test/fixtures/fun/games/_game.erb +1 -0
  225. data/test/fixtures/fun/games/hello_world.erb +1 -0
  226. data/test/fixtures/fun/serious/games/_game.erb +1 -0
  227. data/test/fixtures/functional_caching/_partial.erb +3 -0
  228. data/test/fixtures/functional_caching/formatted_fragment_cached.html.erb +3 -0
  229. data/test/fixtures/functional_caching/formatted_fragment_cached.js.rjs +6 -0
  230. data/test/fixtures/functional_caching/formatted_fragment_cached.xml.builder +5 -0
  231. data/test/fixtures/functional_caching/fragment_cached.html.erb +2 -0
  232. data/test/fixtures/functional_caching/html_fragment_cached_with_partial.html.erb +1 -0
  233. data/test/fixtures/functional_caching/inline_fragment_cached.html.erb +2 -0
  234. data/test/fixtures/functional_caching/js_fragment_cached_with_partial.js.rjs +1 -0
  235. data/test/fixtures/good_customers/_good_customer.html.erb +1 -0
  236. data/test/fixtures/helpers/abc_helper.rb +5 -0
  237. data/test/fixtures/helpers/fun/games_helper.rb +3 -0
  238. data/test/fixtures/helpers/fun/pdf_helper.rb +3 -0
  239. data/test/fixtures/layout_tests/abs_path_layout.rhtml +1 -0
  240. data/test/fixtures/layout_tests/alt/hello.rhtml +1 -0
  241. data/test/fixtures/layout_tests/alt/layouts/alt.rhtml +0 -0
  242. data/test/fixtures/layout_tests/layouts/controller_name_space/nested.rhtml +1 -0
  243. data/test/fixtures/layout_tests/layouts/item.rhtml +1 -0
  244. data/test/fixtures/layout_tests/layouts/layout_test.rhtml +1 -0
  245. data/test/fixtures/layout_tests/layouts/multiple_extensions.html.erb +1 -0
  246. data/test/fixtures/layout_tests/layouts/third_party_template_library.mab +1 -0
  247. data/test/fixtures/layout_tests/views/hello.rhtml +1 -0
  248. data/test/fixtures/layouts/_column.html.erb +2 -0
  249. data/test/fixtures/layouts/block_with_layout.erb +3 -0
  250. data/test/fixtures/layouts/builder.builder +3 -0
  251. data/test/fixtures/layouts/default_html.html.erb +1 -0
  252. data/test/fixtures/layouts/partial_with_layout.erb +3 -0
  253. data/test/fixtures/layouts/standard.erb +1 -0
  254. data/test/fixtures/layouts/talk_from_action.erb +2 -0
  255. data/test/fixtures/layouts/xhr.html.erb +2 -0
  256. data/test/fixtures/layouts/yield.erb +2 -0
  257. data/test/fixtures/mascot.rb +3 -0
  258. data/test/fixtures/mascots/_mascot.html.erb +1 -0
  259. data/test/fixtures/mascots.yml +4 -0
  260. data/test/fixtures/multipart/binary_file +0 -0
  261. data/test/fixtures/multipart/boundary_problem_file +10 -0
  262. data/test/fixtures/multipart/bracketed_param +5 -0
  263. data/test/fixtures/multipart/empty +10 -0
  264. data/test/fixtures/multipart/hello.txt +1 -0
  265. data/test/fixtures/multipart/large_text_file +10 -0
  266. data/test/fixtures/multipart/mixed_files +0 -0
  267. data/test/fixtures/multipart/mona_lisa.jpg +0 -0
  268. data/test/fixtures/multipart/none +9 -0
  269. data/test/fixtures/multipart/single_parameter +5 -0
  270. data/test/fixtures/multipart/text_file +10 -0
  271. data/test/fixtures/override/test/hello_world.erb +1 -0
  272. data/test/fixtures/override2/layouts/test/sub.erb +1 -0
  273. data/test/fixtures/post_test/layouts/post.html.erb +1 -0
  274. data/test/fixtures/post_test/layouts/super_post.iphone.erb +1 -0
  275. data/test/fixtures/post_test/post/index.html.erb +1 -0
  276. data/test/fixtures/post_test/post/index.iphone.erb +1 -0
  277. data/test/fixtures/post_test/super_post/index.html.erb +1 -0
  278. data/test/fixtures/post_test/super_post/index.iphone.erb +1 -0
  279. data/test/fixtures/project.rb +3 -0
  280. data/test/fixtures/projects/_project.erb +1 -0
  281. data/test/fixtures/projects.yml +7 -0
  282. data/test/fixtures/public/404.html +1 -0
  283. data/test/fixtures/public/500.da.html +1 -0
  284. data/test/fixtures/public/500.html +1 -0
  285. data/test/fixtures/public/absolute/test.css +23 -0
  286. data/test/fixtures/public/absolute/test.js +63 -0
  287. data/test/fixtures/public/images/rails.png +0 -0
  288. data/test/fixtures/public/javascripts/application.js +1 -0
  289. data/test/fixtures/public/javascripts/bank.js +1 -0
  290. data/test/fixtures/public/javascripts/controls.js +1 -0
  291. data/test/fixtures/public/javascripts/dragdrop.js +1 -0
  292. data/test/fixtures/public/javascripts/effects.js +1 -0
  293. data/test/fixtures/public/javascripts/prototype.js +1 -0
  294. data/test/fixtures/public/javascripts/robber.js +1 -0
  295. data/test/fixtures/public/javascripts/subdir/subdir.js +1 -0
  296. data/test/fixtures/public/javascripts/version.1.0.js +1 -0
  297. data/test/fixtures/public/stylesheets/bank.css +1 -0
  298. data/test/fixtures/public/stylesheets/robber.css +1 -0
  299. data/test/fixtures/public/stylesheets/subdir/subdir.css +1 -0
  300. data/test/fixtures/public/stylesheets/version.1.0.css +1 -0
  301. data/test/fixtures/quiz/questions/_question.html.erb +1 -0
  302. data/test/fixtures/replies/_reply.erb +1 -0
  303. data/test/fixtures/replies.yml +15 -0
  304. data/test/fixtures/reply.rb +7 -0
  305. data/test/fixtures/respond_to/all_types_with_layout.html.erb +1 -0
  306. data/test/fixtures/respond_to/all_types_with_layout.js.rjs +1 -0
  307. data/test/fixtures/respond_to/custom_constant_handling_without_block.mobile.erb +1 -0
  308. data/test/fixtures/respond_to/iphone_with_html_response_type.html.erb +1 -0
  309. data/test/fixtures/respond_to/iphone_with_html_response_type.iphone.erb +1 -0
  310. data/test/fixtures/respond_to/layouts/missing.html.erb +1 -0
  311. data/test/fixtures/respond_to/layouts/standard.html.erb +1 -0
  312. data/test/fixtures/respond_to/layouts/standard.iphone.erb +1 -0
  313. data/test/fixtures/respond_to/using_defaults.html.erb +1 -0
  314. data/test/fixtures/respond_to/using_defaults.js.rjs +1 -0
  315. data/test/fixtures/respond_to/using_defaults.xml.builder +1 -0
  316. data/test/fixtures/respond_to/using_defaults_with_type_list.html.erb +1 -0
  317. data/test/fixtures/respond_to/using_defaults_with_type_list.js.rjs +1 -0
  318. data/test/fixtures/respond_to/using_defaults_with_type_list.xml.builder +1 -0
  319. data/test/fixtures/scope/test/modgreet.erb +1 -0
  320. data/test/fixtures/shared.html.erb +1 -0
  321. data/test/fixtures/symlink_parent/symlinked_layout.erb +5 -0
  322. data/test/fixtures/test/_counter.html.erb +1 -0
  323. data/test/fixtures/test/_customer.erb +1 -0
  324. data/test/fixtures/test/_customer_counter.erb +1 -0
  325. data/test/fixtures/test/_customer_greeting.erb +1 -0
  326. data/test/fixtures/test/_customer_with_var.erb +1 -0
  327. data/test/fixtures/test/_form.erb +1 -0
  328. data/test/fixtures/test/_from_helper.erb +1 -0
  329. data/test/fixtures/test/_hash_greeting.erb +1 -0
  330. data/test/fixtures/test/_hash_object.erb +2 -0
  331. data/test/fixtures/test/_hello.builder +1 -0
  332. data/test/fixtures/test/_labelling_form.erb +1 -0
  333. data/test/fixtures/test/_layout_for_block_with_args.html.erb +3 -0
  334. data/test/fixtures/test/_layout_for_partial.html.erb +3 -0
  335. data/test/fixtures/test/_local_inspector.html.erb +1 -0
  336. data/test/fixtures/test/_one.html.erb +1 -0
  337. data/test/fixtures/test/_partial.erb +1 -0
  338. data/test/fixtures/test/_partial.html.erb +1 -0
  339. data/test/fixtures/test/_partial.js.erb +1 -0
  340. data/test/fixtures/test/_partial_for_use_in_layout.html.erb +1 -0
  341. data/test/fixtures/test/_partial_only.erb +1 -0
  342. data/test/fixtures/test/_partial_with_only_html_version.html.erb +1 -0
  343. data/test/fixtures/test/_person.erb +2 -0
  344. data/test/fixtures/test/_raise.html.erb +1 -0
  345. data/test/fixtures/test/_two.html.erb +1 -0
  346. data/test/fixtures/test/action_talk_to_layout.erb +2 -0
  347. data/test/fixtures/test/calling_partial_with_layout.html.erb +1 -0
  348. data/test/fixtures/test/capturing.erb +4 -0
  349. data/test/fixtures/test/content_for.erb +2 -0
  350. data/test/fixtures/test/content_for_concatenated.erb +3 -0
  351. data/test/fixtures/test/content_for_with_parameter.erb +2 -0
  352. data/test/fixtures/test/delete_with_js.rjs +2 -0
  353. data/test/fixtures/test/dont_pick_me +1 -0
  354. data/test/fixtures/test/dot.directory/render_file_with_ivar.erb +1 -0
  355. data/test/fixtures/test/enum_rjs_test.rjs +6 -0
  356. data/test/fixtures/test/formatted_html_erb.html.erb +1 -0
  357. data/test/fixtures/test/formatted_xml_erb.builder +1 -0
  358. data/test/fixtures/test/formatted_xml_erb.html.erb +1 -0
  359. data/test/fixtures/test/formatted_xml_erb.xml.erb +1 -0
  360. data/test/fixtures/test/greeting.erb +1 -0
  361. data/test/fixtures/test/greeting.js.rjs +1 -0
  362. data/test/fixtures/test/hello.builder +4 -0
  363. data/test/fixtures/test/hello_world.da.html.erb +1 -0
  364. data/test/fixtures/test/hello_world.erb +1 -0
  365. data/test/fixtures/test/hello_world.erb~ +1 -0
  366. data/test/fixtures/test/hello_world.pt-BR.html.erb +1 -0
  367. data/test/fixtures/test/hello_world_container.builder +3 -0
  368. data/test/fixtures/test/hello_world_from_rxml.builder +4 -0
  369. data/test/fixtures/test/hello_world_with_layout_false.erb +1 -0
  370. data/test/fixtures/test/hello_xml_world.builder +11 -0
  371. data/test/fixtures/test/hyphen-ated.erb +1 -0
  372. data/test/fixtures/test/implicit_content_type.atom.builder +2 -0
  373. data/test/fixtures/test/list.erb +1 -0
  374. data/test/fixtures/test/malformed/malformed.en.html.erb~ +1 -0
  375. data/test/fixtures/test/malformed/malformed.erb~ +1 -0
  376. data/test/fixtures/test/malformed/malformed.html.erb~ +1 -0
  377. data/test/fixtures/test/nested_layout.erb +3 -0
  378. data/test/fixtures/test/non_erb_block_content_for.builder +4 -0
  379. data/test/fixtures/test/potential_conflicts.erb +4 -0
  380. data/test/fixtures/test/render_explicit_html_template.js.rjs +1 -0
  381. data/test/fixtures/test/render_file_from_template.html.erb +1 -0
  382. data/test/fixtures/test/render_file_with_ivar.erb +1 -0
  383. data/test/fixtures/test/render_file_with_locals.erb +1 -0
  384. data/test/fixtures/test/render_implicit_html_template.js.rjs +1 -0
  385. data/test/fixtures/test/render_implicit_html_template_from_xhr_request.da.html.erb +1 -0
  386. data/test/fixtures/test/render_implicit_html_template_from_xhr_request.html.erb +1 -0
  387. data/test/fixtures/test/render_implicit_js_template_without_layout.js.erb +1 -0
  388. data/test/fixtures/test/render_to_string_test.erb +1 -0
  389. data/test/fixtures/test/sub_template_raise.html.erb +1 -0
  390. data/test/fixtures/test/template.erb +1 -0
  391. data/test/fixtures/test/update_element_with_capture.erb +9 -0
  392. data/test/fixtures/test/using_layout_around_block.html.erb +1 -0
  393. data/test/fixtures/test/using_layout_around_block_with_args.html.erb +1 -0
  394. data/test/fixtures/test/utf8.html.erb +2 -0
  395. data/test/fixtures/topic.rb +3 -0
  396. data/test/fixtures/topics/_topic.html.erb +1 -0
  397. data/test/fixtures/topics.yml +22 -0
  398. data/test/template/active_record_helper_i18n_test.rb +44 -0
  399. data/test/template/active_record_helper_test.rb +302 -0
  400. data/test/template/asset_tag_helper_test.rb +771 -0
  401. data/test/template/atom_feed_helper_test.rb +315 -0
  402. data/test/template/benchmark_helper_test.rb +86 -0
  403. data/test/template/compiled_templates_test.rb +204 -0
  404. data/test/template/date_helper_i18n_test.rb +121 -0
  405. data/test/template/date_helper_test.rb +2485 -0
  406. data/test/template/erb_util_test.rb +24 -0
  407. data/test/template/form_helper_test.rb +1393 -0
  408. data/test/template/form_options_helper_i18n_test.rb +27 -0
  409. data/test/template/form_options_helper_test.rb +807 -0
  410. data/test/template/form_tag_helper_test.rb +344 -0
  411. data/test/template/javascript_helper_test.rb +106 -0
  412. data/test/template/number_helper_i18n_test.rb +69 -0
  413. data/test/template/number_helper_test.rb +132 -0
  414. data/test/template/prototype_helper_test.rb +639 -0
  415. data/test/template/raw_output_helper_test.rb +21 -0
  416. data/test/template/record_tag_helper_test.rb +58 -0
  417. data/test/template/render_test.rb +290 -0
  418. data/test/template/sanitize_helper_test.rb +57 -0
  419. data/test/template/scriptaculous_helper_test.rb +90 -0
  420. data/test/template/tag_helper_test.rb +98 -0
  421. data/test/template/template_test.rb +32 -0
  422. data/test/template/test_test.rb +54 -0
  423. data/test/template/text_helper_test.rb +543 -0
  424. data/test/template/translation_helper_test.rb +32 -0
  425. data/test/template/url_helper_test.rb +622 -0
  426. data/test/testing_sandbox.rb +15 -0
  427. data/test/view/safe_buffer_test.rb +36 -0
  428. data/test/view/test_case_test.rb +176 -0
  429. metadata +531 -0
@@ -0,0 +1,487 @@
1
+ require 'cgi'
2
+ require 'action_view/helpers/tag_helper'
3
+
4
+ module ActionView
5
+ module Helpers
6
+ # Provides a number of methods for creating form tags that doesn't rely on an Active Record object assigned to the template like
7
+ # FormHelper does. Instead, you provide the names and values manually.
8
+ #
9
+ # NOTE: The HTML options <tt>disabled</tt>, <tt>readonly</tt>, and <tt>multiple</tt> can all be treated as booleans. So specifying
10
+ # <tt>:disabled => true</tt> will give <tt>disabled="disabled"</tt>.
11
+ module FormTagHelper
12
+ # Starts a form tag that points the action to an url configured with <tt>url_for_options</tt> just like
13
+ # ActionController::Base#url_for. The method for the form defaults to POST.
14
+ #
15
+ # ==== Options
16
+ # * <tt>:multipart</tt> - If set to true, the enctype is set to "multipart/form-data".
17
+ # * <tt>:method</tt> - The method to use when submitting the form, usually either "get" or "post".
18
+ # If "put", "delete", or another verb is used, a hidden input with name <tt>_method</tt>
19
+ # is added to simulate the verb over post.
20
+ # * A list of parameters to feed to the URL the form will be posted to.
21
+ #
22
+ # ==== Examples
23
+ # form_tag('/posts')
24
+ # # => <form action="/posts" method="post">
25
+ #
26
+ # form_tag('/posts/1', :method => :put)
27
+ # # => <form action="/posts/1" method="put">
28
+ #
29
+ # form_tag('/upload', :multipart => true)
30
+ # # => <form action="/upload" method="post" enctype="multipart/form-data">
31
+ #
32
+ # <% form_tag '/posts' do -%>
33
+ # <div><%= submit_tag 'Save' %></div>
34
+ # <% end -%>
35
+ # # => <form action="/posts" method="post"><div><input type="submit" name="submit" value="Save" /></div></form>
36
+ def form_tag(url_for_options = {}, options = {}, *parameters_for_url, &block)
37
+ html_options = html_options_for_form(url_for_options, options, *parameters_for_url)
38
+ if block_given?
39
+ form_tag_in_block(html_options, &block)
40
+ else
41
+ form_tag_html(html_options)
42
+ end
43
+ end
44
+
45
+ # Creates a dropdown selection box, or if the <tt>:multiple</tt> option is set to true, a multiple
46
+ # choice selection box.
47
+ #
48
+ # Helpers::FormOptions can be used to create common select boxes such as countries, time zones, or
49
+ # associated records. <tt>option_tags</tt> is a string containing the option tags for the select box.
50
+ #
51
+ # ==== Options
52
+ # * <tt>:multiple</tt> - If set to true the selection will allow multiple choices.
53
+ # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
54
+ # * Any other key creates standard HTML attributes for the tag.
55
+ #
56
+ # ==== Examples
57
+ # select_tag "people", "<option>David</option>"
58
+ # # => <select id="people" name="people"><option>David</option></select>
59
+ #
60
+ # select_tag "count", "<option>1</option><option>2</option><option>3</option><option>4</option>"
61
+ # # => <select id="count" name="count"><option>1</option><option>2</option>
62
+ # # <option>3</option><option>4</option></select>
63
+ #
64
+ # select_tag "colors", "<option>Red</option><option>Green</option><option>Blue</option>", :multiple => true
65
+ # # => <select id="colors" multiple="multiple" name="colors[]"><option>Red</option>
66
+ # # <option>Green</option><option>Blue</option></select>
67
+ #
68
+ # select_tag "locations", "<option>Home</option><option selected="selected">Work</option><option>Out</option>"
69
+ # # => <select id="locations" name="locations"><option>Home</option><option selected='selected'>Work</option>
70
+ # # <option>Out</option></select>
71
+ #
72
+ # select_tag "access", "<option>Read</option><option>Write</option>", :multiple => true, :class => 'form_input'
73
+ # # => <select class="form_input" id="access" multiple="multiple" name="access[]"><option>Read</option>
74
+ # # <option>Write</option></select>
75
+ #
76
+ # select_tag "destination", "<option>NYC</option><option>Paris</option><option>Rome</option>", :disabled => true
77
+ # # => <select disabled="disabled" id="destination" name="destination"><option>NYC</option>
78
+ # # <option>Paris</option><option>Rome</option></select>
79
+ def select_tag(name, option_tags = nil, options = {})
80
+ html_name = (options[:multiple] == true && !name.to_s.ends_with?("[]")) ? "#{name}[]" : name
81
+ content_tag :select, option_tags, { "name" => html_name, "id" => sanitize_to_id(name) }.update(options.stringify_keys)
82
+ end
83
+
84
+ # Creates a standard text field; use these text fields to input smaller chunks of text like a username
85
+ # or a search query.
86
+ #
87
+ # ==== Options
88
+ # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
89
+ # * <tt>:size</tt> - The number of visible characters that will fit in the input.
90
+ # * <tt>:maxlength</tt> - The maximum number of characters that the browser will allow the user to enter.
91
+ # * Any other key creates standard HTML attributes for the tag.
92
+ #
93
+ # ==== Examples
94
+ # text_field_tag 'name'
95
+ # # => <input id="name" name="name" type="text" />
96
+ #
97
+ # text_field_tag 'query', 'Enter your search query here'
98
+ # # => <input id="query" name="query" type="text" value="Enter your search query here" />
99
+ #
100
+ # text_field_tag 'request', nil, :class => 'special_input'
101
+ # # => <input class="special_input" id="request" name="request" type="text" />
102
+ #
103
+ # text_field_tag 'address', '', :size => 75
104
+ # # => <input id="address" name="address" size="75" type="text" value="" />
105
+ #
106
+ # text_field_tag 'zip', nil, :maxlength => 5
107
+ # # => <input id="zip" maxlength="5" name="zip" type="text" />
108
+ #
109
+ # text_field_tag 'payment_amount', '$0.00', :disabled => true
110
+ # # => <input disabled="disabled" id="payment_amount" name="payment_amount" type="text" value="$0.00" />
111
+ #
112
+ # text_field_tag 'ip', '0.0.0.0', :maxlength => 15, :size => 20, :class => "ip-input"
113
+ # # => <input class="ip-input" id="ip" maxlength="15" name="ip" size="20" type="text" value="0.0.0.0" />
114
+ def text_field_tag(name, value = nil, options = {})
115
+ tag :input, { "type" => "text", "name" => name, "id" => sanitize_to_id(name), "value" => value }.update(options.stringify_keys)
116
+ end
117
+
118
+ # Creates a label field
119
+ #
120
+ # ==== Options
121
+ # * Creates standard HTML attributes for the tag.
122
+ #
123
+ # ==== Examples
124
+ # label_tag 'name'
125
+ # # => <label for="name">Name</label>
126
+ #
127
+ # label_tag 'name', 'Your name'
128
+ # # => <label for="name">Your Name</label>
129
+ #
130
+ # label_tag 'name', nil, :class => 'small_label'
131
+ # # => <label for="name" class="small_label">Name</label>
132
+ def label_tag(name, text = nil, options = {})
133
+ content_tag :label, text || name.to_s.humanize, { "for" => sanitize_to_id(name) }.update(options.stringify_keys)
134
+ end
135
+
136
+ # Creates a hidden form input field used to transmit data that would be lost due to HTTP's statelessness or
137
+ # data that should be hidden from the user.
138
+ #
139
+ # ==== Options
140
+ # * Creates standard HTML attributes for the tag.
141
+ #
142
+ # ==== Examples
143
+ # hidden_field_tag 'tags_list'
144
+ # # => <input id="tags_list" name="tags_list" type="hidden" />
145
+ #
146
+ # hidden_field_tag 'token', 'VUBJKB23UIVI1UU1VOBVI@'
147
+ # # => <input id="token" name="token" type="hidden" value="VUBJKB23UIVI1UU1VOBVI@" />
148
+ #
149
+ # hidden_field_tag 'collected_input', '', :onchange => "alert('Input collected!')"
150
+ # # => <input id="collected_input" name="collected_input" onchange="alert('Input collected!')"
151
+ # # type="hidden" value="" />
152
+ def hidden_field_tag(name, value = nil, options = {})
153
+ text_field_tag(name, value, options.stringify_keys.update("type" => "hidden"))
154
+ end
155
+
156
+ # Creates a file upload field. If you are using file uploads then you will also need
157
+ # to set the multipart option for the form tag:
158
+ #
159
+ # <% form_tag '/upload', :multipart => true do %>
160
+ # <label for="file">File to Upload</label> <%= file_field_tag "file" %>
161
+ # <%= submit_tag %>
162
+ # <% end %>
163
+ #
164
+ # The specified URL will then be passed a File object containing the selected file, or if the field
165
+ # was left blank, a StringIO object.
166
+ #
167
+ # ==== Options
168
+ # * Creates standard HTML attributes for the tag.
169
+ # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
170
+ #
171
+ # ==== Examples
172
+ # file_field_tag 'attachment'
173
+ # # => <input id="attachment" name="attachment" type="file" />
174
+ #
175
+ # file_field_tag 'avatar', :class => 'profile-input'
176
+ # # => <input class="profile-input" id="avatar" name="avatar" type="file" />
177
+ #
178
+ # file_field_tag 'picture', :disabled => true
179
+ # # => <input disabled="disabled" id="picture" name="picture" type="file" />
180
+ #
181
+ # file_field_tag 'resume', :value => '~/resume.doc'
182
+ # # => <input id="resume" name="resume" type="file" value="~/resume.doc" />
183
+ #
184
+ # file_field_tag 'user_pic', :accept => 'image/png,image/gif,image/jpeg'
185
+ # # => <input accept="image/png,image/gif,image/jpeg" id="user_pic" name="user_pic" type="file" />
186
+ #
187
+ # file_field_tag 'file', :accept => 'text/html', :class => 'upload', :value => 'index.html'
188
+ # # => <input accept="text/html" class="upload" id="file" name="file" type="file" value="index.html" />
189
+ def file_field_tag(name, options = {})
190
+ text_field_tag(name, nil, options.update("type" => "file"))
191
+ end
192
+
193
+ # Creates a password field, a masked text field that will hide the users input behind a mask character.
194
+ #
195
+ # ==== Options
196
+ # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
197
+ # * <tt>:size</tt> - The number of visible characters that will fit in the input.
198
+ # * <tt>:maxlength</tt> - The maximum number of characters that the browser will allow the user to enter.
199
+ # * Any other key creates standard HTML attributes for the tag.
200
+ #
201
+ # ==== Examples
202
+ # password_field_tag 'pass'
203
+ # # => <input id="pass" name="pass" type="password" />
204
+ #
205
+ # password_field_tag 'secret', 'Your secret here'
206
+ # # => <input id="secret" name="secret" type="password" value="Your secret here" />
207
+ #
208
+ # password_field_tag 'masked', nil, :class => 'masked_input_field'
209
+ # # => <input class="masked_input_field" id="masked" name="masked" type="password" />
210
+ #
211
+ # password_field_tag 'token', '', :size => 15
212
+ # # => <input id="token" name="token" size="15" type="password" value="" />
213
+ #
214
+ # password_field_tag 'key', nil, :maxlength => 16
215
+ # # => <input id="key" maxlength="16" name="key" type="password" />
216
+ #
217
+ # password_field_tag 'confirm_pass', nil, :disabled => true
218
+ # # => <input disabled="disabled" id="confirm_pass" name="confirm_pass" type="password" />
219
+ #
220
+ # password_field_tag 'pin', '1234', :maxlength => 4, :size => 6, :class => "pin-input"
221
+ # # => <input class="pin-input" id="pin" maxlength="4" name="pin" size="6" type="password" value="1234" />
222
+ def password_field_tag(name = "password", value = nil, options = {})
223
+ text_field_tag(name, value, options.update("type" => "password"))
224
+ end
225
+
226
+ # Creates a text input area; use a textarea for longer text inputs such as blog posts or descriptions.
227
+ #
228
+ # ==== Options
229
+ # * <tt>:size</tt> - A string specifying the dimensions (columns by rows) of the textarea (e.g., "25x10").
230
+ # * <tt>:rows</tt> - Specify the number of rows in the textarea
231
+ # * <tt>:cols</tt> - Specify the number of columns in the textarea
232
+ # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
233
+ # * <tt>:escape</tt> - By default, the contents of the text input are HTML escaped.
234
+ # If you need unescaped contents, set this to false.
235
+ # * Any other key creates standard HTML attributes for the tag.
236
+ #
237
+ # ==== Examples
238
+ # text_area_tag 'post'
239
+ # # => <textarea id="post" name="post"></textarea>
240
+ #
241
+ # text_area_tag 'bio', @user.bio
242
+ # # => <textarea id="bio" name="bio">This is my biography.</textarea>
243
+ #
244
+ # text_area_tag 'body', nil, :rows => 10, :cols => 25
245
+ # # => <textarea cols="25" id="body" name="body" rows="10"></textarea>
246
+ #
247
+ # text_area_tag 'body', nil, :size => "25x10"
248
+ # # => <textarea name="body" id="body" cols="25" rows="10"></textarea>
249
+ #
250
+ # text_area_tag 'description', "Description goes here.", :disabled => true
251
+ # # => <textarea disabled="disabled" id="description" name="description">Description goes here.</textarea>
252
+ #
253
+ # text_area_tag 'comment', nil, :class => 'comment_input'
254
+ # # => <textarea class="comment_input" id="comment" name="comment"></textarea>
255
+ def text_area_tag(name, content = nil, options = {})
256
+ options.stringify_keys!
257
+
258
+ if size = options.delete("size")
259
+ options["cols"], options["rows"] = size.split("x") if size.respond_to?(:split)
260
+ end
261
+
262
+ escape = options.key?("escape") ? options.delete("escape") : true
263
+ content = html_escape(content) if escape
264
+
265
+ content_tag :textarea, content, { "name" => name, "id" => sanitize_to_id(name) }.update(options.stringify_keys)
266
+ end
267
+
268
+ # Creates a check box form input tag.
269
+ #
270
+ # ==== Options
271
+ # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
272
+ # * Any other key creates standard HTML options for the tag.
273
+ #
274
+ # ==== Examples
275
+ # check_box_tag 'accept'
276
+ # # => <input id="accept" name="accept" type="checkbox" value="1" />
277
+ #
278
+ # check_box_tag 'rock', 'rock music'
279
+ # # => <input id="rock" name="rock" type="checkbox" value="rock music" />
280
+ #
281
+ # check_box_tag 'receive_email', 'yes', true
282
+ # # => <input checked="checked" id="receive_email" name="receive_email" type="checkbox" value="yes" />
283
+ #
284
+ # check_box_tag 'tos', 'yes', false, :class => 'accept_tos'
285
+ # # => <input class="accept_tos" id="tos" name="tos" type="checkbox" value="yes" />
286
+ #
287
+ # check_box_tag 'eula', 'accepted', false, :disabled => true
288
+ # # => <input disabled="disabled" id="eula" name="eula" type="checkbox" value="accepted" />
289
+ def check_box_tag(name, value = "1", checked = false, options = {})
290
+ html_options = { "type" => "checkbox", "name" => name, "id" => sanitize_to_id(name), "value" => value }.update(options.stringify_keys)
291
+ html_options["checked"] = "checked" if checked
292
+ tag :input, html_options
293
+ end
294
+
295
+ # Creates a radio button; use groups of radio buttons named the same to allow users to
296
+ # select from a group of options.
297
+ #
298
+ # ==== Options
299
+ # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
300
+ # * Any other key creates standard HTML options for the tag.
301
+ #
302
+ # ==== Examples
303
+ # radio_button_tag 'gender', 'male'
304
+ # # => <input id="gender_male" name="gender" type="radio" value="male" />
305
+ #
306
+ # radio_button_tag 'receive_updates', 'no', true
307
+ # # => <input checked="checked" id="receive_updates_no" name="receive_updates" type="radio" value="no" />
308
+ #
309
+ # radio_button_tag 'time_slot', "3:00 p.m.", false, :disabled => true
310
+ # # => <input disabled="disabled" id="time_slot_300_pm" name="time_slot" type="radio" value="3:00 p.m." />
311
+ #
312
+ # radio_button_tag 'color', "green", true, :class => "color_input"
313
+ # # => <input checked="checked" class="color_input" id="color_green" name="color" type="radio" value="green" />
314
+ def radio_button_tag(name, value, checked = false, options = {})
315
+ pretty_tag_value = value.to_s.gsub(/\s/, "_").gsub(/(?!-)\W/, "").downcase
316
+ pretty_name = name.to_s.gsub(/\[/, "_").gsub(/\]/, "")
317
+ html_options = { "type" => "radio", "name" => name, "id" => "#{pretty_name}_#{pretty_tag_value}", "value" => value }.update(options.stringify_keys)
318
+ html_options["checked"] = "checked" if checked
319
+ tag :input, html_options
320
+ end
321
+
322
+ # Creates a submit button with the text <tt>value</tt> as the caption.
323
+ #
324
+ # ==== Options
325
+ # * <tt>:confirm => 'question?'</tt> - This will add a JavaScript confirm
326
+ # prompt with the question specified. If the user accepts, the form is
327
+ # processed normally, otherwise no action is taken.
328
+ # * <tt>:disabled</tt> - If true, the user will not be able to use this input.
329
+ # * <tt>:disable_with</tt> - Value of this parameter will be used as the value for a disabled version
330
+ # of the submit button when the form is submitted.
331
+ # * Any other key creates standard HTML options for the tag.
332
+ #
333
+ # ==== Examples
334
+ # submit_tag
335
+ # # => <input name="commit" type="submit" value="Save changes" />
336
+ #
337
+ # submit_tag "Edit this article"
338
+ # # => <input name="commit" type="submit" value="Edit this article" />
339
+ #
340
+ # submit_tag "Save edits", :disabled => true
341
+ # # => <input disabled="disabled" name="commit" type="submit" value="Save edits" />
342
+ #
343
+ # submit_tag "Complete sale", :disable_with => "Please wait..."
344
+ # # => <input name="commit" onclick="this.disabled=true;this.value='Please wait...';this.form.submit();"
345
+ # # type="submit" value="Complete sale" />
346
+ #
347
+ # submit_tag nil, :class => "form_submit"
348
+ # # => <input class="form_submit" name="commit" type="submit" />
349
+ #
350
+ # submit_tag "Edit", :disable_with => "Editing...", :class => "edit-button"
351
+ # # => <input class="edit-button" onclick="this.disabled=true;this.value='Editing...';this.form.submit();"
352
+ # # name="commit" type="submit" value="Edit" />
353
+ def submit_tag(value = "Save changes", options = {})
354
+ options.stringify_keys!
355
+
356
+ if disable_with = options.delete("disable_with")
357
+ disable_with = "this.value='#{disable_with}'"
358
+ disable_with << ";#{options.delete('onclick')}" if options['onclick']
359
+
360
+ options["onclick"] = "if (window.hiddenCommit) { window.hiddenCommit.setAttribute('value', this.value); }"
361
+ options["onclick"] << "else { hiddenCommit = document.createElement('input');hiddenCommit.type = 'hidden';"
362
+ options["onclick"] << "hiddenCommit.value = this.value;hiddenCommit.name = this.name;this.form.appendChild(hiddenCommit); }"
363
+ options["onclick"] << "this.setAttribute('originalValue', this.value);this.disabled = true;#{disable_with};"
364
+ options["onclick"] << "result = (this.form.onsubmit ? (this.form.onsubmit() ? this.form.submit() : false) : this.form.submit());"
365
+ options["onclick"] << "if (result == false) { this.value = this.getAttribute('originalValue');this.disabled = false; }return result;"
366
+ end
367
+
368
+ if confirm = options.delete("confirm")
369
+ options["onclick"] ||= 'return true;'
370
+ options["onclick"] = "if (!#{confirm_javascript_function(confirm)}) return false; #{options['onclick']}"
371
+ end
372
+
373
+ tag :input, { "type" => "submit", "name" => "commit", "value" => value }.update(options.stringify_keys)
374
+ end
375
+
376
+ # Displays an image which when clicked will submit the form.
377
+ #
378
+ # <tt>source</tt> is passed to AssetTagHelper#image_path
379
+ #
380
+ # ==== Options
381
+ # * <tt>:confirm => 'question?'</tt> - This will add a JavaScript confirm
382
+ # prompt with the question specified. If the user accepts, the form is
383
+ # processed normally, otherwise no action is taken.
384
+ # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
385
+ # * Any other key creates standard HTML options for the tag.
386
+ #
387
+ # ==== Examples
388
+ # image_submit_tag("login.png")
389
+ # # => <input src="/images/login.png" type="image" />
390
+ #
391
+ # image_submit_tag("purchase.png", :disabled => true)
392
+ # # => <input disabled="disabled" src="/images/purchase.png" type="image" />
393
+ #
394
+ # image_submit_tag("search.png", :class => 'search-button')
395
+ # # => <input class="search-button" src="/images/search.png" type="image" />
396
+ #
397
+ # image_submit_tag("agree.png", :disabled => true, :class => "agree-disagree-button")
398
+ # # => <input class="agree-disagree-button" disabled="disabled" src="/images/agree.png" type="image" />
399
+ def image_submit_tag(source, options = {})
400
+ options.stringify_keys!
401
+
402
+ if confirm = options.delete("confirm")
403
+ options["onclick"] ||= ''
404
+ options["onclick"] += "return #{confirm_javascript_function(confirm)};"
405
+ end
406
+
407
+ tag :input, { "type" => "image", "src" => path_to_image(source) }.update(options.stringify_keys)
408
+ end
409
+
410
+ # Creates a field set for grouping HTML form elements.
411
+ #
412
+ # <tt>legend</tt> will become the fieldset's title (optional as per W3C).
413
+ # <tt>options</tt> accept the same values as tag.
414
+ #
415
+ # === Examples
416
+ # <% field_set_tag do %>
417
+ # <p><%= text_field_tag 'name' %></p>
418
+ # <% end %>
419
+ # # => <fieldset><p><input id="name" name="name" type="text" /></p></fieldset>
420
+ #
421
+ # <% field_set_tag 'Your details' do %>
422
+ # <p><%= text_field_tag 'name' %></p>
423
+ # <% end %>
424
+ # # => <fieldset><legend>Your details</legend><p><input id="name" name="name" type="text" /></p></fieldset>
425
+ #
426
+ # <% field_set_tag nil, :class => 'format' do %>
427
+ # <p><%= text_field_tag 'name' %></p>
428
+ # <% end %>
429
+ # # => <fieldset class="format"><p><input id="name" name="name" type="text" /></p></fieldset>
430
+ def field_set_tag(legend = nil, options = nil, &block)
431
+ content = capture(&block)
432
+ concat(tag(:fieldset, options, true))
433
+ concat(content_tag(:legend, legend)) unless legend.blank?
434
+ concat(content)
435
+ concat("</fieldset>".html_safe!)
436
+ end
437
+
438
+ private
439
+ def html_options_for_form(url_for_options, options, *parameters_for_url)
440
+ returning options.stringify_keys do |html_options|
441
+ html_options["enctype"] = "multipart/form-data" if html_options.delete("multipart")
442
+ html_options["action"] = url_for(url_for_options, *parameters_for_url)
443
+ end
444
+ end
445
+
446
+ def extra_tags_for_form(html_options)
447
+ case method = html_options.delete("method").to_s
448
+ when /^get$/i # must be case-insentive, but can't use downcase as might be nil
449
+ html_options["method"] = "get"
450
+ ''
451
+ when /^post$/i, "", nil
452
+ html_options["method"] = "post"
453
+ protect_against_forgery? ? content_tag(:div, token_tag, :style => 'margin:0;padding:0;display:inline') : ''
454
+ else
455
+ html_options["method"] = "post"
456
+ content_tag(:div, tag(:input, :type => "hidden", :name => "_method", :value => method) + token_tag, :style => 'margin:0;padding:0;display:inline')
457
+ end
458
+ end
459
+
460
+ def form_tag_html(html_options)
461
+ extra_tags = extra_tags_for_form(html_options)
462
+ (tag(:form, html_options, true) + extra_tags).html_safe!
463
+ end
464
+
465
+ def form_tag_in_block(html_options, &block)
466
+ content = capture(&block)
467
+ concat(form_tag_html(html_options))
468
+ concat(content)
469
+ concat("</form>".html_safe!)
470
+ end
471
+
472
+ def token_tag
473
+ unless protect_against_forgery?
474
+ ''
475
+ else
476
+ tag(:input, :type => "hidden", :name => request_forgery_protection_token.to_s, :value => form_authenticity_token)
477
+ end
478
+ end
479
+
480
+ # see http://www.w3.org/TR/html4/types.html#type-name
481
+ def sanitize_to_id(name)
482
+ name.to_s.gsub(']','').gsub(/[^-a-zA-Z0-9:.]/, "_")
483
+ end
484
+
485
+ end
486
+ end
487
+ end