require 'padrino-helpers/form_helpers/errors'
require 'padrino-helpers/form_helpers/options'
require 'padrino-helpers/form_helpers/security'
module Padrino
module Helpers
##
# Helpers related to producing form related tags and inputs into templates.
#
module FormHelpers
def self.included(base)
base.send(:include, FormHelpers::Errors)
base.send(:include, FormHelpers::Options)
base.send(:include, FormHelpers::Security)
end
##
# Constructs a form for object using given or default form_builder.
#
# @param [Object] object
# The object for which the form is being built.
# @param [String] URL
# The url this form will submit to.
# @param [Hash] options
# The settings associated with this form.
# Accepts a :namespace option that will be prepended to the id attributes of the form's elements.
# Also accepts HTML options.
# @option settings [String] :builder ("StandardFormBuilder")
# The FormBuilder class to use such as StandardFormBuilder.
# @option settings [Symbol] :as
# Sets custom form object name.
# @param [Proc] block
# The fields and content inside this form.
#
# @yield [AbstractFormBuilder] The form builder used to compose fields.
#
# @return [String] The html object-backed form with the specified options and input fields.
#
# @example
# form_for :user, '/register' do |f| ... end
# form_for @user, '/register', :id => 'register' do |f| ... end
# form_for @user, '/register', :as => :customer do |f| ... end
#
def form_for(object, url, options={}, &block)
instance = builder_instance(object, options)
# this can erect instance.multipart flag if the block calls instance.file_field
html = capture_html(instance, &block)
options = { :multipart => instance.multipart }.update(options.except(:namespace, :as))
form_tag(url, options) { html }
end
##
# Constructs form fields for an object using given or default form_builder.
# Used within an existing form to allow alternate objects within one form.
#
# @param [Object] object
# The object for which the fields are being built.
# @param [Hash] options
# The settings associated with these fields. Accepts HTML options.
# @param [Proc] block
# The content inside this set of fields.
#
# @return [String] The html fields with the specified options.
#
# @example
# fields_for @user.assignment do |assignment| ... end
# fields_for :assignment do |assigment| ... end
#
def fields_for(object, options={}, &block)
instance = builder_instance(object, options)
fields_html = capture_html(instance, &block)
fields_html << instance.hidden_field(:id) if instance.send(:nested_object_id)
concat_content fields_html
end
##
# Constructs a form without object based on options.
#
# @param [String] url
# The URL this form will submit to.
# @param [Hash] options
# The html options associated with this form.
# @param [Proc] block
# The fields and content inside this form.
#
# @return [String] The HTML form with the specified options and input fields.
#
# @example
# form_tag '/register', :class => "registration_form" do ... end
#
def form_tag(url, options={}, &block)
options = {
:action => url,
:protect_from_csrf => is_protected_from_csrf?,
'accept-charset' => 'UTF-8'
}.update(options)
options[:enctype] = 'multipart/form-data' if options.delete(:multipart)
if (desired_method = options[:method]) =~ /get/i
options.delete(:protect_from_csrf)
else
options[:method] = 'post'
end
inner_form_html = hidden_form_method_field(desired_method)
inner_form_html << csrf_token_field if options.delete(:protect_from_csrf)
concat_content content_tag(:form, inner_form_html << capture_html(&block), options)
end
##
# Returns the hidden method field for 'put' and 'delete' forms.
# Only 'get' and 'post' are allowed within browsers;
# 'put' and 'delete' are just specified using hidden fields with form action still 'put'.
#
# @param [String] desired_method
# The method this hidden field represents (i.e put or delete).
#
# @return [String] The hidden field representing the +desired_method+ for the form.
#
# @example
# # Generate:
# hidden_form_method_field('delete')
#
def hidden_form_method_field(desired_method)
return ActiveSupport::SafeBuffer.new if desired_method.blank? || desired_method.to_s =~ /get|post/i
hidden_field_tag(:_method, :value => desired_method)
end
##
# Constructs a field_set to group fields with given options.
#
# @overload field_set_tag(legend=nil, options={}, &block)
# @param [String] legend The legend caption for the fieldset
# @param [Hash] options The html options for the fieldset.
# @param [Proc] block The content inside the fieldset.
# @overload field_set_tag(options={}, &block)
# @param [Hash] options The html options for the fieldset.
# @param [Proc] block The content inside the fieldset.
#
# @return [String] The html for the fieldset tag based on given +options+.
#
# @example
# field_set_tag(:class => "office-set") { }
# field_set_tag("Office", :class => 'office-set') { }
#
def field_set_tag(*args, &block)
options = args.extract_options!
legend_text = args.first
legend_html = legend_text.blank? ? ActiveSupport::SafeBuffer.new : content_tag(:legend, legend_text)
concat_content content_tag(:fieldset, legend_html << capture_html(&block), options)
end
##
# Constructs a label tag from the given options.
#
# @param [String] name
# The name of the field to label.
# @param [Hash] options
# The html options for this label.
# @option options :caption
# The caption for this label.
# @param [Proc] block
# The content to be inserted into the label.
#
# @return [String] The html for this label with the given +options+.
#
# @example
# label_tag :username, :class => 'long-label'
# label_tag :username, :class => 'long-label' do ... end
#
def label_tag(name, options={}, &block)
options = { :caption => "#{name.to_s.humanize}: ", :for => name }.update(options)
caption_text = ActiveSupport::SafeBuffer.new << options.delete(:caption)
caption_text << "* ".html_safe if options.delete(:required)
if block_given?
concat_content content_tag(:label, caption_text << capture_html(&block), options)
else
content_tag(:label, caption_text, options)
end
end
##
# Creates a text field input with the given name and options.
#
# @macro [new] text_field
# @param [Symbol] name
# The name of the input to create.
# @param [Hash] options
# The HTML options to include in this field.
#
# @option options [String] :id
# Specifies a unique identifier for the field.
# @option options [String] :class
# Specifies the stylesheet class of the field.
# @option options [String] :name
# Specifies the name of the field.
# @option options [String] :accesskey
# Specifies a shortcut key to access the field.
# @option options [Integer] :tabindex
# Specifies the tab order of the field.
# @option options [Integer] :maxlength
# Specifies the maximum length, in characters, of the field.
# @option options [Integer] :size
# Specifies the width, in characters, of the field.
# @option options [String] :placeholder
# Specifies a short hint that describes the expected value of the field.
# @option options [Boolean] :hidden
# Specifies whether or not the field is hidden from view.
# @option options [Boolean] :spellcheck
# Specifies whether or not the field should have it's spelling and grammar checked for errors.
# @option options [Boolean] :draggable
# Specifies whether or not the field is draggable. (true, false, :auto).
# @option options [String] :pattern
# Specifies the regular expression pattern that the field's value is checked against.
# @option options [Symbol] :autocomplete
# Specifies whether or not the field should have autocomplete enabled. (:on, :off).
# @option options [Boolean] :autofocus
# Specifies whether or not the field should automatically get focus when the page loads.
# @option options [Boolean] :required
# Specifies whether or not the field is required to be completed before the form is submitted.
# @option options [Boolean] :readonly
# Specifies whether or not the field is read only.
# @option options [Boolean] :disabled
# Specifies whether or not the field is disabled.
#
# @return [String]
# Generated HTML with specified +options+.
#
# @example
# text_field_tag :first_name, :maxlength => 40, :required => true
# # =>
#
# text_field_tag :last_name, :class => 'string', :size => 40
# # =>
#
# text_field_tag :username, :placeholder => 'Your Username'
# # =>
#
def text_field_tag(name, options={})
input_tag(:text, { :name => name }.update(options))
end
##
# Creates a number field input with the given name and options.
#
# @macro [new] number_field
# @param [Symbol] name
# The name of the input to create.
# @param [Hash] options
# The HTML options to include in this field.
#
# @option options [String] :id
# Specifies a unique identifier for the field.
# @option options [String] :class
# Specifies the stylesheet class of the field.
# @option options [String] :name
# Specifies the name of the field.
# @option options [String] :accesskey
# Specifies a shortcut key to access the field.
# @option options [Integer] :tabindex
# Specifies the tab order of the field.
# @option options [Integer] :min
# Specifies the minimum value of the field.
# @option options [Integer] :max
# Specifies the maximum value of the field.
# @option options [Integer] :step
# Specifies the legal number intervals of the field.
# @option options [Boolean] :hidden
# Specifies whether or not the field is hidden from view.
# @option options [Boolean] :spellcheck
# Specifies whether or not the field should have it's spelling and grammar checked for errors.
# @option options [Boolean] :draggable
# Specifies whether or not the field is draggable. (true, false, :auto).
# @option options [String] :pattern
# Specifies the regular expression pattern that the field's value is checked against.
# @option options [Symbol] :autocomplete
# Specifies whether or not the field should have autocomplete enabled. (:on, :off).
# @option options [Boolean] :autofocus
# Specifies whether or not the field should automatically get focus when the page loads.
# @option options [Boolean] :required
# Specifies whether or not the field is required to be completeled before the form is submitted.
# @option options [Boolean] :readonly
# Specifies whether or not the field is read only.
# @option options [Boolean] :disabled
# Specifies whether or not the field is disabled.
#
# @return [String]
# Generated HTML with specified +options+.
#
# @example
# number_field_tag :quantity, :class => 'numeric'
# # =>
#
# number_field_tag :zip_code, :pattern => /[0-9]{5}/
# # =>
#
# number_field_tag :credit_card, :autocomplete => :off
# # =>
#
# number_field_tag :age, :min => 18, :max => 120, :step => 1
# # =>
#
def number_field_tag(name, options={})
input_tag(:number, { :name => name }.update(options))
end
##
# Creates a telephone field input with the given name and options.
#
# @macro text_field
#
# @example
# telephone_field_tag :phone_number, :class => 'string'
# # =>
#
# telephone_field_tag :cell_phone, :tabindex => 1
# telephone_field_tag :work_phone, :tabindex => 2
# telephone_field_tag :home_phone, :tabindex => 3
#
# # =>
# # =>
# # =>
#
def telephone_field_tag(name, options={})
input_tag(:tel, { :name => name }.update(options))
end
alias_method :phone_field_tag, :telephone_field_tag
##
# Creates an email field input with the given name and options.
#
# @macro text_field
#
# @example
# email_field_tag :email, :placeholder => 'you@example.com'
# # =>
#
# email_field_tag :email, :value => 'padrinorb@gmail.com', :readonly => true
# # =>
#
def email_field_tag(name, options={})
input_tag(:email, { :name => name }.update(options))
end
##
# Creates a search field input with the given name and options.
#
# @macro text_field
#
# @example
# search_field_tag :search, :placeholder => 'Search this website...'
# # =>
#
# search_field_tag :search, :maxlength => 15, :class => ['search', 'string']
# # =>
#
# search_field_tag :search, :id => 'search'
# # =>
#
# search_field_tag :search, :autofocus => true
# # =>
#
def search_field_tag(name, options={})
input_tag(:search, { :name => name }.update(options))
end
##
# Creates a URL field input with the given name and options.
#
# @macro text_field
#
# @example
# url_field_tag :favorite_website, :placeholder => 'http://padrinorb.com'
#
#
# url_field_tag :home_page, :class => 'string url'
#
#
def url_field_tag(name, options={})
input_tag(:url, { :name => name }.update(options))
end
##
# Constructs a hidden field input from the given options.
#
# @example
# hidden_field_tag :session_key, :value => "__secret__"
#
def hidden_field_tag(name, options={})
input_tag(:hidden, { :name => name }.update(options))
end
##
# Constructs a text area input from the given options.
#
# @example
# text_area_tag :username, :class => 'long', :value => "Demo?"
#
def text_area_tag(name, options={})
inner_html = TagHelpers::NEWLINE + options.delete(:value).to_s
options = { :name => name, :rows => "", :cols => "" }.update(options)
content_tag(:textarea, inner_html, options)
end
##
# Constructs a password field input from the given options.
#
# @example
# password_field_tag :password, :class => 'long'
#
# @api public
def password_field_tag(name, options={})
input_tag(:password, { :name => name }.update(options))
end
##
# Constructs a check_box from the given options.
#
# @example
# check_box_tag :remember_me, :value => 'Yes'
#
def check_box_tag(name, options={})
input_tag(:checkbox, { :name => name, :value => '1' }.update(options))
end
##
# Constructs a radio_button from the given options.
#
# @example
# radio_button_tag :remember_me, :value => 'true'
#
def radio_button_tag(name, options={})
input_tag(:radio, { :name => name }.update(options))
end
##
# Constructs a file field input from the given options.
#
# @example
# file_field_tag :photo, :class => 'long'
#
# @api public
def file_field_tag(name, options={})
name = "#{name}[]" if options[:multiple]
input_tag(:file, { :name => name }.update(options))
end
##
# Constructs a select from the given options.
#
# @example
# options = [['caption', 'value'], ['Green', 'green1'], ['Blue', 'blue1'], ['Black', "black1"]]
# options = ['option', 'red', 'yellow' ]
# select_tag(:favorite_color, :options => ['red', 'yellow'], :selected => 'green1')
# select_tag(:country, :collection => @countries, :fields => [:name, :code], :include_blank => 'None')
#
# # Optgroups can be generated using :grouped_options => (Hash or nested Array)
# grouped_options = [['Friends',['Yoda',['Obiwan',1]]],['Enemies',['Palpatine',['Darth Vader',3]]]]
# grouped_options = {'Friends' => ['Yoda',['Obiwan',1]],'Enemies' => ['Palpatine',['Darth Vader',3]]}
# select_tag(:color, :grouped_options => [['warm',['red','yellow']],['cool',['blue', 'purple']]])
#
# # Optgroups can be generated using the rails-style attribute hash.
# grouped_options = {
# "Friends" => ["Yoda", ["Obiwan", 2, {:magister => 'no'}], {:lame => 'yes'}],
# "Enemies" => [["Palpatine", "Palpatine", {:scary => 'yes', :old => 'yes'}], ["Darth Vader", 3, {:disabled => true}]]
# }
# select_tag(:name, :grouped_options => grouped_options)
#
# @param [String] name
# The name of the input field.
# @param [Hash] options
# The html options for the input field.
# @option options [Array] :options
# Explicit options to display in the select. Can be strings or string tuples.
# @option options [Array] :grouped_options
# List of options for each group in the select. See examples for details.
# @option options [Array