我们都看到了辉煌的complex forms railscast,其中Ryan Bates解释了如何使用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
#Application Helper(直接来自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方法有关 - Formtastic表单的DOM树与常规rails表单有很大不同。
我已经在网上看过几次这个问题,但还没有找到答案 - 现在你知道帮助将不仅仅受到我的赞赏!
杰克
答案 0 :(得分:25)
你走在正确的轨道上:
...... Formtastic形式的DOM树 与常规导轨有很大不同 形式。
为了使Ryan的示例适用于formtastic,请注意semantic_fields_for
助手类似于semantic_form_for
助手,它以列表形式输出输入。
为了尽可能接近Railscast代码,您需要:
subjects
CSS ID的div。)nested-fields
CSS类。)以下是您的文件应该包含的内容。
教师表单(带有嵌套的主题字段):
- 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)
});
}
使用以下宝石:
答案 1 :(得分:3)
jQuery的application.js:
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));
}
答案 2 :(得分:1)
在Rails 3中,不需要在帮助程序中使用h()函数。只是省略,它应该工作。
答案 3 :(得分:1)
我多年来一直在打击和关闭这个,我今天想出了一些我很自豪的东西 - 优雅,不引人注目,你只能做两三个(真的很长)代码行。
ParentFoo has_many NestedBars,带有相应的accepts_nested_parameters和所有这些好东西。您可以使用:child_index设置一个可以在以后替换的字符串,将nested_bar的一组字段渲染到链接上的data-add-nested属性中。您在第二个参数data-add-nested-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'
嵌套字段partial不是什么花哨的
parent_foos / _nested_bar_fields.html.haml
- f.inputs do
= f.input :name
= f.input :_delete, :as => :boolean
在您的jQuery中,您将元素的点击与数据添加嵌套字段绑定(我在这里使用了live(),因此ajax加载的表单将起作用)将字段插入DOM,用字符串替换字符串新ID。在这里我做了一个简单的事情就是在链接之前插入新字段,但你也可以在链接上提供一个add-nested-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))
})
当然,你可以把它放在帮助器里;我在这里直接在视图中呈现它,所以原理很清楚,而不会在元编程中徘徊。如果您担心名称冲突,可以在该帮助程序中生成唯一ID,并以嵌套替换方式传递它。
了解删除行为是留给读者的练习。
(显示为Rails 2,因为这是我今天正在处理的网站 - 在Rails 3中几乎相同)