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
-
#error_message_on(object, method, *args) ⇒ Object
Returns a string containing the error message attached to the
method
on theobject
if one exists. -
#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. -
#form(record_name, options = {}) {|contents| ... } ⇒ Object
Returns an entire form with all needed input tags for a specified Active Record object.
-
#input(record_name, method, options = {}) ⇒ Object
Returns a default input tag for the type of object returned by the method.
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 (object, method, *args) = args. 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) [:prepend_text] = args[0] || '' [:append_text] = args[1] || '' [:html_tag] = args[2] || 'div' [:css_class] = args[3] || 'formError' end .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 content_tag([:html_tag], ([:prepend_text].html_safe << errors.first).safe_concat([:append_text]), :class => [: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. Passnil
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. Passnil
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:
'user'
You can also supply an object:
@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:
'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.
@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):
'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 (*params) = params..symbolize_keys objects = Array.wrap(.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) [: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 .include?(key) value = [key] html[key] = value unless value.blank? else html[key] = 'errorExplanation' end end [:object_name] ||= params.first I18n. :locale => [:locale], :scope => [:errors, :template] do |locale| = if .include?(:header_message) [:header_message] else locale.t :header, :count => count, :model => [:object_name].to_s.gsub('_', ' ') end = .include?(:message) ? [:message] : locale.t(:body) = objects.sum do |object| object.errors..map do |msg| content_tag(:li, msg) end end.join.html_safe contents = '' contents << content_tag([:header_tag] || :h2, ) unless .blank? contents << content_tag(:p, ) unless .blank? contents << content_tag(:ul, ) content_tag(: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 << content_tag("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, otherwiseupdate
). -
: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”).
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, = {}) record = instance_variable_get("@#{record_name}") record = convert_to_model(record) = .symbolize_keys [:action] ||= record.persisted? ? "update" : "create" action = url_for(:action => [:action], :id => record) submit_value = [:submit_value] || [:action].gsub(/[^\w]/, '').capitalize contents = form_tag({:action => action}, :method =>([:method] || 'post'), :enctype => [:multipart] ? 'multipart/form-data': nil) contents.safe_concat hidden_field(record_name, :id) if record.persisted? contents.safe_concat (record, record_name, ) 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, = {}) InstanceTag.new(record_name, method, self).to_tag() end |