Как добавлять и удалять вложенные поля модели динамически с помощью Haml и Formtastic
Мы все видели блестящие сложные формы railscast, где Райан Бейтс объясняет, как динамически добавлять или удалять вложенные объекты в родительской форме объекта, используя Javascript.
Есть ли у кого-нибудь идеи о том, как эти методы нужно модифицировать, чтобы работать с Haml Formtastic?
Чтобы добавить некоторый контекст здесь, упрощенная версия проблемы, с которой я сталкиваюсь сейчас:
# Форма учителя (которая имеет вложенные формы темы) [из моего приложения]
- semantic_form_for(@teacher) do |form|
- form.inputs do
= form.input :first_name
= form.input :surname
= form.input :city
= render 'subject_fields', :form => form
= link_to_add_fields "Add Subject", form, :subjects
# Индивидуальный Предмет формы частично [из моей заявки]
- form.fields_for :subjects do |ff|
#subject_field
= ff.input :name
= ff.input :exam
= ff.input :level
= ff.hidden_field :_destroy
= link_to_remove_fields "Remove Subject", ff
# Помощник приложения (прямо из Railscasts)
def link_to_remove_fields(name, f)
f.hidden_field(:_destroy) + link_to_function(name, "remove_fields(this)")
end
def link_to_add_fields(name, f, association)
new_object = f.object.class.reflect_on_association(association).klass.new
fields = f.fields_for(association, new_object, :child_index => "new_#{association}") do |builder|
render(association.to_s.singularize + "_fields", :f => builder)
end
link_to_function(name, h("add_fields(this, \"#{association}\", \"#{escape_javascript(fields)} \")"))
end
# Application.js(прямо из Railscasts)
function remove_fields(link) {
$(link).previous("input[type=hidden]").value = "1";
$(link).up(".fields").hide();
}
function add_fields(link, association, content) {
var new_id = new Date().getTime();
var regexp = new RegExp("new_" + association, "g")
$(link).up().insert({
before: content.replace(regexp, new_id)
});
}
Проблема с реализацией, по-видимому, связана с методами javascript - дерево DOM формы Formtastic сильно отличается от обычной формы рельсов.
Я видел этот вопрос несколько раз в Интернете, но еще не получил ответа - теперь вы знаете, что помощь будет оценена не только мной!
Джек
Ответы
Ответ 1
Вы на правильном пути:
... дерево DOM формы Formtastic сильно отличается от обычных рельсов форма.
Чтобы адаптировать пример Ryan для formtastic, полезно напомнить, что помощник semantic_fields_for
похож на помощник semantic_form_for
, который выводит входы в виде списка.
Чтобы как можно ближе доставлять код Railscast, вам необходимо:
- заключить коллекцию вложенных полей в оболочку (я использую div с
subjects
CSS ID.)
- заключить вложенные поля в оболочку ul/ol (я применил класс CSS
nested-fields
).
Здесь вам понравятся ваши файлы.
Форма учителя (с вложенными полями Subject):
- semantic_form_for(@teacher) do |form|
- form.inputs do
= form.input :first_name
= form.input :surname
= form.input :city
%h2 Subjects
#subjects
- form.semantic_fields_for :subjects do |builder|
= render :partial => "subject_fields", :locals => { :f => builder }
.links
= link_to_add_fields "Add Subject", form, :subjects
Субъект поля (для вложенного объекта):
%ul.nested-fields
= f.input :name
= f.input :exam
= f.input :level
= link_to_remove_fields "Remove Subject", f
ApplicationHelper:
def link_to_remove_fields(name, f)
f.hidden_field(:_destroy) + link_to_function(name, "remove_fields(this)")
end
def link_to_add_fields(name, f, association)
new_object = f.object.class.reflect_on_association(association).klass.new
fields = f.fields_for(association, new_object, :child_index => "new_#{association}") do |builder|
render(association.to_s.singularize + "_fields", :f => builder)
end
link_to_function(name, h("add_fields(this, \"#{association}\", \"#{escape_javascript(fields)}\")"))
end
application.js:
function remove_fields(link) {
$(link).previous("input[type=hidden]").value = "1";
$(link).up(".nested-fields").hide();
}
function add_fields(link, association, content) {
var new_id = new Date().getTime();
var regexp = new RegExp("new_" + association, "g")
$(link).up().insert({
before: content.replace(regexp, new_id)
});
}
Использование следующих драгоценных камней:
- formtastic (0.9.10)
- haml (3.0.2)
- gherkin (1.0.26)
- рельсы (2.3.5)
Ответ 2
application.js для jQuery:
function remove_fields(link) {
$(link).prev("input[type=hidden]").val("1");
$(link).parent(".nested-fields").hide();
}
function add_fields(link, association, content) {
var new_id = new Date().getTime();
var regexp = new RegExp("new_" + association, "g")
$(link).parent().before(content.replace(regexp, new_id));
}
Ответ 3
В Rails 3 нет необходимости использовать функцию h() в помощнике. Просто опустите, и он должен работать.
Ответ 4
Я много лет бил головой об этом, и сегодня я придумал то, чем я горжусь - изящный, ненавязчивый, и вы можете сделать это всего два или три (очень долго) строки кода.
ParentFoo has_many NestedBars, с соответствующими параметрами accepts_nested_parameters и всеми этими лакомствами. Вы передаете набор полей для nested_bar в атрибут добавления данных в ссылку, используя: child_index, чтобы установить строку, которую вы можете заменить позже. Вы передаете эту строку во втором параметре, data-add-inested-replace
parent_foos/_form.html.haml
- semantic_form_for @parent_foo do |f|
-# render existing records
- f.semantic_fields_for :nested_bars do |i|
= render 'nested_bar_fields', :f => i
# magic!
- reuse_fields = f.semantic_fields_for :nested_bars, @parent_foo.nested_bars.build, :child_index => 'new_nested_bar_fields' do |n| render('nested_bar_fields', :f => n) end
= link_to 'Add nested bar', '#', 'data-add-nested' => reuse_fields, 'data-add-nested-replace' => 'new_nested_bar_fields'
Вложенные поля, частичные, не представляют ничего фантастического
parent_foos/_nested_bar_fields.html.haml
- f.inputs do
= f.input :name
= f.input :_delete, :as => :boolean
В jQuery вы привязываете клик элементов с помощью поля с добавлением данных (я использовал live() здесь, так что формы, загруженные ajax, будут работать), чтобы вставить поля в DOM, заменив строку на новый идентификатор. Здесь я делаю простую вставку новых полей до() ссылки, но вы также можете указать атрибут add-inested-replace-target в ссылке, где указано, где в DOM вы хотите, чтобы новые поля завершались.
application.js
$('a[data-add-nested]').live('click', function(){
var regexp = new RegExp($(this).data('add-nested-replace'), "g")
$($(this).data('add-nested').replace(regexp, (new Date).getTime())).insertBefore($(this))
})
Разумеется, вы могли бы поставить это в помощнике; Я представляю его здесь прямо в представлении, поэтому принцип ясен, не обманывая метапрограммирование. Если вы беспокоитесь о конфликтах имен, вы можете сгенерировать уникальный идентификатор в этом помощнике и передать его вложенной-заменой.
получение фантазии с поведением удаления остается в качестве упражнения для читателя.
(показано для Rails 2, потому что сайт, над которым я работаю сегодня - почти то же самое в Rails 3)