actionpack-rack-upgrade 2.3.14
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.
- data/CHANGELOG +5250 -0
- data/MIT-LICENSE +21 -0
- data/README +409 -0
- data/RUNNING_UNIT_TESTS +24 -0
- data/Rakefile +158 -0
- data/install.rb +30 -0
- data/lib/action_controller/assertions/dom_assertions.rb +55 -0
- data/lib/action_controller/assertions/model_assertions.rb +21 -0
- data/lib/action_controller/assertions/response_assertions.rb +169 -0
- data/lib/action_controller/assertions/routing_assertions.rb +146 -0
- data/lib/action_controller/assertions/selector_assertions.rb +638 -0
- data/lib/action_controller/assertions/tag_assertions.rb +127 -0
- data/lib/action_controller/base.rb +1425 -0
- data/lib/action_controller/benchmarking.rb +107 -0
- data/lib/action_controller/caching/actions.rb +177 -0
- data/lib/action_controller/caching/fragments.rb +120 -0
- data/lib/action_controller/caching/pages.rb +152 -0
- data/lib/action_controller/caching/sweeper.rb +45 -0
- data/lib/action_controller/caching/sweeping.rb +55 -0
- data/lib/action_controller/caching.rb +71 -0
- data/lib/action_controller/cgi_ext/cookie.rb +112 -0
- data/lib/action_controller/cgi_ext/query_extension.rb +22 -0
- data/lib/action_controller/cgi_ext/stdinput.rb +24 -0
- data/lib/action_controller/cgi_ext.rb +15 -0
- data/lib/action_controller/cgi_process.rb +77 -0
- data/lib/action_controller/cookies.rb +197 -0
- data/lib/action_controller/dispatcher.rb +133 -0
- data/lib/action_controller/failsafe.rb +87 -0
- data/lib/action_controller/filters.rb +680 -0
- data/lib/action_controller/flash.rb +213 -0
- data/lib/action_controller/headers.rb +33 -0
- data/lib/action_controller/helpers.rb +225 -0
- data/lib/action_controller/http_authentication.rb +309 -0
- data/lib/action_controller/integration.rb +708 -0
- data/lib/action_controller/layout.rb +286 -0
- data/lib/action_controller/middleware_stack.rb +119 -0
- data/lib/action_controller/middlewares.rb +14 -0
- data/lib/action_controller/mime_responds.rb +193 -0
- data/lib/action_controller/mime_type.rb +212 -0
- data/lib/action_controller/mime_types.rb +21 -0
- data/lib/action_controller/params_parser.rb +77 -0
- data/lib/action_controller/performance_test.rb +15 -0
- data/lib/action_controller/polymorphic_routes.rb +189 -0
- data/lib/action_controller/rack_lint_patch.rb +36 -0
- data/lib/action_controller/record_identifier.rb +104 -0
- data/lib/action_controller/reloader.rb +54 -0
- data/lib/action_controller/request.rb +495 -0
- data/lib/action_controller/request_forgery_protection.rb +116 -0
- data/lib/action_controller/rescue.rb +183 -0
- data/lib/action_controller/resources.rb +682 -0
- data/lib/action_controller/response.rb +237 -0
- data/lib/action_controller/routing/builder.rb +197 -0
- data/lib/action_controller/routing/optimisations.rb +130 -0
- data/lib/action_controller/routing/recognition_optimisation.rb +167 -0
- data/lib/action_controller/routing/route.rb +265 -0
- data/lib/action_controller/routing/route_set.rb +503 -0
- data/lib/action_controller/routing/routing_ext.rb +49 -0
- data/lib/action_controller/routing/segments.rb +343 -0
- data/lib/action_controller/routing.rb +388 -0
- data/lib/action_controller/session/abstract_store.rb +276 -0
- data/lib/action_controller/session/cookie_store.rb +240 -0
- data/lib/action_controller/session/mem_cache_store.rb +60 -0
- data/lib/action_controller/session_management.rb +54 -0
- data/lib/action_controller/status_codes.rb +88 -0
- data/lib/action_controller/streaming.rb +181 -0
- data/lib/action_controller/string_coercion.rb +29 -0
- data/lib/action_controller/templates/rescues/_request_and_response.erb +24 -0
- data/lib/action_controller/templates/rescues/_trace.erb +26 -0
- data/lib/action_controller/templates/rescues/diagnostics.erb +11 -0
- data/lib/action_controller/templates/rescues/layout.erb +29 -0
- data/lib/action_controller/templates/rescues/missing_template.erb +2 -0
- data/lib/action_controller/templates/rescues/routing_error.erb +10 -0
- data/lib/action_controller/templates/rescues/template_error.erb +21 -0
- data/lib/action_controller/templates/rescues/unknown_action.erb +2 -0
- data/lib/action_controller/test_case.rb +209 -0
- data/lib/action_controller/test_process.rb +580 -0
- data/lib/action_controller/translation.rb +13 -0
- data/lib/action_controller/uploaded_file.rb +44 -0
- data/lib/action_controller/url_rewriter.rb +229 -0
- data/lib/action_controller/vendor/html-scanner/html/document.rb +68 -0
- data/lib/action_controller/vendor/html-scanner/html/node.rb +537 -0
- data/lib/action_controller/vendor/html-scanner/html/sanitizer.rb +173 -0
- data/lib/action_controller/vendor/html-scanner/html/selector.rb +828 -0
- data/lib/action_controller/vendor/html-scanner/html/tokenizer.rb +105 -0
- data/lib/action_controller/vendor/html-scanner/html/version.rb +11 -0
- data/lib/action_controller/vendor/html-scanner.rb +16 -0
- data/lib/action_controller/verification.rb +130 -0
- data/lib/action_controller.rb +113 -0
- data/lib/action_pack/version.rb +9 -0
- data/lib/action_pack.rb +24 -0
- data/lib/action_view/base.rb +362 -0
- data/lib/action_view/helpers/active_record_helper.rb +305 -0
- data/lib/action_view/helpers/asset_tag_helper.rb +695 -0
- data/lib/action_view/helpers/atom_feed_helper.rb +198 -0
- data/lib/action_view/helpers/benchmark_helper.rb +54 -0
- data/lib/action_view/helpers/cache_helper.rb +39 -0
- data/lib/action_view/helpers/capture_helper.rb +136 -0
- data/lib/action_view/helpers/csrf_helper.rb +14 -0
- data/lib/action_view/helpers/date_helper.rb +989 -0
- data/lib/action_view/helpers/debug_helper.rb +38 -0
- data/lib/action_view/helpers/form_helper.rb +1118 -0
- data/lib/action_view/helpers/form_options_helper.rb +599 -0
- data/lib/action_view/helpers/form_tag_helper.rb +490 -0
- data/lib/action_view/helpers/javascript_helper.rb +208 -0
- data/lib/action_view/helpers/number_helper.rb +308 -0
- data/lib/action_view/helpers/prototype_helper.rb +1305 -0
- data/lib/action_view/helpers/raw_output_helper.rb +9 -0
- data/lib/action_view/helpers/record_identification_helper.rb +20 -0
- data/lib/action_view/helpers/record_tag_helper.rb +58 -0
- data/lib/action_view/helpers/sanitize_helper.rb +251 -0
- data/lib/action_view/helpers/scriptaculous_helper.rb +226 -0
- data/lib/action_view/helpers/tag_helper.rb +151 -0
- data/lib/action_view/helpers/text_helper.rb +597 -0
- data/lib/action_view/helpers/translation_helper.rb +67 -0
- data/lib/action_view/helpers/url_helper.rb +637 -0
- data/lib/action_view/helpers.rb +61 -0
- data/lib/action_view/inline_template.rb +19 -0
- data/lib/action_view/locale/en.yml +117 -0
- data/lib/action_view/partials.rb +241 -0
- data/lib/action_view/paths.rb +77 -0
- data/lib/action_view/reloadable_template.rb +117 -0
- data/lib/action_view/renderable.rb +109 -0
- data/lib/action_view/renderable_partial.rb +53 -0
- data/lib/action_view/template.rb +252 -0
- data/lib/action_view/template_error.rb +99 -0
- data/lib/action_view/template_handler.rb +34 -0
- data/lib/action_view/template_handlers/builder.rb +17 -0
- data/lib/action_view/template_handlers/erb.rb +25 -0
- data/lib/action_view/template_handlers/rjs.rb +13 -0
- data/lib/action_view/template_handlers.rb +48 -0
- data/lib/action_view/test_case.rb +162 -0
- data/lib/action_view.rb +58 -0
- data/lib/actionpack.rb +2 -0
- data/test/abstract_unit.rb +78 -0
- data/test/active_record_unit.rb +104 -0
- data/test/activerecord/active_record_store_test.rb +221 -0
- data/test/activerecord/render_partial_with_record_identification_test.rb +188 -0
- data/test/adv_attr_test.rb +20 -0
- data/test/controller/action_pack_assertions_test.rb +545 -0
- data/test/controller/addresses_render_test.rb +37 -0
- data/test/controller/assert_select_test.rb +735 -0
- data/test/controller/base_test.rb +217 -0
- data/test/controller/benchmark_test.rb +32 -0
- data/test/controller/caching_test.rb +743 -0
- data/test/controller/capture_test.rb +66 -0
- data/test/controller/content_type_test.rb +178 -0
- data/test/controller/controller_fixtures/app/controllers/admin/user_controller.rb +0 -0
- data/test/controller/controller_fixtures/app/controllers/user_controller.rb +0 -0
- data/test/controller/controller_fixtures/vendor/plugins/bad_plugin/lib/plugin_controller.rb +0 -0
- data/test/controller/cookie_test.rb +208 -0
- data/test/controller/deprecation/deprecated_base_methods_test.rb +32 -0
- data/test/controller/dispatcher_test.rb +144 -0
- data/test/controller/dom_assertions_test.rb +53 -0
- data/test/controller/failsafe_test.rb +60 -0
- data/test/controller/fake_controllers.rb +33 -0
- data/test/controller/fake_models.rb +19 -0
- data/test/controller/filter_params_test.rb +52 -0
- data/test/controller/filters_test.rb +885 -0
- data/test/controller/flash_test.rb +174 -0
- data/test/controller/header_test.rb +14 -0
- data/test/controller/helper_test.rb +224 -0
- data/test/controller/html-scanner/cdata_node_test.rb +15 -0
- data/test/controller/html-scanner/document_test.rb +148 -0
- data/test/controller/html-scanner/node_test.rb +89 -0
- data/test/controller/html-scanner/sanitizer_test.rb +281 -0
- data/test/controller/html-scanner/tag_node_test.rb +238 -0
- data/test/controller/html-scanner/text_node_test.rb +50 -0
- data/test/controller/html-scanner/tokenizer_test.rb +131 -0
- data/test/controller/http_basic_authentication_test.rb +113 -0
- data/test/controller/http_digest_authentication_test.rb +254 -0
- data/test/controller/integration_test.rb +526 -0
- data/test/controller/layout_test.rb +215 -0
- data/test/controller/localized_templates_test.rb +24 -0
- data/test/controller/logging_test.rb +46 -0
- data/test/controller/middleware_stack_test.rb +90 -0
- data/test/controller/mime_responds_test.rb +536 -0
- data/test/controller/mime_type_test.rb +93 -0
- data/test/controller/output_escaping_test.rb +19 -0
- data/test/controller/polymorphic_routes_test.rb +297 -0
- data/test/controller/rack_test.rb +308 -0
- data/test/controller/record_identifier_test.rb +139 -0
- data/test/controller/redirect_test.rb +285 -0
- data/test/controller/reloader_test.rb +125 -0
- data/test/controller/render_test.rb +1783 -0
- data/test/controller/request/json_params_parsing_test.rb +65 -0
- data/test/controller/request/multipart_params_parsing_test.rb +177 -0
- data/test/controller/request/query_string_parsing_test.rb +120 -0
- data/test/controller/request/test_request_test.rb +35 -0
- data/test/controller/request/url_encoded_params_parsing_test.rb +146 -0
- data/test/controller/request/xml_params_parsing_test.rb +103 -0
- data/test/controller/request_forgery_protection_test.rb +233 -0
- data/test/controller/request_test.rb +395 -0
- data/test/controller/rescue_test.rb +541 -0
- data/test/controller/resources_test.rb +1393 -0
- data/test/controller/routing_test.rb +2592 -0
- data/test/controller/selector_test.rb +628 -0
- data/test/controller/send_file_test.rb +171 -0
- data/test/controller/session/abstract_store_test.rb +64 -0
- data/test/controller/session/cookie_store_test.rb +354 -0
- data/test/controller/session/mem_cache_store_test.rb +187 -0
- data/test/controller/session/test_session_test.rb +58 -0
- data/test/controller/test_test.rb +700 -0
- data/test/controller/translation_test.rb +26 -0
- data/test/controller/url_rewriter_test.rb +395 -0
- data/test/controller/verification_test.rb +270 -0
- data/test/controller/view_paths_test.rb +141 -0
- data/test/controller/webservice_test.rb +260 -0
- data/test/fixtures/_top_level_partial.html.erb +1 -0
- data/test/fixtures/_top_level_partial_only.erb +1 -0
- data/test/fixtures/addresses/list.erb +1 -0
- data/test/fixtures/alternate_helpers/foo_helper.rb +3 -0
- data/test/fixtures/bad_customers/_bad_customer.html.erb +1 -0
- data/test/fixtures/companies.yml +24 -0
- data/test/fixtures/company.rb +10 -0
- data/test/fixtures/content_type/render_default_content_types_for_respond_to.rhtml +1 -0
- data/test/fixtures/content_type/render_default_for_rhtml.rhtml +1 -0
- data/test/fixtures/content_type/render_default_for_rjs.rjs +1 -0
- data/test/fixtures/content_type/render_default_for_rxml.rxml +1 -0
- data/test/fixtures/customers/_customer.html.erb +1 -0
- data/test/fixtures/db_definitions/sqlite.sql +49 -0
- data/test/fixtures/developer.rb +9 -0
- data/test/fixtures/developers/_developer.erb +1 -0
- data/test/fixtures/developers.yml +21 -0
- data/test/fixtures/developers_projects.yml +13 -0
- data/test/fixtures/failsafe/500.html +1 -0
- data/test/fixtures/fun/games/_game.erb +1 -0
- data/test/fixtures/fun/games/hello_world.erb +1 -0
- data/test/fixtures/fun/serious/games/_game.erb +1 -0
- data/test/fixtures/functional_caching/_partial.erb +3 -0
- data/test/fixtures/functional_caching/formatted_fragment_cached.html.erb +3 -0
- data/test/fixtures/functional_caching/formatted_fragment_cached.js.rjs +6 -0
- data/test/fixtures/functional_caching/formatted_fragment_cached.xml.builder +5 -0
- data/test/fixtures/functional_caching/fragment_cached.html.erb +2 -0
- data/test/fixtures/functional_caching/html_fragment_cached_with_partial.html.erb +1 -0
- data/test/fixtures/functional_caching/inline_fragment_cached.html.erb +2 -0
- data/test/fixtures/functional_caching/js_fragment_cached_with_partial.js.rjs +1 -0
- data/test/fixtures/good_customers/_good_customer.html.erb +1 -0
- data/test/fixtures/helpers/abc_helper.rb +5 -0
- data/test/fixtures/helpers/fun/games_helper.rb +3 -0
- data/test/fixtures/helpers/fun/pdf_helper.rb +3 -0
- data/test/fixtures/layout_tests/abs_path_layout.rhtml +1 -0
- data/test/fixtures/layout_tests/alt/hello.rhtml +1 -0
- data/test/fixtures/layout_tests/alt/layouts/alt.rhtml +0 -0
- data/test/fixtures/layout_tests/layouts/controller_name_space/nested.rhtml +1 -0
- data/test/fixtures/layout_tests/layouts/item.rhtml +1 -0
- data/test/fixtures/layout_tests/layouts/layout_test.rhtml +1 -0
- data/test/fixtures/layout_tests/layouts/multiple_extensions.html.erb +1 -0
- data/test/fixtures/layout_tests/layouts/third_party_template_library.mab +1 -0
- data/test/fixtures/layout_tests/views/hello.rhtml +1 -0
- data/test/fixtures/layouts/_column.html.erb +2 -0
- data/test/fixtures/layouts/block_with_layout.erb +3 -0
- data/test/fixtures/layouts/builder.builder +3 -0
- data/test/fixtures/layouts/default_html.html.erb +1 -0
- data/test/fixtures/layouts/partial_with_layout.erb +3 -0
- data/test/fixtures/layouts/standard.erb +1 -0
- data/test/fixtures/layouts/talk_from_action.erb +2 -0
- data/test/fixtures/layouts/xhr.html.erb +2 -0
- data/test/fixtures/layouts/yield.erb +2 -0
- data/test/fixtures/localized/hello_world.de.html +1 -0
- data/test/fixtures/localized/hello_world.en.html +1 -0
- data/test/fixtures/mascot.rb +3 -0
- data/test/fixtures/mascots/_mascot.html.erb +1 -0
- data/test/fixtures/mascots.yml +4 -0
- data/test/fixtures/multipart/binary_file +0 -0
- data/test/fixtures/multipart/boundary_problem_file +10 -0
- data/test/fixtures/multipart/bracketed_param +5 -0
- data/test/fixtures/multipart/empty +10 -0
- data/test/fixtures/multipart/hello.txt +1 -0
- data/test/fixtures/multipart/large_text_file +10 -0
- data/test/fixtures/multipart/mixed_files +0 -0
- data/test/fixtures/multipart/mona_lisa.jpg +0 -0
- data/test/fixtures/multipart/none +9 -0
- data/test/fixtures/multipart/single_parameter +5 -0
- data/test/fixtures/multipart/text_file +10 -0
- data/test/fixtures/override/test/hello_world.erb +1 -0
- data/test/fixtures/override2/layouts/test/sub.erb +1 -0
- data/test/fixtures/post_test/layouts/post.html.erb +1 -0
- data/test/fixtures/post_test/layouts/super_post.iphone.erb +1 -0
- data/test/fixtures/post_test/post/index.html.erb +1 -0
- data/test/fixtures/post_test/post/index.iphone.erb +1 -0
- data/test/fixtures/post_test/super_post/index.html.erb +1 -0
- data/test/fixtures/post_test/super_post/index.iphone.erb +1 -0
- data/test/fixtures/project.rb +3 -0
- data/test/fixtures/projects/_project.erb +1 -0
- data/test/fixtures/projects.yml +7 -0
- data/test/fixtures/public/404.html +1 -0
- data/test/fixtures/public/500.da.html +1 -0
- data/test/fixtures/public/500.html +1 -0
- data/test/fixtures/public/absolute/test.css +23 -0
- data/test/fixtures/public/absolute/test.js +63 -0
- data/test/fixtures/public/images/rails.png +0 -0
- data/test/fixtures/public/javascripts/application.js +1 -0
- data/test/fixtures/public/javascripts/bank.js +1 -0
- data/test/fixtures/public/javascripts/controls.js +1 -0
- data/test/fixtures/public/javascripts/dragdrop.js +1 -0
- data/test/fixtures/public/javascripts/effects.js +1 -0
- data/test/fixtures/public/javascripts/prototype.js +1 -0
- data/test/fixtures/public/javascripts/robber.js +1 -0
- data/test/fixtures/public/javascripts/subdir/subdir.js +1 -0
- data/test/fixtures/public/javascripts/version.1.0.js +1 -0
- data/test/fixtures/public/stylesheets/bank.css +1 -0
- data/test/fixtures/public/stylesheets/robber.css +1 -0
- data/test/fixtures/public/stylesheets/subdir/subdir.css +1 -0
- data/test/fixtures/public/stylesheets/version.1.0.css +1 -0
- data/test/fixtures/quiz/questions/_question.html.erb +1 -0
- data/test/fixtures/replies/_reply.erb +1 -0
- data/test/fixtures/replies.yml +15 -0
- data/test/fixtures/reply.rb +7 -0
- data/test/fixtures/respond_to/all_types_with_layout.html.erb +1 -0
- data/test/fixtures/respond_to/all_types_with_layout.js.rjs +1 -0
- data/test/fixtures/respond_to/custom_constant_handling_without_block.mobile.erb +1 -0
- data/test/fixtures/respond_to/iphone_with_html_response_type.html.erb +1 -0
- data/test/fixtures/respond_to/iphone_with_html_response_type.iphone.erb +1 -0
- data/test/fixtures/respond_to/layouts/missing.html.erb +1 -0
- data/test/fixtures/respond_to/layouts/standard.html.erb +1 -0
- data/test/fixtures/respond_to/layouts/standard.iphone.erb +1 -0
- data/test/fixtures/respond_to/using_defaults.html.erb +1 -0
- data/test/fixtures/respond_to/using_defaults.js.rjs +1 -0
- data/test/fixtures/respond_to/using_defaults.xml.builder +1 -0
- data/test/fixtures/respond_to/using_defaults_with_type_list.html.erb +1 -0
- data/test/fixtures/respond_to/using_defaults_with_type_list.js.rjs +1 -0
- data/test/fixtures/respond_to/using_defaults_with_type_list.xml.builder +1 -0
- data/test/fixtures/scope/test/modgreet.erb +1 -0
- data/test/fixtures/session_autoload_test/session_autoload_test/foo.rb +10 -0
- data/test/fixtures/shared.html.erb +1 -0
- data/test/fixtures/symlink_parent/symlinked_layout.erb +5 -0
- data/test/fixtures/test/_counter.html.erb +1 -0
- data/test/fixtures/test/_customer.erb +1 -0
- data/test/fixtures/test/_customer_counter.erb +1 -0
- data/test/fixtures/test/_customer_counter_with_as.erb +1 -0
- data/test/fixtures/test/_customer_greeting.erb +1 -0
- data/test/fixtures/test/_customer_with_var.erb +1 -0
- data/test/fixtures/test/_form.erb +1 -0
- data/test/fixtures/test/_from_helper.erb +1 -0
- data/test/fixtures/test/_hash_greeting.erb +1 -0
- data/test/fixtures/test/_hash_object.erb +2 -0
- data/test/fixtures/test/_hello.builder +1 -0
- data/test/fixtures/test/_labelling_form.erb +1 -0
- data/test/fixtures/test/_layout_for_block_with_args.html.erb +3 -0
- data/test/fixtures/test/_layout_for_partial.html.erb +3 -0
- data/test/fixtures/test/_local_inspector.html.erb +1 -0
- data/test/fixtures/test/_one.html.erb +1 -0
- data/test/fixtures/test/_partial.erb +1 -0
- data/test/fixtures/test/_partial.html.erb +1 -0
- data/test/fixtures/test/_partial.js.erb +1 -0
- data/test/fixtures/test/_partial_for_use_in_layout.html.erb +1 -0
- data/test/fixtures/test/_partial_only.erb +1 -0
- data/test/fixtures/test/_partial_with_only_html_version.html.erb +1 -0
- data/test/fixtures/test/_person.erb +2 -0
- data/test/fixtures/test/_raise.html.erb +1 -0
- data/test/fixtures/test/_two.html.erb +1 -0
- data/test/fixtures/test/_utf8_partial.html.erb +1 -0
- data/test/fixtures/test/_utf8_partial_magic.html.erb +2 -0
- data/test/fixtures/test/action_talk_to_layout.erb +2 -0
- data/test/fixtures/test/array_translation.erb +1 -0
- data/test/fixtures/test/calling_partial_with_layout.html.erb +1 -0
- data/test/fixtures/test/capturing.erb +4 -0
- data/test/fixtures/test/content_for.erb +2 -0
- data/test/fixtures/test/content_for_concatenated.erb +3 -0
- data/test/fixtures/test/content_for_with_parameter.erb +2 -0
- data/test/fixtures/test/delete_with_js.rjs +2 -0
- data/test/fixtures/test/dont_pick_me +1 -0
- data/test/fixtures/test/dot.directory/render_file_with_ivar.erb +1 -0
- data/test/fixtures/test/enum_rjs_test.rjs +6 -0
- data/test/fixtures/test/formatted_html_erb.html.erb +1 -0
- data/test/fixtures/test/formatted_xml_erb.builder +1 -0
- data/test/fixtures/test/formatted_xml_erb.html.erb +1 -0
- data/test/fixtures/test/formatted_xml_erb.xml.erb +1 -0
- data/test/fixtures/test/greeting.erb +1 -0
- data/test/fixtures/test/greeting.js.rjs +1 -0
- data/test/fixtures/test/hello.builder +4 -0
- data/test/fixtures/test/hello_world.da.html.erb +1 -0
- data/test/fixtures/test/hello_world.erb +1 -0
- data/test/fixtures/test/hello_world.erb~ +1 -0
- data/test/fixtures/test/hello_world.pt-BR.html.erb +1 -0
- data/test/fixtures/test/hello_world_container.builder +3 -0
- data/test/fixtures/test/hello_world_from_rxml.builder +4 -0
- data/test/fixtures/test/hello_world_with_layout_false.erb +1 -0
- data/test/fixtures/test/hello_xml_world.builder +11 -0
- data/test/fixtures/test/hyphen-ated.erb +1 -0
- data/test/fixtures/test/implicit_content_type.atom.builder +2 -0
- data/test/fixtures/test/list.erb +1 -0
- data/test/fixtures/test/malformed/malformed.en.html.erb~ +1 -0
- data/test/fixtures/test/malformed/malformed.erb~ +1 -0
- data/test/fixtures/test/malformed/malformed.html.erb~ +1 -0
- data/test/fixtures/test/nested_layout.erb +3 -0
- data/test/fixtures/test/non_erb_block_content_for.builder +4 -0
- data/test/fixtures/test/potential_conflicts.erb +4 -0
- data/test/fixtures/test/render_explicit_html_template.js.rjs +1 -0
- data/test/fixtures/test/render_file_from_template.html.erb +1 -0
- data/test/fixtures/test/render_file_with_ivar.erb +1 -0
- data/test/fixtures/test/render_file_with_locals.erb +1 -0
- data/test/fixtures/test/render_implicit_html_template.js.rjs +1 -0
- data/test/fixtures/test/render_implicit_html_template_from_xhr_request.da.html.erb +1 -0
- data/test/fixtures/test/render_implicit_html_template_from_xhr_request.html.erb +1 -0
- data/test/fixtures/test/render_implicit_js_template_without_layout.js.erb +1 -0
- data/test/fixtures/test/render_to_string_test.erb +1 -0
- data/test/fixtures/test/scoped_array_translation.erb +1 -0
- data/test/fixtures/test/sub_template_raise.html.erb +1 -0
- data/test/fixtures/test/template.erb +1 -0
- data/test/fixtures/test/translation.erb +1 -0
- data/test/fixtures/test/update_element_with_capture.erb +9 -0
- data/test/fixtures/test/using_layout_around_block.html.erb +1 -0
- data/test/fixtures/test/using_layout_around_block_with_args.html.erb +1 -0
- data/test/fixtures/test/utf8.html.erb +4 -0
- data/test/fixtures/test/utf8_magic.html.erb +5 -0
- data/test/fixtures/test/utf8_magic_with_bare_partial.html.erb +5 -0
- data/test/fixtures/topic.rb +3 -0
- data/test/fixtures/topics/_topic.html.erb +1 -0
- data/test/fixtures/topics.yml +22 -0
- data/test/template/active_record_helper_i18n_test.rb +51 -0
- data/test/template/active_record_helper_test.rb +302 -0
- data/test/template/asset_tag_helper_test.rb +770 -0
- data/test/template/atom_feed_helper_test.rb +315 -0
- data/test/template/benchmark_helper_test.rb +86 -0
- data/test/template/compiled_templates_test.rb +204 -0
- data/test/template/date_helper_i18n_test.rb +121 -0
- data/test/template/date_helper_test.rb +2603 -0
- data/test/template/erb_util_test.rb +36 -0
- data/test/template/form_helper_test.rb +1447 -0
- data/test/template/form_options_helper_i18n_test.rb +27 -0
- data/test/template/form_options_helper_test.rb +811 -0
- data/test/template/form_tag_helper_test.rb +356 -0
- data/test/template/javascript_helper_test.rb +106 -0
- data/test/template/number_helper_i18n_test.rb +69 -0
- data/test/template/number_helper_test.rb +132 -0
- data/test/template/prototype_helper_test.rb +639 -0
- data/test/template/raw_output_helper_test.rb +21 -0
- data/test/template/record_tag_helper_test.rb +58 -0
- data/test/template/render_test.rb +329 -0
- data/test/template/sanitize_helper_test.rb +57 -0
- data/test/template/scriptaculous_helper_test.rb +90 -0
- data/test/template/tag_helper_test.rb +98 -0
- data/test/template/template_test.rb +32 -0
- data/test/template/test_test.rb +54 -0
- data/test/template/text_helper_test.rb +597 -0
- data/test/template/translation_helper_test.rb +95 -0
- data/test/template/url_helper_test.rb +641 -0
- data/test/testing_sandbox.rb +15 -0
- data/test/view/test_case_test.rb +176 -0
- metadata +519 -0
|
@@ -0,0 +1,362 @@
|
|
|
1
|
+
module ActionView #:nodoc:
|
|
2
|
+
class ActionViewError < StandardError #:nodoc:
|
|
3
|
+
end
|
|
4
|
+
|
|
5
|
+
class MissingTemplate < ActionViewError #:nodoc:
|
|
6
|
+
attr_reader :path
|
|
7
|
+
|
|
8
|
+
def initialize(paths, path, template_format = nil)
|
|
9
|
+
@path = path
|
|
10
|
+
full_template_path = path.include?('.') ? path : "#{path}.erb"
|
|
11
|
+
display_paths = paths.compact.join(":")
|
|
12
|
+
template_type = (path =~ /layouts/i) ? 'layout' : 'template'
|
|
13
|
+
super("Missing #{template_type} #{full_template_path} in view path #{display_paths}")
|
|
14
|
+
end
|
|
15
|
+
end
|
|
16
|
+
|
|
17
|
+
# Action View templates can be written in three ways. If the template file has a <tt>.erb</tt> (or <tt>.rhtml</tt>) extension then it uses a mixture of ERb
|
|
18
|
+
# (included in Ruby) and HTML. If the template file has a <tt>.builder</tt> (or <tt>.rxml</tt>) extension then Jim Weirich's Builder::XmlMarkup library is used.
|
|
19
|
+
# If the template file has a <tt>.rjs</tt> extension then it will use ActionView::Helpers::PrototypeHelper::JavaScriptGenerator.
|
|
20
|
+
#
|
|
21
|
+
# = ERb
|
|
22
|
+
#
|
|
23
|
+
# You trigger ERb by using embeddings such as <% %>, <% -%>, and <%= %>. The <%= %> tag set is used when you want output. Consider the
|
|
24
|
+
# following loop for names:
|
|
25
|
+
#
|
|
26
|
+
# <b>Names of all the people</b>
|
|
27
|
+
# <% for person in @people %>
|
|
28
|
+
# Name: <%= person.name %><br/>
|
|
29
|
+
# <% end %>
|
|
30
|
+
#
|
|
31
|
+
# The loop is setup in regular embedding tags <% %> and the name is written using the output embedding tag <%= %>. Note that this
|
|
32
|
+
# is not just a usage suggestion. Regular output functions like print or puts won't work with ERb templates. So this would be wrong:
|
|
33
|
+
#
|
|
34
|
+
# Hi, Mr. <% puts "Frodo" %>
|
|
35
|
+
#
|
|
36
|
+
# If you absolutely must write from within a function, you can use the TextHelper#concat.
|
|
37
|
+
#
|
|
38
|
+
# <%- and -%> suppress leading and trailing whitespace, including the trailing newline, and can be used interchangeably with <% and %>.
|
|
39
|
+
#
|
|
40
|
+
# == Using sub templates
|
|
41
|
+
#
|
|
42
|
+
# Using sub templates allows you to sidestep tedious replication and extract common display structures in shared templates. The
|
|
43
|
+
# classic example is the use of a header and footer (even though the Action Pack-way would be to use Layouts):
|
|
44
|
+
#
|
|
45
|
+
# <%= render "shared/header" %>
|
|
46
|
+
# Something really specific and terrific
|
|
47
|
+
# <%= render "shared/footer" %>
|
|
48
|
+
#
|
|
49
|
+
# As you see, we use the output embeddings for the render methods. The render call itself will just return a string holding the
|
|
50
|
+
# result of the rendering. The output embedding writes it to the current template.
|
|
51
|
+
#
|
|
52
|
+
# But you don't have to restrict yourself to static includes. Templates can share variables amongst themselves by using instance
|
|
53
|
+
# variables defined using the regular embedding tags. Like this:
|
|
54
|
+
#
|
|
55
|
+
# <% @page_title = "A Wonderful Hello" %>
|
|
56
|
+
# <%= render "shared/header" %>
|
|
57
|
+
#
|
|
58
|
+
# Now the header can pick up on the <tt>@page_title</tt> variable and use it for outputting a title tag:
|
|
59
|
+
#
|
|
60
|
+
# <title><%= @page_title %></title>
|
|
61
|
+
#
|
|
62
|
+
# == Passing local variables to sub templates
|
|
63
|
+
#
|
|
64
|
+
# You can pass local variables to sub templates by using a hash with the variable names as keys and the objects as values:
|
|
65
|
+
#
|
|
66
|
+
# <%= render "shared/header", { :headline => "Welcome", :person => person } %>
|
|
67
|
+
#
|
|
68
|
+
# These can now be accessed in <tt>shared/header</tt> with:
|
|
69
|
+
#
|
|
70
|
+
# Headline: <%= headline %>
|
|
71
|
+
# First name: <%= person.first_name %>
|
|
72
|
+
#
|
|
73
|
+
# If you need to find out whether a certain local variable has been assigned a value in a particular render call,
|
|
74
|
+
# you need to use the following pattern:
|
|
75
|
+
#
|
|
76
|
+
# <% if local_assigns.has_key? :headline %>
|
|
77
|
+
# Headline: <%= headline %>
|
|
78
|
+
# <% end %>
|
|
79
|
+
#
|
|
80
|
+
# Testing using <tt>defined? headline</tt> will not work. This is an implementation restriction.
|
|
81
|
+
#
|
|
82
|
+
# == Template caching
|
|
83
|
+
#
|
|
84
|
+
# By default, Rails will compile each template to a method in order to render it. When you alter a template, Rails will
|
|
85
|
+
# check the file's modification time and recompile it.
|
|
86
|
+
#
|
|
87
|
+
# == Builder
|
|
88
|
+
#
|
|
89
|
+
# Builder templates are a more programmatic alternative to ERb. They are especially useful for generating XML content. An XmlMarkup object
|
|
90
|
+
# named +xml+ is automatically made available to templates with a <tt>.builder</tt> extension.
|
|
91
|
+
#
|
|
92
|
+
# Here are some basic examples:
|
|
93
|
+
#
|
|
94
|
+
# xml.em("emphasized") # => <em>emphasized</em>
|
|
95
|
+
# xml.em { xml.b("emph & bold") } # => <em><b>emph & bold</b></em>
|
|
96
|
+
# xml.a("A Link", "href"=>"http://onestepback.org") # => <a href="http://onestepback.org">A Link</a>
|
|
97
|
+
# xml.target("name"=>"compile", "option"=>"fast") # => <target option="fast" name="compile"\>
|
|
98
|
+
# # NOTE: order of attributes is not specified.
|
|
99
|
+
#
|
|
100
|
+
# Any method with a block will be treated as an XML markup tag with nested markup in the block. For example, the following:
|
|
101
|
+
#
|
|
102
|
+
# xml.div {
|
|
103
|
+
# xml.h1(@person.name)
|
|
104
|
+
# xml.p(@person.bio)
|
|
105
|
+
# }
|
|
106
|
+
#
|
|
107
|
+
# would produce something like:
|
|
108
|
+
#
|
|
109
|
+
# <div>
|
|
110
|
+
# <h1>David Heinemeier Hansson</h1>
|
|
111
|
+
# <p>A product of Danish Design during the Winter of '79...</p>
|
|
112
|
+
# </div>
|
|
113
|
+
#
|
|
114
|
+
# A full-length RSS example actually used on Basecamp:
|
|
115
|
+
#
|
|
116
|
+
# xml.rss("version" => "2.0", "xmlns:dc" => "http://purl.org/dc/elements/1.1/") do
|
|
117
|
+
# xml.channel do
|
|
118
|
+
# xml.title(@feed_title)
|
|
119
|
+
# xml.link(@url)
|
|
120
|
+
# xml.description "Basecamp: Recent items"
|
|
121
|
+
# xml.language "en-us"
|
|
122
|
+
# xml.ttl "40"
|
|
123
|
+
#
|
|
124
|
+
# for item in @recent_items
|
|
125
|
+
# xml.item do
|
|
126
|
+
# xml.title(item_title(item))
|
|
127
|
+
# xml.description(item_description(item)) if item_description(item)
|
|
128
|
+
# xml.pubDate(item_pubDate(item))
|
|
129
|
+
# xml.guid(@person.firm.account.url + @recent_items.url(item))
|
|
130
|
+
# xml.link(@person.firm.account.url + @recent_items.url(item))
|
|
131
|
+
#
|
|
132
|
+
# xml.tag!("dc:creator", item.author_name) if item_has_creator?(item)
|
|
133
|
+
# end
|
|
134
|
+
# end
|
|
135
|
+
# end
|
|
136
|
+
# end
|
|
137
|
+
#
|
|
138
|
+
# More builder documentation can be found at http://builder.rubyforge.org.
|
|
139
|
+
#
|
|
140
|
+
# == JavaScriptGenerator
|
|
141
|
+
#
|
|
142
|
+
# JavaScriptGenerator templates end in <tt>.rjs</tt>. Unlike conventional templates which are used to
|
|
143
|
+
# render the results of an action, these templates generate instructions on how to modify an already rendered page. This makes it easy to
|
|
144
|
+
# modify multiple elements on your page in one declarative Ajax response. Actions with these templates are called in the background with Ajax
|
|
145
|
+
# and make updates to the page where the request originated from.
|
|
146
|
+
#
|
|
147
|
+
# An instance of the JavaScriptGenerator object named +page+ is automatically made available to your template, which is implicitly wrapped in an ActionView::Helpers::PrototypeHelper#update_page block.
|
|
148
|
+
#
|
|
149
|
+
# When an <tt>.rjs</tt> action is called with +link_to_remote+, the generated JavaScript is automatically evaluated. Example:
|
|
150
|
+
#
|
|
151
|
+
# link_to_remote :url => {:action => 'delete'}
|
|
152
|
+
#
|
|
153
|
+
# The subsequently rendered <tt>delete.rjs</tt> might look like:
|
|
154
|
+
#
|
|
155
|
+
# page.replace_html 'sidebar', :partial => 'sidebar'
|
|
156
|
+
# page.remove "person-#{@person.id}"
|
|
157
|
+
# page.visual_effect :highlight, 'user-list'
|
|
158
|
+
#
|
|
159
|
+
# This refreshes the sidebar, removes a person element and highlights the user list.
|
|
160
|
+
#
|
|
161
|
+
# See the ActionView::Helpers::PrototypeHelper::GeneratorMethods documentation for more details.
|
|
162
|
+
class Base
|
|
163
|
+
include Helpers, Partials, ::ERB::Util
|
|
164
|
+
extend ActiveSupport::Memoizable
|
|
165
|
+
|
|
166
|
+
attr_accessor :base_path, :assigns, :template_extension
|
|
167
|
+
attr_accessor :controller
|
|
168
|
+
|
|
169
|
+
attr_writer :template_format
|
|
170
|
+
|
|
171
|
+
attr_accessor :output_buffer
|
|
172
|
+
|
|
173
|
+
class << self
|
|
174
|
+
delegate :erb_trim_mode=, :to => 'ActionView::TemplateHandlers::ERB'
|
|
175
|
+
delegate :logger, :to => 'ActionController::Base'
|
|
176
|
+
end
|
|
177
|
+
|
|
178
|
+
@@debug_rjs = false
|
|
179
|
+
##
|
|
180
|
+
# :singleton-method:
|
|
181
|
+
# Specify whether RJS responses should be wrapped in a try/catch block
|
|
182
|
+
# that alert()s the caught exception (and then re-raises it).
|
|
183
|
+
cattr_accessor :debug_rjs
|
|
184
|
+
|
|
185
|
+
# Specify whether templates should be cached. Otherwise the file we be read everytime it is accessed.
|
|
186
|
+
# Automatically reloading templates are not thread safe and should only be used in development mode.
|
|
187
|
+
@@cache_template_loading = nil
|
|
188
|
+
cattr_accessor :cache_template_loading
|
|
189
|
+
|
|
190
|
+
# :nodoc:
|
|
191
|
+
def self.xss_safe?
|
|
192
|
+
false
|
|
193
|
+
end
|
|
194
|
+
|
|
195
|
+
def self.cache_template_loading?
|
|
196
|
+
ActionController::Base.allow_concurrency || (cache_template_loading.nil? ? !ActiveSupport::Dependencies.load? : cache_template_loading)
|
|
197
|
+
end
|
|
198
|
+
|
|
199
|
+
attr_internal :request
|
|
200
|
+
|
|
201
|
+
delegate :request_forgery_protection_token, :params, :session, :cookies, :response, :headers,
|
|
202
|
+
:flash, :logger, :action_name, :controller_name, :to => :controller
|
|
203
|
+
|
|
204
|
+
module CompiledTemplates #:nodoc:
|
|
205
|
+
# holds compiled template code
|
|
206
|
+
end
|
|
207
|
+
include CompiledTemplates
|
|
208
|
+
|
|
209
|
+
def self.process_view_paths(value)
|
|
210
|
+
ActionView::PathSet.new(Array(value))
|
|
211
|
+
end
|
|
212
|
+
|
|
213
|
+
attr_reader :helpers
|
|
214
|
+
|
|
215
|
+
class ProxyModule < Module
|
|
216
|
+
def initialize(receiver)
|
|
217
|
+
@receiver = receiver
|
|
218
|
+
end
|
|
219
|
+
|
|
220
|
+
def include(*args)
|
|
221
|
+
super(*args)
|
|
222
|
+
@receiver.extend(*args)
|
|
223
|
+
end
|
|
224
|
+
end
|
|
225
|
+
|
|
226
|
+
def initialize(view_paths = [], assigns_for_first_render = {}, controller = nil)#:nodoc:
|
|
227
|
+
@assigns = assigns_for_first_render
|
|
228
|
+
@assigns_added = nil
|
|
229
|
+
@controller = controller
|
|
230
|
+
@helpers = ProxyModule.new(self)
|
|
231
|
+
self.view_paths = view_paths
|
|
232
|
+
|
|
233
|
+
@_first_render = nil
|
|
234
|
+
@_current_render = nil
|
|
235
|
+
end
|
|
236
|
+
|
|
237
|
+
attr_reader :view_paths
|
|
238
|
+
|
|
239
|
+
def view_paths=(paths)
|
|
240
|
+
@view_paths = self.class.process_view_paths(paths)
|
|
241
|
+
# we might be using ReloadableTemplates, so we need to let them know this a new request
|
|
242
|
+
@view_paths.load!
|
|
243
|
+
end
|
|
244
|
+
|
|
245
|
+
# Returns the result of a render that's dictated by the options hash. The primary options are:
|
|
246
|
+
#
|
|
247
|
+
# * <tt>:partial</tt> - See ActionView::Partials.
|
|
248
|
+
# * <tt>:update</tt> - Calls update_page with the block given.
|
|
249
|
+
# * <tt>:file</tt> - Renders an explicit template file (this used to be the old default), add :locals to pass in those.
|
|
250
|
+
# * <tt>:inline</tt> - Renders an inline template similar to how it's done in the controller.
|
|
251
|
+
# * <tt>:text</tt> - Renders the text passed in out.
|
|
252
|
+
#
|
|
253
|
+
# If no options hash is passed or :update specified, the default is to render a partial and use the second parameter
|
|
254
|
+
# as the locals hash.
|
|
255
|
+
def render(options = {}, local_assigns = {}, &block) #:nodoc:
|
|
256
|
+
local_assigns ||= {}
|
|
257
|
+
|
|
258
|
+
case options
|
|
259
|
+
when Hash
|
|
260
|
+
options = options.reverse_merge(:locals => {})
|
|
261
|
+
if options[:layout]
|
|
262
|
+
_render_with_layout(options, local_assigns, &block)
|
|
263
|
+
elsif options[:file]
|
|
264
|
+
template = self.view_paths.find_template(options[:file], template_format)
|
|
265
|
+
template.render_template(self, options[:locals])
|
|
266
|
+
elsif options[:partial]
|
|
267
|
+
render_partial(options)
|
|
268
|
+
elsif options[:inline]
|
|
269
|
+
InlineTemplate.new(options[:inline], options[:type]).render(self, options[:locals])
|
|
270
|
+
elsif options[:text]
|
|
271
|
+
options[:text]
|
|
272
|
+
end
|
|
273
|
+
when :update
|
|
274
|
+
update_page(&block)
|
|
275
|
+
else
|
|
276
|
+
render_partial(:partial => options, :locals => local_assigns)
|
|
277
|
+
end
|
|
278
|
+
end
|
|
279
|
+
|
|
280
|
+
# The format to be used when choosing between multiple templates with
|
|
281
|
+
# the same name but differing formats. See +Request#template_format+
|
|
282
|
+
# for more details.
|
|
283
|
+
def template_format
|
|
284
|
+
if defined? @template_format
|
|
285
|
+
@template_format
|
|
286
|
+
elsif controller && controller.respond_to?(:request)
|
|
287
|
+
@template_format = controller.request.template_format.to_sym
|
|
288
|
+
else
|
|
289
|
+
@template_format = :html
|
|
290
|
+
end
|
|
291
|
+
end
|
|
292
|
+
|
|
293
|
+
# Access the current template being rendered.
|
|
294
|
+
# Returns a ActionView::Template object.
|
|
295
|
+
def template
|
|
296
|
+
@_current_render
|
|
297
|
+
end
|
|
298
|
+
|
|
299
|
+
def template=(template) #:nodoc:
|
|
300
|
+
@_first_render ||= template
|
|
301
|
+
@_current_render = template
|
|
302
|
+
end
|
|
303
|
+
|
|
304
|
+
def with_template(current_template)
|
|
305
|
+
last_template, self.template = template, current_template
|
|
306
|
+
yield
|
|
307
|
+
ensure
|
|
308
|
+
self.template = last_template
|
|
309
|
+
end
|
|
310
|
+
|
|
311
|
+
private
|
|
312
|
+
# Evaluates the local assigns and controller ivars, pushes them to the view.
|
|
313
|
+
def _evaluate_assigns_and_ivars #:nodoc:
|
|
314
|
+
unless @assigns_added
|
|
315
|
+
@assigns.each { |key, value| instance_variable_set("@#{key}", value) }
|
|
316
|
+
_copy_ivars_from_controller
|
|
317
|
+
@assigns_added = true
|
|
318
|
+
end
|
|
319
|
+
end
|
|
320
|
+
|
|
321
|
+
def _copy_ivars_from_controller #:nodoc:
|
|
322
|
+
if @controller
|
|
323
|
+
variables = @controller.instance_variable_names
|
|
324
|
+
variables -= @controller.protected_instance_variables if @controller.respond_to?(:protected_instance_variables)
|
|
325
|
+
variables.each { |name| instance_variable_set(name, @controller.instance_variable_get(name)) }
|
|
326
|
+
end
|
|
327
|
+
end
|
|
328
|
+
|
|
329
|
+
def _set_controller_content_type(content_type) #:nodoc:
|
|
330
|
+
if controller.respond_to?(:response)
|
|
331
|
+
controller.response.content_type ||= content_type
|
|
332
|
+
end
|
|
333
|
+
end
|
|
334
|
+
|
|
335
|
+
def _render_with_layout(options, local_assigns, &block) #:nodoc:
|
|
336
|
+
partial_layout = options.delete(:layout)
|
|
337
|
+
|
|
338
|
+
if block_given?
|
|
339
|
+
begin
|
|
340
|
+
@_proc_for_layout = block
|
|
341
|
+
concat(render(options.merge(:partial => partial_layout)))
|
|
342
|
+
ensure
|
|
343
|
+
@_proc_for_layout = nil
|
|
344
|
+
end
|
|
345
|
+
else
|
|
346
|
+
begin
|
|
347
|
+
original_content_for_layout = @content_for_layout if defined?(@content_for_layout)
|
|
348
|
+
@content_for_layout = render(options)
|
|
349
|
+
|
|
350
|
+
if (options[:inline] || options[:file] || options[:text])
|
|
351
|
+
@cached_content_for_layout = @content_for_layout
|
|
352
|
+
render(:file => partial_layout, :locals => local_assigns)
|
|
353
|
+
else
|
|
354
|
+
render(options.merge(:partial => partial_layout))
|
|
355
|
+
end
|
|
356
|
+
ensure
|
|
357
|
+
@content_for_layout = original_content_for_layout
|
|
358
|
+
end
|
|
359
|
+
end
|
|
360
|
+
end
|
|
361
|
+
end
|
|
362
|
+
end
|
|
@@ -0,0 +1,305 @@
|
|
|
1
|
+
require 'cgi'
|
|
2
|
+
require 'action_view/helpers/form_helper'
|
|
3
|
+
|
|
4
|
+
module ActionView
|
|
5
|
+
class Base
|
|
6
|
+
@@field_error_proc = Proc.new{ |html_tag, instance| "<div class=\"fieldWithErrors\">#{html_tag}</div>".html_safe }
|
|
7
|
+
cattr_accessor :field_error_proc
|
|
8
|
+
end
|
|
9
|
+
|
|
10
|
+
module Helpers
|
|
11
|
+
# The Active Record Helper makes it easier to create forms for records kept in instance variables. The most far-reaching is the +form+
|
|
12
|
+
# method that creates a complete form for all the basic content types of the record (not associations or aggregations, though). This
|
|
13
|
+
# is a great way of making the record quickly available for editing, but likely to prove lackluster for a complicated real-world form.
|
|
14
|
+
# In that case, it's better to use the +input+ method and the specialized +form+ methods in link:classes/ActionView/Helpers/FormHelper.html
|
|
15
|
+
module ActiveRecordHelper
|
|
16
|
+
# Returns a default input tag for the type of object returned by the method. For example, if <tt>@post</tt>
|
|
17
|
+
# has an attribute +title+ mapped to a +VARCHAR+ column that holds "Hello World":
|
|
18
|
+
#
|
|
19
|
+
# input("post", "title")
|
|
20
|
+
# # => <input id="post_title" name="post[title]" size="30" type="text" value="Hello World" />
|
|
21
|
+
def input(record_name, method, options = {})
|
|
22
|
+
InstanceTag.new(record_name, method, self).to_tag(options)
|
|
23
|
+
end
|
|
24
|
+
|
|
25
|
+
# Returns an entire form with all needed input tags for a specified Active Record object. For example, if <tt>@post</tt>
|
|
26
|
+
# has attributes named +title+ of type +VARCHAR+ and +body+ of type +TEXT+ then
|
|
27
|
+
#
|
|
28
|
+
# form("post")
|
|
29
|
+
#
|
|
30
|
+
# would yield a form like the following (modulus formatting):
|
|
31
|
+
#
|
|
32
|
+
# <form action='/posts/create' method='post'>
|
|
33
|
+
# <p>
|
|
34
|
+
# <label for="post_title">Title</label><br />
|
|
35
|
+
# <input id="post_title" name="post[title]" size="30" type="text" value="Hello World" />
|
|
36
|
+
# </p>
|
|
37
|
+
# <p>
|
|
38
|
+
# <label for="post_body">Body</label><br />
|
|
39
|
+
# <textarea cols="40" id="post_body" name="post[body]" rows="20"></textarea>
|
|
40
|
+
# </p>
|
|
41
|
+
# <input name="commit" type="submit" value="Create" />
|
|
42
|
+
# </form>
|
|
43
|
+
#
|
|
44
|
+
# It's possible to specialize the form builder by using a different action name and by supplying another
|
|
45
|
+
# block renderer. For example, if <tt>@entry</tt> has an attribute +message+ of type +VARCHAR+ then
|
|
46
|
+
#
|
|
47
|
+
# form("entry",
|
|
48
|
+
# :action => "sign",
|
|
49
|
+
# :input_block => Proc.new { |record, column|
|
|
50
|
+
# "#{column.human_name}: #{input(record, column.name)}<br />"
|
|
51
|
+
# })
|
|
52
|
+
#
|
|
53
|
+
# would yield a form like the following (modulus formatting):
|
|
54
|
+
#
|
|
55
|
+
# <form action="/entries/sign" method="post">
|
|
56
|
+
# Message:
|
|
57
|
+
# <input id="entry_message" name="entry[message]" size="30" type="text" /><br />
|
|
58
|
+
# <input name="commit" type="submit" value="Sign" />
|
|
59
|
+
# </form>
|
|
60
|
+
#
|
|
61
|
+
# It's also possible to add additional content to the form by giving it a block, such as:
|
|
62
|
+
#
|
|
63
|
+
# form("entry", :action => "sign") do |form|
|
|
64
|
+
# form << content_tag("b", "Department")
|
|
65
|
+
# form << collection_select("department", "id", @departments, "id", "name")
|
|
66
|
+
# end
|
|
67
|
+
#
|
|
68
|
+
# The following options are available:
|
|
69
|
+
#
|
|
70
|
+
# * <tt>:action</tt> - The action used when submitting the form (default: +create+ if a new record, otherwise +update+).
|
|
71
|
+
# * <tt>:input_block</tt> - Specialize the output using a different block, see above.
|
|
72
|
+
# * <tt>:method</tt> - The method used when submitting the form (default: +post+).
|
|
73
|
+
# * <tt>:multipart</tt> - Whether to change the enctype of the form to "multipart/form-data", used when uploading a file (default: +false+).
|
|
74
|
+
# * <tt>:submit_value</tt> - The text of the submit button (default: "Create" if a new record, otherwise "Update").
|
|
75
|
+
def form(record_name, options = {})
|
|
76
|
+
record = instance_variable_get("@#{record_name}")
|
|
77
|
+
|
|
78
|
+
options = options.symbolize_keys
|
|
79
|
+
options[:action] ||= record.new_record? ? "create" : "update"
|
|
80
|
+
action = url_for(:action => options[:action], :id => record)
|
|
81
|
+
|
|
82
|
+
submit_value = options[:submit_value] || options[:action].gsub(/[^\w]/, '').capitalize
|
|
83
|
+
|
|
84
|
+
contents = form_tag({:action => action}, :method =>(options[:method] || 'post'), :enctype => options[:multipart] ? 'multipart/form-data': nil)
|
|
85
|
+
contents.safe_concat hidden_field(record_name, :id) unless record.new_record?
|
|
86
|
+
contents.safe_concat all_input_tags(record, record_name, options)
|
|
87
|
+
yield contents if block_given?
|
|
88
|
+
contents.safe_concat submit_tag(submit_value)
|
|
89
|
+
contents.safe_concat '</form>'
|
|
90
|
+
end
|
|
91
|
+
|
|
92
|
+
# Returns a string containing the error message attached to the +method+ on the +object+ if one exists.
|
|
93
|
+
# This error message is wrapped in a <tt>DIV</tt> tag, which can be extended to include a <tt>:prepend_text</tt>
|
|
94
|
+
# and/or <tt>:append_text</tt> (to properly explain the error), and a <tt>:css_class</tt> to style it
|
|
95
|
+
# accordingly. +object+ should either be the name of an instance variable or the actual object. The method can be
|
|
96
|
+
# passed in either as a string or a symbol.
|
|
97
|
+
# As an example, let's say you have a model <tt>@post</tt> that has an error message on the +title+ attribute:
|
|
98
|
+
#
|
|
99
|
+
# <%= error_message_on "post", "title" %>
|
|
100
|
+
# # => <div class="formError">can't be empty</div>
|
|
101
|
+
#
|
|
102
|
+
# <%= error_message_on @post, :title %>
|
|
103
|
+
# # => <div class="formError">can't be empty</div>
|
|
104
|
+
#
|
|
105
|
+
# <%= error_message_on "post", "title",
|
|
106
|
+
# :prepend_text => "Title simply ",
|
|
107
|
+
# :append_text => " (or it won't work).",
|
|
108
|
+
# :css_class => "inputError" %>
|
|
109
|
+
def error_message_on(object, method, *args)
|
|
110
|
+
options = args.extract_options!
|
|
111
|
+
unless args.empty?
|
|
112
|
+
ActiveSupport::Deprecation.warn('error_message_on takes an option hash instead of separate ' +
|
|
113
|
+
'prepend_text, append_text, and css_class arguments', caller)
|
|
114
|
+
|
|
115
|
+
options[:prepend_text] = args[0] || ''
|
|
116
|
+
options[:append_text] = args[1] || ''
|
|
117
|
+
options[:css_class] = args[2] || 'formError'
|
|
118
|
+
end
|
|
119
|
+
options.reverse_merge!(:prepend_text => '', :append_text => '', :css_class => 'formError')
|
|
120
|
+
|
|
121
|
+
if (obj = (object.respond_to?(:errors) ? object : instance_variable_get("@#{object}"))) &&
|
|
122
|
+
(errors = obj.errors.on(method))
|
|
123
|
+
content_tag("div",
|
|
124
|
+
"#{options[:prepend_text]}#{ERB::Util.html_escape(errors.is_a?(Array) ? errors.first : errors)}#{options[:append_text]}".html_safe,
|
|
125
|
+
:class => options[:css_class]
|
|
126
|
+
)
|
|
127
|
+
else
|
|
128
|
+
''
|
|
129
|
+
end
|
|
130
|
+
end
|
|
131
|
+
|
|
132
|
+
# Returns a string with a <tt>DIV</tt> containing all of the error messages for the objects located as instance variables by the names
|
|
133
|
+
# given. If more than one object is specified, the errors for the objects are displayed in the order that the object names are
|
|
134
|
+
# provided.
|
|
135
|
+
#
|
|
136
|
+
# This <tt>DIV</tt> can be tailored by the following options:
|
|
137
|
+
#
|
|
138
|
+
# * <tt>:header_tag</tt> - Used for the header of the error div (default: "h2").
|
|
139
|
+
# * <tt>:id</tt> - The id of the error div (default: "errorExplanation").
|
|
140
|
+
# * <tt>:class</tt> - The class of the error div (default: "errorExplanation").
|
|
141
|
+
# * <tt>:object</tt> - The object (or array of objects) for which to display errors,
|
|
142
|
+
# if you need to escape the instance variable convention.
|
|
143
|
+
# * <tt>:object_name</tt> - The object name to use in the header, or any text that you prefer.
|
|
144
|
+
# If <tt>:object_name</tt> is not set, the name of the first object will be used.
|
|
145
|
+
# * <tt>:header_message</tt> - The message in the header of the error div. Pass +nil+
|
|
146
|
+
# or an empty string to avoid the header message altogether. (Default: "X errors
|
|
147
|
+
# prohibited this object from being saved").
|
|
148
|
+
# * <tt>:message</tt> - The explanation message after the header message and before
|
|
149
|
+
# the error list. Pass +nil+ or an empty string to avoid the explanation message
|
|
150
|
+
# altogether. (Default: "There were problems with the following fields:").
|
|
151
|
+
#
|
|
152
|
+
# To specify the display for one object, you simply provide its name as a parameter.
|
|
153
|
+
# For example, for the <tt>@user</tt> model:
|
|
154
|
+
#
|
|
155
|
+
# error_messages_for 'user'
|
|
156
|
+
#
|
|
157
|
+
# To specify more than one object, you simply list them; optionally, you can add an extra <tt>:object_name</tt> parameter, which
|
|
158
|
+
# will be the name used in the header message:
|
|
159
|
+
#
|
|
160
|
+
# error_messages_for 'user_common', 'user', :object_name => 'user'
|
|
161
|
+
#
|
|
162
|
+
# If the objects cannot be located as instance variables, you can add an extra <tt>:object</tt> parameter which gives the actual
|
|
163
|
+
# object (or array of objects to use):
|
|
164
|
+
#
|
|
165
|
+
# error_messages_for 'user', :object => @question.user
|
|
166
|
+
#
|
|
167
|
+
# NOTE: This is a pre-packaged presentation of the errors with embedded strings and a certain HTML structure. If what
|
|
168
|
+
# you need is significantly different from the default presentation, it makes plenty of sense to access the <tt>object.errors</tt>
|
|
169
|
+
# instance yourself and set it up. View the source of this method to see how easy it is.
|
|
170
|
+
def error_messages_for(*params)
|
|
171
|
+
options = params.extract_options!.symbolize_keys
|
|
172
|
+
|
|
173
|
+
if object = options.delete(:object)
|
|
174
|
+
objects = Array.wrap(object)
|
|
175
|
+
else
|
|
176
|
+
objects = params.collect {|object_name| instance_variable_get("@#{object_name}") }.compact
|
|
177
|
+
end
|
|
178
|
+
|
|
179
|
+
count = objects.inject(0) {|sum, object| sum + object.errors.count }
|
|
180
|
+
unless count.zero?
|
|
181
|
+
html = {}
|
|
182
|
+
[:id, :class].each do |key|
|
|
183
|
+
if options.include?(key)
|
|
184
|
+
value = options[key]
|
|
185
|
+
html[key] = value unless value.blank?
|
|
186
|
+
else
|
|
187
|
+
html[key] = 'errorExplanation'
|
|
188
|
+
end
|
|
189
|
+
end
|
|
190
|
+
options[:object_name] ||= params.first
|
|
191
|
+
|
|
192
|
+
I18n.with_options :locale => options[:locale], :scope => [:activerecord, :errors, :template] do |locale|
|
|
193
|
+
header_message = if options.include?(:header_message)
|
|
194
|
+
options[:header_message]
|
|
195
|
+
else
|
|
196
|
+
object_name = options[:object_name].to_s
|
|
197
|
+
object_name = I18n.t(object_name, :default => object_name.gsub('_', ' '), :scope => [:activerecord, :models], :count => 1)
|
|
198
|
+
locale.t :header, :count => count, :model => object_name
|
|
199
|
+
end
|
|
200
|
+
message = options.include?(:message) ? options[:message] : locale.t(:body)
|
|
201
|
+
error_messages = objects.sum {|object| object.errors.full_messages.map {|msg| content_tag(:li, ERB::Util.html_escape(msg)) } }.join.html_safe
|
|
202
|
+
|
|
203
|
+
contents = ''
|
|
204
|
+
contents << content_tag(options[:header_tag] || :h2, header_message) unless header_message.blank?
|
|
205
|
+
contents << content_tag(:p, message) unless message.blank?
|
|
206
|
+
contents << content_tag(:ul, error_messages)
|
|
207
|
+
|
|
208
|
+
content_tag(:div, contents.html_safe, html)
|
|
209
|
+
end
|
|
210
|
+
else
|
|
211
|
+
''
|
|
212
|
+
end
|
|
213
|
+
end
|
|
214
|
+
|
|
215
|
+
private
|
|
216
|
+
def all_input_tags(record, record_name, options)
|
|
217
|
+
input_block = options[:input_block] || default_input_block
|
|
218
|
+
record.class.content_columns.collect{ |column| input_block.call(record_name, column) }.join("\n")
|
|
219
|
+
end
|
|
220
|
+
|
|
221
|
+
def default_input_block
|
|
222
|
+
Proc.new { |record, column| %(<p><label for="#{record}_#{column.name}">#{column.human_name}</label><br />#{input(record, column.name)}</p>) }
|
|
223
|
+
end
|
|
224
|
+
end
|
|
225
|
+
|
|
226
|
+
class InstanceTag #:nodoc:
|
|
227
|
+
def to_tag(options = {})
|
|
228
|
+
case column_type
|
|
229
|
+
when :string
|
|
230
|
+
field_type = @method_name.include?("password") ? "password" : "text"
|
|
231
|
+
to_input_field_tag(field_type, options)
|
|
232
|
+
when :text
|
|
233
|
+
to_text_area_tag(options)
|
|
234
|
+
when :integer, :float, :decimal
|
|
235
|
+
to_input_field_tag("text", options)
|
|
236
|
+
when :date
|
|
237
|
+
to_date_select_tag(options)
|
|
238
|
+
when :datetime, :timestamp
|
|
239
|
+
to_datetime_select_tag(options)
|
|
240
|
+
when :time
|
|
241
|
+
to_time_select_tag(options)
|
|
242
|
+
when :boolean
|
|
243
|
+
to_boolean_select_tag(options)
|
|
244
|
+
end
|
|
245
|
+
end
|
|
246
|
+
|
|
247
|
+
alias_method :tag_without_error_wrapping, :tag
|
|
248
|
+
def tag(name, options)
|
|
249
|
+
if object.respond_to?(:errors) && object.errors.respond_to?(:on)
|
|
250
|
+
error_wrapping(tag_without_error_wrapping(name, options), object.errors.on(@method_name))
|
|
251
|
+
else
|
|
252
|
+
tag_without_error_wrapping(name, options)
|
|
253
|
+
end
|
|
254
|
+
end
|
|
255
|
+
|
|
256
|
+
alias_method :content_tag_without_error_wrapping, :content_tag
|
|
257
|
+
def content_tag(name, value, options)
|
|
258
|
+
if object.respond_to?(:errors) && object.errors.respond_to?(:on)
|
|
259
|
+
error_wrapping(content_tag_without_error_wrapping(name, value, options), object.errors.on(@method_name))
|
|
260
|
+
else
|
|
261
|
+
content_tag_without_error_wrapping(name, value, options)
|
|
262
|
+
end
|
|
263
|
+
end
|
|
264
|
+
|
|
265
|
+
alias_method :to_date_select_tag_without_error_wrapping, :to_date_select_tag
|
|
266
|
+
def to_date_select_tag(options = {}, html_options = {})
|
|
267
|
+
if object.respond_to?(:errors) && object.errors.respond_to?(:on)
|
|
268
|
+
error_wrapping(to_date_select_tag_without_error_wrapping(options, html_options), object.errors.on(@method_name))
|
|
269
|
+
else
|
|
270
|
+
to_date_select_tag_without_error_wrapping(options, html_options)
|
|
271
|
+
end
|
|
272
|
+
end
|
|
273
|
+
|
|
274
|
+
alias_method :to_datetime_select_tag_without_error_wrapping, :to_datetime_select_tag
|
|
275
|
+
def to_datetime_select_tag(options = {}, html_options = {})
|
|
276
|
+
if object.respond_to?(:errors) && object.errors.respond_to?(:on)
|
|
277
|
+
error_wrapping(to_datetime_select_tag_without_error_wrapping(options, html_options), object.errors.on(@method_name))
|
|
278
|
+
else
|
|
279
|
+
to_datetime_select_tag_without_error_wrapping(options, html_options)
|
|
280
|
+
end
|
|
281
|
+
end
|
|
282
|
+
|
|
283
|
+
alias_method :to_time_select_tag_without_error_wrapping, :to_time_select_tag
|
|
284
|
+
def to_time_select_tag(options = {}, html_options = {})
|
|
285
|
+
if object.respond_to?(:errors) && object.errors.respond_to?(:on)
|
|
286
|
+
error_wrapping(to_time_select_tag_without_error_wrapping(options, html_options), object.errors.on(@method_name))
|
|
287
|
+
else
|
|
288
|
+
to_time_select_tag_without_error_wrapping(options, html_options)
|
|
289
|
+
end
|
|
290
|
+
end
|
|
291
|
+
|
|
292
|
+
def error_wrapping(html_tag, has_error)
|
|
293
|
+
has_error ? Base.field_error_proc.call(html_tag, self) : html_tag
|
|
294
|
+
end
|
|
295
|
+
|
|
296
|
+
def error_message
|
|
297
|
+
object.errors.on(@method_name)
|
|
298
|
+
end
|
|
299
|
+
|
|
300
|
+
def column_type
|
|
301
|
+
object.send(:column_for_attribute, @method_name).type
|
|
302
|
+
end
|
|
303
|
+
end
|
|
304
|
+
end
|
|
305
|
+
end
|