form_tag_helper.rb 30.9 KB
Newer Older
1
require 'cgi'
2
require 'action_view/helpers/tag_helper'
3
require 'active_support/core_ext/object/blank'
4
require 'active_support/core_ext/string/output_safety'
5 6

module ActionView
7
  # = Action View Form Tag Helpers
8
  module Helpers
P
Pratik Naik 已提交
9
    # Provides a number of methods for creating form tags that doesn't rely on an Active Record object assigned to the template like
10
    # FormHelper does. Instead, you provide the names and values manually.
11
    #
12
    # NOTE: The HTML options <tt>disabled</tt>, <tt>readonly</tt>, and <tt>multiple</tt> can all be treated as booleans. So specifying
13
    # <tt>:disabled => true</tt> will give <tt>disabled="disabled"</tt>.
14
    module FormTagHelper
W
wycats 已提交
15 16 17 18 19
      extend ActiveSupport::Concern

      include UrlHelper
      include TextHelper

20
      # Starts a form tag that points the action to an url configured with <tt>url_for_options</tt> just like
21 22
      # ActionController::Base#url_for. The method for the form defaults to POST.
      #
23
      # ==== Options
24
      # * <tt>:multipart</tt> - If set to true, the enctype is set to "multipart/form-data".
P
Pratik Naik 已提交
25 26 27
      # * <tt>:method</tt> - The method to use when submitting the form, usually either "get" or "post".
      #   If "put", "delete", or another verb is used, a hidden input with name <tt>_method</tt>
      #   is added to simulate the verb over post.
28 29 30
      # * <tt>:authenticity_token</tt> - Authenticity token to use in the form. Use only if you need to
      #   pass custom authenticity token string, or to not add authenticity_token field at all
      #   (by passing <tt>false</tt>).
31
      # * A list of parameters to feed to the URL the form will be posted to.
32
      # * <tt>:remote</tt> - If set to true, will allow the Unobtrusive JavaScript drivers to control the
33
      #   submit behavior. By default this behavior is an ajax submit.
34 35
      #
      # ==== Examples
36
      #   form_tag('/posts')
37 38
      #   # => <form action="/posts" method="post">
      #
39
      #   form_tag('/posts/1', :method => :put)
40 41
      #   # => <form action="/posts/1" method="put">
      #
42
      #   form_tag('/upload', :multipart => true)
43
      #   # => <form action="/upload" method="post" enctype="multipart/form-data">
44
      #
A
Akira Matsuda 已提交
45
      #   <%= form_tag('/posts') do -%>
46 47 48
      #     <div><%= submit_tag 'Save' %></div>
      #   <% end -%>
      #   # => <form action="/posts" method="post"><div><input type="submit" name="submit" value="Save" /></div></form>
49
      #
50
      #   <%= form_tag('/posts', :remote => true) %>
S
Stefan Penner 已提交
51
      #   # => <form action="/posts" method="post" data-remote="true">
52
      #
53 54 55 56 57 58
      #   form_tag('http://far.away.com/form', :authenticity_token => false)
      #   # form without authenticity token
      #
      #   form_tag('http://far.away.com/form', :authenticity_token => "cf50faa3fe97702ca1ae")
      #   # form with custom authenticity token
      #
59 60
      def form_tag(url_for_options = {}, options = {}, &block)
        html_options = html_options_for_form(url_for_options, options)
61
        if block_given?
62
          form_tag_in_block(html_options, &block)
63
        else
64
          form_tag_html(html_options)
65
        end
66 67
      end

68 69 70 71
      # Creates a dropdown selection box, or if the <tt>:multiple</tt> option is set to true, a multiple
      # choice selection box.
      #
      # Helpers::FormOptions can be used to create common select boxes such as countries, time zones, or
72 73 74 75 76
      # associated records. <tt>option_tags</tt> is a string containing the option tags for the select box.
      #
      # ==== Options
      # * <tt>:multiple</tt> - If set to true the selection will allow multiple choices.
      # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
77 78
      # * <tt>:include_blank</tt> - If set to true, an empty option will be create
      # * <tt>:prompt</tt> - Create a prompt option with blank value and the text asking user to select something
79
      # * Any other key creates standard HTML attributes for the tag.
80
      #
81
      # ==== Examples
82
      #   select_tag "people", options_from_collection_for_select(@people, "id", "name")
P
Pratik Naik 已提交
83
      #   # <select id="people" name="people"><option value="1">David</option></select>
84
      #
85
      #   select_tag "people", "<option>David</option>".html_safe
86
      #   # => <select id="people" name="people"><option>David</option></select>
87
      #
88
      #   select_tag "count", "<option>1</option><option>2</option><option>3</option><option>4</option>".html_safe
89 90 91
      #   # => <select id="count" name="count"><option>1</option><option>2</option>
      #   #    <option>3</option><option>4</option></select>
      #
92
      #   select_tag "colors", "<option>Red</option><option>Green</option><option>Blue</option>".html_safe, :multiple => true
93
      #   # => <select id="colors" multiple="multiple" name="colors[]"><option>Red</option>
94 95
      #   #    <option>Green</option><option>Blue</option></select>
      #
96
      #   select_tag "locations", "<option>Home</option><option selected="selected">Work</option><option>Out</option>".html_safe
97 98 99
      #   # => <select id="locations" name="locations"><option>Home</option><option selected='selected'>Work</option>
      #   #    <option>Out</option></select>
      #
100
      #   select_tag "access", "<option>Read</option><option>Write</option>".html_safe, :multiple => true, :class => 'form_input'
101
      #   # => <select class="form_input" id="access" multiple="multiple" name="access[]"><option>Read</option>
102 103
      #   #    <option>Write</option></select>
      #
104 105 106 107 108 109
      #   select_tag "people", options_from_collection_for_select(@people, "id", "name"), :include_blank => true
      #   # => <select id="people" name="people"><option value=""></option><option value="1">David</option></select>
      #
      #   select_tag "people", options_from_collection_for_select(@people, "id", "name"), :prompt => "Select something"
      #   # => <select id="people" name="people"><option value="">Select something</option><option value="1">David</option></select>
      #
110
      #   select_tag "destination", "<option>NYC</option><option>Paris</option><option>Rome</option>".html_safe, :disabled => true
111 112
      #   # => <select disabled="disabled" id="destination" name="destination"><option>NYC</option>
      #   #    <option>Paris</option><option>Rome</option></select>
113
      def select_tag(name, option_tags = nil, options = {})
114
        html_name = (options[:multiple] == true && !name.to_s.ends_with?("[]")) ? "#{name}[]" : name
115 116 117

        if options.delete(:include_blank)
          option_tags = "<option value=\"\"></option>".html_safe + option_tags
118
        end
119 120 121 122 123

        if prompt = options.delete(:prompt)
          option_tags = "<option value=\"\">#{prompt}</option>".html_safe + option_tags
        end

124
        content_tag :select, option_tags, { "name" => html_name, "id" => sanitize_to_id(name) }.update(options.stringify_keys)
125 126
      end

127 128
      # Creates a standard text field; use these text fields to input smaller chunks of text like a username
      # or a search query.
129
      #
130
      # ==== Options
131 132 133
      # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
      # * <tt>:size</tt> - The number of visible characters that will fit in the input.
      # * <tt>:maxlength</tt> - The maximum number of characters that the browser will allow the user to enter.
134
      # * <tt>:placeholder</tt> - The text contained in the field by default which is removed when the field receives focus.
135
      # * Any other key creates standard HTML attributes for the tag.
136
      #
137 138 139 140 141 142 143
      # ==== Examples
      #   text_field_tag 'name'
      #   # => <input id="name" name="name" type="text" />
      #
      #   text_field_tag 'query', 'Enter your search query here'
      #   # => <input id="query" name="query" type="text" value="Enter your search query here" />
      #
144 145 146
      #   text_field_tag 'search', nil, :placeholder => 'Enter search term...'
      #   # => <input id="search" name="search" placeholder="Enter search term..." type="text" />
      #
147 148 149 150 151 152 153 154 155 156 157 158 159 160
      #   text_field_tag 'request', nil, :class => 'special_input'
      #   # => <input class="special_input" id="request" name="request" type="text" />
      #
      #   text_field_tag 'address', '', :size => 75
      #   # => <input id="address" name="address" size="75" type="text" value="" />
      #
      #   text_field_tag 'zip', nil, :maxlength => 5
      #   # => <input id="zip" maxlength="5" name="zip" type="text" />
      #
      #   text_field_tag 'payment_amount', '$0.00', :disabled => true
      #   # => <input disabled="disabled" id="payment_amount" name="payment_amount" type="text" value="$0.00" />
      #
      #   text_field_tag 'ip', '0.0.0.0', :maxlength => 15, :size => 20, :class => "ip-input"
      #   # => <input class="ip-input" id="ip" maxlength="15" name="ip" size="20" type="text" value="0.0.0.0" />
161
      def text_field_tag(name, value = nil, options = {})
162
        tag :input, { "type" => "text", "name" => name, "id" => sanitize_to_id(name), "value" => value }.update(options.stringify_keys)
163 164
      end

S
Stephen Celis 已提交
165
      # Creates a label element. Accepts a block.
166
      #
167
      # ==== Options
168 169 170 171 172 173 174 175 176 177 178
      # * Creates standard HTML attributes for the tag.
      #
      # ==== Examples
      #   label_tag 'name'
      #   # => <label for="name">Name</label>
      #
      #   label_tag 'name', 'Your name'
      #   # => <label for="name">Your Name</label>
      #
      #   label_tag 'name', nil, :class => 'small_label'
      #   # => <label for="name" class="small_label">Name</label>
179
      def label_tag(name = nil, content_or_options = nil, options = nil, &block)
S
Santiago Pastorino 已提交
180 181 182 183 184 185
        if block_given? && content_or_options.is_a?(Hash)
          options = content_or_options = content_or_options.stringify_keys
        else
          options ||= {}
          options = options.stringify_keys
        end
S
Stephen Celis 已提交
186
        options["for"] = sanitize_to_id(name) unless name.blank? || options.has_key?("for")
S
Santiago Pastorino 已提交
187
        content_tag :label, content_or_options || name.to_s.humanize, options, &block
188 189
      end

190 191 192 193 194 195 196 197 198
      # Creates a hidden form input field used to transmit data that would be lost due to HTTP's statelessness or
      # data that should be hidden from the user.
      #
      # ==== Options
      # * Creates standard HTML attributes for the tag.
      #
      # ==== Examples
      #   hidden_field_tag 'tags_list'
      #   # => <input id="tags_list" name="tags_list" type="hidden" />
199
      #
200 201 202 203
      #   hidden_field_tag 'token', 'VUBJKB23UIVI1UU1VOBVI@'
      #   # => <input id="token" name="token" type="hidden" value="VUBJKB23UIVI1UU1VOBVI@" />
      #
      #   hidden_field_tag 'collected_input', '', :onchange => "alert('Input collected!')"
204
      #   # => <input id="collected_input" name="collected_input" onchange="alert('Input collected!')"
205
      #   #    type="hidden" value="" />
206
      def hidden_field_tag(name, value = nil, options = {})
207
        text_field_tag(name, value, options.stringify_keys.update("type" => "hidden"))
208 209
      end

210
      # Creates a file upload field. If you are using file uploads then you will also need
211
      # to set the multipart option for the form tag:
212
      #
X
Xavier Noria 已提交
213
      #   <%= form_tag '/upload', :multipart => true do %>
214 215
      #     <label for="file">File to Upload</label> <%= file_field_tag "file" %>
      #     <%= submit_tag %>
P
Pratik Naik 已提交
216
      #   <% end %>
217
      #
218
      # The specified URL will then be passed a File object containing the selected file, or if the field
219
      # was left blank, a StringIO object.
220 221 222 223 224 225 226 227 228
      #
      # ==== Options
      # * Creates standard HTML attributes for the tag.
      # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
      #
      # ==== Examples
      #   file_field_tag 'attachment'
      #   # => <input id="attachment" name="attachment" type="file" />
      #
229 230
      #   file_field_tag 'avatar', :class => 'profile_input'
      #   # => <input class="profile_input" id="avatar" name="avatar" type="file" />
231 232 233 234 235 236 237 238
      #
      #   file_field_tag 'picture', :disabled => true
      #   # => <input disabled="disabled" id="picture" name="picture" type="file" />
      #
      #   file_field_tag 'resume', :value => '~/resume.doc'
      #   # => <input id="resume" name="resume" type="file" value="~/resume.doc" />
      #
      #   file_field_tag 'user_pic', :accept => 'image/png,image/gif,image/jpeg'
239
      #   # => <input accept="image/png,image/gif,image/jpeg" id="user_pic" name="user_pic" type="file" />
240 241 242
      #
      #   file_field_tag 'file', :accept => 'text/html', :class => 'upload', :value => 'index.html'
      #   # => <input accept="text/html" class="upload" id="file" name="file" type="file" value="index.html" />
243
      def file_field_tag(name, options = {})
244
        text_field_tag(name, nil, options.update("type" => "file"))
245 246
      end

247
      # Creates a password field, a masked text field that will hide the users input behind a mask character.
248
      #
249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273
      # ==== Options
      # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
      # * <tt>:size</tt> - The number of visible characters that will fit in the input.
      # * <tt>:maxlength</tt> - The maximum number of characters that the browser will allow the user to enter.
      # * Any other key creates standard HTML attributes for the tag.
      #
      # ==== Examples
      #   password_field_tag 'pass'
      #   # => <input id="pass" name="pass" type="password" />
      #
      #   password_field_tag 'secret', 'Your secret here'
      #   # => <input id="secret" name="secret" type="password" value="Your secret here" />
      #
      #   password_field_tag 'masked', nil, :class => 'masked_input_field'
      #   # => <input class="masked_input_field" id="masked" name="masked" type="password" />
      #
      #   password_field_tag 'token', '', :size => 15
      #   # => <input id="token" name="token" size="15" type="password" value="" />
      #
      #   password_field_tag 'key', nil, :maxlength => 16
      #   # => <input id="key" maxlength="16" name="key" type="password" />
      #
      #   password_field_tag 'confirm_pass', nil, :disabled => true
      #   # => <input disabled="disabled" id="confirm_pass" name="confirm_pass" type="password" />
      #
274 275
      #   password_field_tag 'pin', '1234', :maxlength => 4, :size => 6, :class => "pin_input"
      #   # => <input class="pin_input" id="pin" maxlength="4" name="pin" size="6" type="password" value="1234" />
276
      def password_field_tag(name = "password", value = nil, options = {})
277
        text_field_tag(name, value, options.update("type" => "password"))
278 279
      end

280 281 282
      # Creates a text input area; use a textarea for longer text inputs such as blog posts or descriptions.
      #
      # ==== Options
283
      # * <tt>:size</tt> - A string specifying the dimensions (columns by rows) of the textarea (e.g., "25x10").
284 285 286
      # * <tt>:rows</tt> - Specify the number of rows in the textarea
      # * <tt>:cols</tt> - Specify the number of columns in the textarea
      # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
287 288
      # * <tt>:escape</tt> - By default, the contents of the text input are HTML escaped.
      #   If you need unescaped contents, set this to false.
289 290 291 292 293 294 295 296
      # * Any other key creates standard HTML attributes for the tag.
      #
      # ==== Examples
      #   text_area_tag 'post'
      #   # => <textarea id="post" name="post"></textarea>
      #
      #   text_area_tag 'bio', @user.bio
      #   # => <textarea id="bio" name="bio">This is my biography.</textarea>
297
      #
298 299 300 301 302 303 304 305 306 307 308
      #   text_area_tag 'body', nil, :rows => 10, :cols => 25
      #   # => <textarea cols="25" id="body" name="body" rows="10"></textarea>
      #
      #   text_area_tag 'body', nil, :size => "25x10"
      #   # => <textarea name="body" id="body" cols="25" rows="10"></textarea>
      #
      #   text_area_tag 'description', "Description goes here.", :disabled => true
      #   # => <textarea disabled="disabled" id="description" name="description">Description goes here.</textarea>
      #
      #   text_area_tag 'comment', nil, :class => 'comment_input'
      #   # => <textarea class="comment_input" id="comment" name="comment"></textarea>
309
      def text_area_tag(name, content = nil, options = {})
310
        options = options.stringify_keys
311 312

        if size = options.delete("size")
313
          options["cols"], options["rows"] = size.split("x") if size.respond_to?(:split)
314
        end
315

316
        escape = options.delete("escape") { true }
317
        content = ERB::Util.html_escape(content) if escape
318

319
        content_tag :textarea, content.to_s.html_safe, { "name" => name, "id" => sanitize_to_id(name) }.update(options)
320 321
      end

322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342
      # Creates a check box form input tag.
      #
      # ==== Options
      # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
      # * Any other key creates standard HTML options for the tag.
      #
      # ==== Examples
      #   check_box_tag 'accept'
      #   # => <input id="accept" name="accept" type="checkbox" value="1" />
      #
      #   check_box_tag 'rock', 'rock music'
      #   # => <input id="rock" name="rock" type="checkbox" value="rock music" />
      #
      #   check_box_tag 'receive_email', 'yes', true
      #   # => <input checked="checked" id="receive_email" name="receive_email" type="checkbox" value="yes" />
      #
      #   check_box_tag 'tos', 'yes', false, :class => 'accept_tos'
      #   # => <input class="accept_tos" id="tos" name="tos" type="checkbox" value="yes" />
      #
      #   check_box_tag 'eula', 'accepted', false, :disabled => true
      #   # => <input disabled="disabled" id="eula" name="eula" type="checkbox" value="accepted" />
343
      def check_box_tag(name, value = "1", checked = false, options = {})
344
        html_options = { "type" => "checkbox", "name" => name, "id" => sanitize_to_id(name), "value" => value }.update(options.stringify_keys)
345
        html_options["checked"] = "checked" if checked
346
        tag :input, html_options
347 348
      end

349
      # Creates a radio button; use groups of radio buttons named the same to allow users to
350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367
      # select from a group of options.
      #
      # ==== Options
      # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
      # * Any other key creates standard HTML options for the tag.
      #
      # ==== Examples
      #   radio_button_tag 'gender', 'male'
      #   # => <input id="gender_male" name="gender" type="radio" value="male" />
      #
      #   radio_button_tag 'receive_updates', 'no', true
      #   # => <input checked="checked" id="receive_updates_no" name="receive_updates" type="radio" value="no" />
      #
      #   radio_button_tag 'time_slot', "3:00 p.m.", false, :disabled => true
      #   # => <input disabled="disabled" id="time_slot_300_pm" name="time_slot" type="radio" value="3:00 p.m." />
      #
      #   radio_button_tag 'color', "green", true, :class => "color_input"
      #   # => <input checked="checked" class="color_input" id="color_green" name="color" type="radio" value="green" />
368
      def radio_button_tag(name, value, checked = false, options = {})
369
        html_options = { "type" => "radio", "name" => name, "id" => "#{sanitize_to_id(name)}_#{sanitize_to_id(value)}", "value" => value }.update(options.stringify_keys)
370
        html_options["checked"] = "checked" if checked
371
        tag :input, html_options
372 373
      end

374
      # Creates a submit button with the text <tt>value</tt> as the caption.
375 376
      #
      # ==== Options
377 378
      # * <tt>:confirm => 'question?'</tt> - If present the unobtrusive JavaScript
      #   drivers will provide a prompt with the question specified. If the user accepts,
S
Stefan Penner 已提交
379
      #   the form is processed normally, otherwise no action is taken.
P
Pratik Naik 已提交
380
      # * <tt>:disabled</tt> - If true, the user will not be able to use this input.
381 382
      # * <tt>:disable_with</tt> - Value of this parameter will be used as the value for a
      #   disabled version of the submit button when the form is submitted. This feature is
S
Stefan Penner 已提交
383
      #   provided by the unobtrusive JavaScript driver.
384 385 386 387 388 389 390 391 392 393 394 395 396
      # * Any other key creates standard HTML options for the tag.
      #
      # ==== Examples
      #   submit_tag
      #   # => <input name="commit" type="submit" value="Save changes" />
      #
      #   submit_tag "Edit this article"
      #   # => <input name="commit" type="submit" value="Edit this article" />
      #
      #   submit_tag "Save edits", :disabled => true
      #   # => <input disabled="disabled" name="commit" type="submit" value="Save edits" />
      #
      #   submit_tag "Complete sale", :disable_with => "Please wait..."
V
Vijay Dev 已提交
397
      #   # => <input name="commit" data-disable-with="Please wait..." type="submit" value="Complete sale" />
398 399 400 401
      #
      #   submit_tag nil, :class => "form_submit"
      #   # => <input class="form_submit" name="commit" type="submit" />
      #
402
      #   submit_tag "Edit", :disable_with => "Editing...", :class => "edit_button"
V
Vijay Dev 已提交
403
      #   # => <input class="edit_button" data-disable_with="Editing..." name="commit" type="submit" value="Edit" />
S
Stefan Penner 已提交
404 405
      #
      #   submit_tag "Save", :confirm => "Are you sure?"
V
Vijay Dev 已提交
406
      #   # => <input name='commit' type='submit' value='Save' data-confirm="Are you sure?" />
S
Stefan Penner 已提交
407
      #
408
      def submit_tag(value = "Save changes", options = {})
409
        options = options.stringify_keys
410

411
        if disable_with = options.delete("disable_with")
J
Jeremy Kemper 已提交
412
          options["data-disable-with"] = disable_with
413
        end
414

415
        if confirm = options.delete("confirm")
416
          options["data-confirm"] = confirm
417
        end
418

419
        tag :input, { "type" => "submit", "name" => "commit", "value" => value }.update(options)
420
      end
421

R
Rizwan Reza 已提交
422 423 424 425
      # Creates a button element that defines a <tt>submit</tt> button,
      # <tt>reset</tt>button or a generic button which can be used in
      # JavaScript, for example. You can use the button tag as a regular
      # submit tag but it isn't supported in legacy browsers. However,
426 427
      # the button tag allows richer labels such as images and emphasis,
      # so this helper will also accept a block.
R
Rizwan Reza 已提交
428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443
      #
      # ==== Options
      # * <tt>:confirm => 'question?'</tt> - If present, the
      #   unobtrusive JavaScript drivers will provide a prompt with
      #   the question specified. If the user accepts, the form is
      #   processed normally, otherwise no action is taken.
      # * <tt>:disabled</tt> - If true, the user will not be able to
      #   use this input.
      # * <tt>:disable_with</tt> - Value of this parameter will be
      #   used as the value for a disabled version of the submit
      #   button when the form is submitted. This feature is provided
      #   by the unobtrusive JavaScript driver.
      # * Any other key creates standard HTML options for the tag.
      #
      # ==== Examples
      #   button_tag
444
      #   # => <button name="button" type="submit">Button</button>
R
Rizwan Reza 已提交
445
      #
446 447 448
      #   button_tag(:type => 'button') do
      #     content_tag(:strong, 'Ask me!')
      #   end
R
Rizwan Reza 已提交
449
      #   # => <button name="button" type="button">
450 451
      #   #     <strong>Ask me!</strong>
      #   #    </button>
R
Rizwan Reza 已提交
452 453
      #
      #   button_tag "Checkout", :disable_with => "Please wait..."
V
Vijay Dev 已提交
454
      #   # => <button data-disable-with="Please wait..." name="button" type="submit">Checkout</button>
R
Rizwan Reza 已提交
455
      #
456 457 458
      def button_tag(content_or_options = nil, options = nil, &block)
        options = content_or_options if block_given? && content_or_options.is_a?(Hash)
        options ||= {}
459
        options = options.stringify_keys
R
Rizwan Reza 已提交
460 461 462 463 464 465 466 467 468

        if disable_with = options.delete("disable_with")
          options["data-disable-with"] = disable_with
        end

        if confirm = options.delete("confirm")
          options["data-confirm"] = confirm
        end

469
        options.reverse_merge! 'name' => 'button', 'type' => 'submit'
R
Rizwan Reza 已提交
470

471
        content_tag :button, content_or_options || 'Button', options, &block
R
Rizwan Reza 已提交
472 473
      end

474 475
      # Displays an image which when clicked will submit the form.
      #
476
      # <tt>source</tt> is passed to AssetTagHelper#path_to_image
477 478
      #
      # ==== Options
479 480 481
      # * <tt>:confirm => 'question?'</tt> - This will add a JavaScript confirm
      #   prompt with the question specified. If the user accepts, the form is
      #   processed normally, otherwise no action is taken.
482 483 484 485 486 487 488
      # * <tt>:disabled</tt> - If set to true, the user will not be able to use this input.
      # * Any other key creates standard HTML options for the tag.
      #
      # ==== Examples
      #   image_submit_tag("login.png")
      #   # => <input src="/images/login.png" type="image" />
      #
489
      #   image_submit_tag("purchase.png", :disabled => true)
490 491
      #   # => <input disabled="disabled" src="/images/purchase.png" type="image" />
      #
492 493
      #   image_submit_tag("search.png", :class => 'search_button')
      #   # => <input class="search_button" src="/images/search.png" type="image" />
494
      #
495 496
      #   image_submit_tag("agree.png", :disabled => true, :class => "agree_disagree_button")
      #   # => <input class="agree_disagree_button" disabled="disabled" src="/images/agree.png" type="image" />
497
      def image_submit_tag(source, options = {})
498
        options = options.stringify_keys
499 500

        if confirm = options.delete("confirm")
501
          options["data-confirm"] = confirm
502 503
        end

504
        tag :input, { "type" => "image", "src" => path_to_image(source) }.update(options)
505
      end
506 507 508 509

      # Creates a field set for grouping HTML form elements.
      #
      # <tt>legend</tt> will become the fieldset's title (optional as per W3C).
A
Andrew Kaspick 已提交
510
      # <tt>options</tt> accept the same values as tag.
511
      #
L
lifo 已提交
512
      # ==== Examples
513
      #   <%= field_set_tag do %>
514 515 516 517
      #     <p><%= text_field_tag 'name' %></p>
      #   <% end %>
      #   # => <fieldset><p><input id="name" name="name" type="text" /></p></fieldset>
      #
518
      #   <%= field_set_tag 'Your details' do %>
519 520 521
      #     <p><%= text_field_tag 'name' %></p>
      #   <% end %>
      #   # => <fieldset><legend>Your details</legend><p><input id="name" name="name" type="text" /></p></fieldset>
A
Andrew Kaspick 已提交
522
      #
523
      #   <%= field_set_tag nil, :class => 'format' do %>
A
Andrew Kaspick 已提交
524 525 526 527
      #     <p><%= text_field_tag 'name' %></p>
      #   <% end %>
      #   # => <fieldset class="format"><p><input id="name" name="name" type="text" /></p></fieldset>
      def field_set_tag(legend = nil, options = nil, &block)
W
wycats 已提交
528 529
        output = tag(:fieldset, options, true)
        output.safe_concat(content_tag(:legend, legend)) unless legend.blank?
530
        output.concat(capture(&block)) if block_given?
W
wycats 已提交
531
        output.safe_concat("</fieldset>")
532
      end
533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550

      # Creates a text field of type "search".
      #
      # ==== Options
      # * Accepts the same options as text_field_tag.
      def search_field_tag(name, value = nil, options = {})
        text_field_tag(name, value, options.stringify_keys.update("type" => "search"))
      end

      # Creates a text field of type "tel".
      #
      # ==== Options
      # * Accepts the same options as text_field_tag.
      def telephone_field_tag(name, value = nil, options = {})
        text_field_tag(name, value, options.stringify_keys.update("type" => "tel"))
      end
      alias phone_field_tag telephone_field_tag

551 552 553 554 555 556 557 558
      # Creates a text field of type "date".
      #
      # ==== Options
      # * Accepts the same options as text_field_tag.
      def date_field_tag(name, value = nil, options = {})
        text_field_tag(name, value, options.stringify_keys.update("type" => "date"))
      end

559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586
      # Creates a text field of type "url".
      #
      # ==== Options
      # * Accepts the same options as text_field_tag.
      def url_field_tag(name, value = nil, options = {})
        text_field_tag(name, value, options.stringify_keys.update("type" => "url"))
      end

      # Creates a text field of type "email".
      #
      # ==== Options
      # * Accepts the same options as text_field_tag.
      def email_field_tag(name, value = nil, options = {})
        text_field_tag(name, value, options.stringify_keys.update("type" => "email"))
      end

      # Creates a number field.
      #
      # ==== Options
      # * <tt>:min</tt> - The minimum acceptable value.
      # * <tt>:max</tt> - The maximum acceptable value.
      # * <tt>:in</tt> - A range specifying the <tt>:min</tt> and
      #   <tt>:max</tt> values.
      # * <tt>:step</tt> - The acceptable value granularity.
      # * Otherwise accepts the same options as text_field_tag.
      #
      # ==== Examples
      #   number_field_tag 'quantity', nil, :in => 1...10
587
      #   # => <input id="quantity" name="quantity" min="1" max="9" type="number" />
588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603
      def number_field_tag(name, value = nil, options = {})
        options = options.stringify_keys
        options["type"] ||= "number"
        if range = options.delete("in") || options.delete("within")
          options.update("min" => range.min, "max" => range.max)
        end
        text_field_tag(name, value, options)
      end

      # Creates a range form element.
      #
      # ==== Options
      # * Accepts the same options as number_field_tag.
      def range_field_tag(name, value = nil, options = {})
        number_field_tag(name, value, options.stringify_keys.update("type" => "range"))
      end
604

605
      # Creates the hidden UTF8 enforcer tag. Override this method in a helper
606
      # to customize the tag.
607
      def utf8_enforcer_tag
608
        tag(:input, :type => "hidden", :name => "utf8", :value => "&#x2713;".html_safe)
609 610
      end

611
      private
612
        def html_options_for_form(url_for_options, options)
S
Santiago Pastorino 已提交
613
          options.stringify_keys.tap do |html_options|
614
            html_options["enctype"] = "multipart/form-data" if html_options.delete("multipart")
615
            # The following URL is unescaped, this is just a hash of options, and it is the
R
R.T. Lechow 已提交
616
            # responsibility of the caller to escape all the values.
617
            html_options["action"]  = url_for(url_for_options)
W
wycats 已提交
618
            html_options["accept-charset"] = "UTF-8"
619
            html_options["data-remote"] = true if html_options.delete("remote")
620
            html_options["authenticity_token"] = html_options.delete("authenticity_token") if html_options.has_key?("authenticity_token")
621 622
          end
        end
623

624
        def extra_tags_for_form(html_options)
625
          authenticity_token = html_options.delete("authenticity_token")
W
wycats 已提交
626 627 628
          method = html_options.delete("method").to_s

          method_tag = case method
629
            when /^get$/i # must be case-insensitive, but can't use downcase as might be nil
630 631 632 633
              html_options["method"] = "get"
              ''
            when /^post$/i, "", nil
              html_options["method"] = "post"
634
              token_tag(authenticity_token)
635 636
            else
              html_options["method"] = "post"
R
Rafael Mendonça França 已提交
637
              method_tag(method) + token_tag(authenticity_token)
638
          end
W
wycats 已提交
639

640
          tags = utf8_enforcer_tag << method_tag
W
wycats 已提交
641
          content_tag(:div, tags, :style => 'margin:0;padding:0;display:inline')
642
        end
643

644 645
        def form_tag_html(html_options)
          extra_tags = extra_tags_for_form(html_options)
646
          tag(:form, html_options, true) + extra_tags
647
        end
648

649 650
        def form_tag_in_block(html_options, &block)
          content = capture(&block)
651
          output = form_tag_html(html_options)
W
wycats 已提交
652 653
          output << content
          output.safe_concat("</form>")
654
        end
655

656 657 658 659
        # see http://www.w3.org/TR/html4/types.html#type-name
        def sanitize_to_id(name)
          name.to_s.gsub(']','').gsub(/[^-a-zA-Z0-9:.]/, "_")
        end
660 661 662
    end
  end
end