# Note for posterity: # # Here we have two core customizations on top of Formtastic. First, this allows # you to build forms in the AA DSL without dealing with the HTML return value of # individual form methods (hence the +form_buffers+ object). Second, this provides # an intuitive way to build has_many associated records in the same form. # module ActiveAdmin class FormBuilder < ::Formtastic::FormBuilder attr_reader :form_buffers def initialize(*args) @form_buffers = ["".html_safe] super end def inputs(*args, &block) @use_form_buffer = block_given? form_buffers.last << with_new_form_buffer{ super } end # If this `input` call is inside a `inputs` block, add the content # to the form buffer. Else, return it directly. def input(method, *args) content = with_new_form_buffer{ super } @use_form_buffer ? form_buffers.last << content : content end def cancel_link(url = {action: "index"}, html_options = {}, li_attrs = {}) li_attrs[:class] ||= "cancel" li_content = template.link_to I18n.t('active_admin.cancel'), url, html_options form_buffers.last << template.content_tag(:li, li_content, li_attrs) end def actions(*args, &block) form_buffers.last << with_new_form_buffer do block_given? ? super : super{ commit_action_with_cancel_link } end end def action(*args) form_buffers.last << with_new_form_buffer{ super } end def commit_action_with_cancel_link action(:submit) cancel_link end def has_many(assoc, options = {}, &block) # remove options that should not render as attributes custom_settings = :new_record, :allow_destroy, :heading, :sortable builder_options = {new_record: true}.merge! options.slice *custom_settings options = {for: assoc }.merge! options.except *custom_settings options[:class] = [options[:class], "inputs has_many_fields"].compact.join(' ') # Add Delete Links form_block = proc do |has_many_form| index = parent_child_index options[:parent] if options[:parent] contents = block.call has_many_form, index if has_many_form.object.new_record? contents << template.content_tag(:li) do template.link_to I18n.t('active_admin.has_many_remove'), "#", class: 'button has_many_remove' end elsif builder_options[:allow_destroy] has_many_form.input :_destroy, as: :boolean, wrapper_html: {class: 'has_many_delete'}, label: I18n.t('active_admin.has_many_delete') end if builder_options[:sortable] has_many_form.input builder_options[:sortable], as: :hidden contents << template.content_tag(:li, class: 'handle') do Iconic.icon :move_vertical end end contents end # make sure that the sortable children sorted in stable ascending order if column = builder_options[:sortable] children = object.send(assoc) children = children.sort_by {|o| [o.send(column), o.id]} options[:for] = [assoc, children] end html = without_wrapper do unless builder_options.key?(:heading) && !builder_options[:heading] form_buffers.last << template.content_tag(:h3) do builder_options[:heading] || object.class.reflect_on_association(assoc).klass.model_name.human(count: ::ActiveAdmin::Helpers::I18n::PLURAL_MANY_COUNT) end end inputs options, &form_block form_buffers.last << js_for_has_many(assoc, form_block, template, builder_options[:new_record], options[:class]) if builder_options[:new_record] form_buffers.last end form_buffers.last << if @already_in_an_inputs_block template.content_tag :li, html, class: "has_many_container #{assoc}", 'data-sortable' => builder_options[:sortable] else template.content_tag :div, html, class: "has_many_container #{assoc}", 'data-sortable' => builder_options[:sortable] end end def semantic_errors(*args) form_buffers.last << with_new_form_buffer{ super } end protected def active_admin_input_class_name(as) "ActiveAdmin::Inputs::#{as.to_s.camelize}Input" end def input_class(as) @input_classes_cache ||= {} @input_classes_cache[as] ||= begin begin custom_input_class_name(as).constantize rescue NameError begin active_admin_input_class_name(as).constantize rescue NameError standard_input_class_name(as).constantize end end rescue NameError raise Formtastic::UnknownInputError, "Unable to find input class for #{as}" end end # This method calls the block it's passed (in our case, the `f.inputs` block) # and wraps the resulting HTML in a fieldset. If your block doesn't have a # valid return value but it was otherwise built correctly, we instead use # the most recent part of the Active Admin form buffer. def field_set_and_list_wrapping(*args, &block) block_given? ? super{ (val = yield).is_a?(String) ? val : form_buffers.last } : super end private def with_new_form_buffer form_buffers << ''.html_safe return_value = (yield || '').html_safe form_buffers.pop return_value end def without_wrapper is_being_wrapped = @already_in_an_inputs_block @already_in_an_inputs_block = false html = with_new_form_buffer{ yield } @already_in_an_inputs_block = is_being_wrapped html end # Capture the ADD JS def js_for_has_many(assoc, form_block, template, new_record, class_string) assoc_reflection = object.class.reflect_on_association assoc assoc_name = assoc_reflection.klass.model_name placeholder = "NEW_#{assoc_name.to_s.upcase.split(' ').join('_')}_RECORD" opts = { for: [assoc, assoc_reflection.klass.new], class: class_string, for_options: { child_index: placeholder } } html = with_new_form_buffer{ inputs_for_nested_attributes opts, &form_block } text = new_record.is_a?(String) ? new_record : I18n.t('active_admin.has_many_new', model: assoc_name.human) template.link_to text, '#', class: "button has_many_add", data: { html: CGI.escapeHTML(html).html_safe, placeholder: placeholder } end end end