Module: ActionView::Helpers::DynamicForm

Defined in:
lib/action_view/helpers/dynamic_form.rb

Overview

The Active Record Helper makes it easier to create forms for records kept in instance variables. The most far-reaching is the form method that creates a complete form for all the basic content types of the record (not associations or aggregations, though). This is a great way of making the record quickly available for editing, but likely to prove lackluster for a complicated real-world form. In that case, it’s better to use the input method and the specialized form methods in classes/ActionView/Helpers/FormHelper.html

Defined Under Namespace

Modules: FormBuilderMethods, InstanceTagMethods

Instance Method Summary collapse

Instance Method Details

#error_message_on(object, method, *args) ⇒ Object

Returns a string containing the error message attached to the method on the object if one exists. This error message is wrapped in a DIV tag by default or with :html_tag if specified, which can be extended to include a :prepend_text and/or :append_text (to properly explain the error), and a :css_class to style it accordingly. object should either be the name of an instance variable or the actual object. The method can be passed in either as a string or a symbol. As an example, let’s say you have a model @post that has an error message on the title attribute:

<%= error_message_on "post", "title" %>
# => <div class="formError">can't be empty</div>

<%= error_message_on @post, :title %>
# => <div class="formError">can't be empty</div>

<%= error_message_on "post", "title",
    :prepend_text => "Title simply ",
    :append_text => " (or it won't work).",
    :html_tag => "span",
    :css_class => "inputError" %>
# => <span class="inputError">Title simply can't be empty (or it won't work).</span>


109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
# File 'lib/action_view/helpers/dynamic_form.rb', line 109

def error_message_on(object, method, *args)
  options = args.extract_options!
  unless args.empty?
    ActiveSupport::Deprecation.warn('error_message_on takes an option hash instead of separate ' +
      'prepend_text, append_text, html_tag, and css_class arguments', caller)

    options[:prepend_text] = args[0] || ''
    options[:append_text] = args[1] || ''
    options[:html_tag] = args[2] || 'div'
    options[:css_class] = args[3] || 'formError'
  end
  options.reverse_merge!(:prepend_text => '', :append_text => '', :html_tag => 'div', :css_class => 'formError')

  object = convert_to_model(object)

  if (obj = (object.respond_to?(:errors) ? object : instance_variable_get("@#{object}"))) &&
    (errors = obj.errors[method]).presence
    (options[:html_tag],
      (options[:prepend_text].html_safe << errors.first).safe_concat(options[:append_text]),
      :class => options[:css_class]
    )
  else
    ''
  end
end

#error_messages_for(*params) ⇒ Object

Returns a string with a DIV containing all of the error messages for the objects located as instance variables by the names given. If more than one object is specified, the errors for the objects are displayed in the order that the object names are provided.

This DIV can be tailored by the following options:

  • :header_tag - Used for the header of the error div (default: “h2”).

  • :id - The id of the error div (default: “errorExplanation”).

  • :class - The class of the error div (default: “errorExplanation”).

  • :object - The object (or array of objects) for which to display errors, if you need to escape the instance variable convention.

  • :object_name - The object name to use in the header, or any text that you prefer. If :object_name is not set, the name of the first object will be used.

  • :header_message - The message in the header of the error div. Pass nil or an empty string to avoid the header message altogether. (Default: “X errors prohibited this object from being saved”).

  • :message - The explanation message after the header message and before the error list. Pass nil or an empty string to avoid the explanation message altogether. (Default: “There were problems with the following fields:”).

To specify the display for one object, you simply provide its name as a parameter. For example, for the @user model:

error_messages_for 'user'

You can also supply an object:

error_messages_for @user

This will use the last part of the model name in the presentation. For instance, if this is a MyKlass::User object, this will use “user” as the name in the String. This is taken from MyKlass::User.model_name.human, which can be overridden.

To specify more than one object, you simply list them; optionally, you can add an extra :object_name parameter, which will be the name used in the header message:

error_messages_for 'user_common', 'user', :object_name => 'user'

You can also use a number of objects, which will have the same naming semantics as a single object.

error_messages_for @user, @post

If the objects cannot be located as instance variables, you can add an extra :object parameter which gives the actual object (or array of objects to use):

error_messages_for 'user', :object => @question.user

NOTE: This is a pre-packaged presentation of the errors with embedded strings and a certain HTML structure. If what you need is significantly different from the default presentation, it makes plenty of sense to access the object.errors instance yourself and set it up. View the source of this method to see how easy it is.



186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
# File 'lib/action_view/helpers/dynamic_form.rb', line 186

def error_messages_for(*params)
  options = params.extract_options!.symbolize_keys

  objects = Array.wrap(options.delete(:object) || params).map do |object|
    object = instance_variable_get("@#{object}") unless object.respond_to?(:to_model)
    object = convert_to_model(object)

    if object.class.respond_to?(:model_name)
      options[:object_name] ||= object.class.model_name.human.downcase
    end

    object
  end

  objects.compact!
  count = objects.inject(0) {|sum, object| sum + object.errors.count }

  unless count.zero?
    html = {}
    [:id, :class].each do |key|
      if options.include?(key)
        value = options[key]
        html[key] = value unless value.blank?
      else
        html[key] = 'errorExplanation'
      end
    end
    options[:object_name] ||= params.first

    I18n.with_options :locale => options[:locale], :scope => [:activerecord, :errors, :template] do |locale|
      header_message = if options.include?(:header_message)
        options[:header_message]
      else
        locale.t :header, :count => count, :model => options[:object_name].to_s.gsub('_', ' ')
      end

      message = options.include?(:message) ? options[:message] : locale.t(:body)

      error_messages = objects.sum do |object|
        object.errors.full_messages.map do |msg|
          (:li, msg)
        end
      end.join.html_safe

      contents = ''
      contents << (options[:header_tag] || :h2, header_message) unless header_message.blank?
      contents << (:p, message) unless message.blank?
      contents << (:ul, error_messages)

      (:div, contents.html_safe, html)
    end
  else
    ''
  end
end

#form(record_name, options = {}) {|contents| ... } ⇒ Object

Returns an entire form with all needed input tags for a specified Active Record object. For example, if @post has attributes named title of type VARCHAR and body of type TEXT then

form("post")

would yield a form like the following (modulus formatting):

<form action='/posts/create' method='post'>
  <p>
    <label for="post_title">Title</label><br />
    <input id="post_title" name="post[title]" size="30" type="text" value="Hello World" />
  </p>
  <p>
    <label for="post_body">Body</label><br />
    <textarea cols="40" id="post_body" name="post[body]" rows="20"></textarea>
  </p>
  <input name="commit" type="submit" value="Create" />
</form>

It’s possible to specialize the form builder by using a different action name and by supplying another block renderer. For example, if @entry has an attribute message of type VARCHAR then

form("entry",
  :action => "sign",
  :input_block => Proc.new { |record, column|
    "#{column.human_name}: #{input(record, column.name)}<br />"
})

would yield a form like the following (modulus formatting):

<form action="/entries/sign" method="post">
  Message:
  <input id="entry_message" name="entry[message]" size="30" type="text" /><br />
  <input name="commit" type="submit" value="Sign" />
</form>

It’s also possible to add additional content to the form by giving it a block, such as:

form("entry", :action => "sign") do |form|
  form << ("b", "Department")
  form << collection_select("department", "id", @departments, "id", "name")
end

The following options are available:

  • :action - The action used when submitting the form (default: create if a new record, otherwise update).

  • :input_block - Specialize the output using a different block, see above.

  • :method - The method used when submitting the form (default: post).

  • :multipart - Whether to change the enctype of the form to “multipart/form-data”, used when uploading a file (default: false).

  • :submit_value - The text of the submit button (default: “Create” if a new record, otherwise “Update”).

Yields:

  • (contents)


72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
# File 'lib/action_view/helpers/dynamic_form.rb', line 72

def form(record_name, options = {})
  record = instance_variable_get("@#{record_name}")
  record = convert_to_model(record)

  options = options.symbolize_keys
  options[:action] ||= record.persisted? ? "update" : "create"
  action = url_for(:action => options[:action], :id => record)

  submit_value = options[:submit_value] || options[:action].gsub(/[^\w]/, '').capitalize

  contents = form_tag({:action => action}, :method =>(options[:method] || 'post'), :enctype => options[:multipart] ? 'multipart/form-data': nil)
  contents.safe_concat hidden_field(record_name, :id) if record.persisted?
  contents.safe_concat all_input_tags(record, record_name, options)
  yield contents if block_given?
  contents.safe_concat submit_tag(submit_value)
  contents.safe_concat('</form>')
end

#input(record_name, method, options = {}) ⇒ Object

Returns a default input tag for the type of object returned by the method. For example, if @post has an attribute title mapped to a VARCHAR column that holds “Hello World”:

input("post", "title")
# => <input id="post_title" name="post[title]" size="30" type="text" value="Hello World" />


18
19
20
# File 'lib/action_view/helpers/dynamic_form.rb', line 18

def input(record_name, method, options = {})
  InstanceTag.new(record_name, method, self).to_tag(options)
end