我正在尝试使用cocoon gem来构建嵌套表单。
我有Organization,Package :: Bip和Tenor的模型。
关联是:
组织
has_many :bips, as: :ipable, class_name: Package::Bip
accepts_nested_attributes_for :bips, reject_if: :all_blank, allow_destroy: true
包:: Bip(多态)
belongs_to :ipable, :polymorphic => true, optional: true, inverse_of: :bip
has_one :tenor, as: :tenor
accepts_nested_attributes_for :tenor, reject_if: :all_blank, allow_destroy: true
Tenor(多态)
belongs_to :tenorable, :polymorphic => true, optional: true
表格有:
在我的组织/ _form.html.erb中,我有:
<%= f.simple_fields_for :bips do |f| %>
<%= f.error_notification %>
<%= render 'package/bips/bip_fields', f: f %>
<% end %>
<%= link_to_add_association 'Add another intellectual property resource', f, :bips, partial: 'package/bips/bip_fields' %>
在我的bip_fields.html.erb嵌套表格中,我有:
<%# if @package_bips.tenor.blank? %>
<%= link_to_add_association 'Add timing', f, :tenor, partial: 'tenors/tenor_fields' %>
<%# end %>
<%= f.simple_fields_for :tenor do |tenor_form| %>
<%= f.error_notification %>
<%= render 'tenors/tenor_fields', f: tenor_form %>
<% end %>
的Javascript
cocoon文档建议添加一个js文件来指定关联插入节点作为函数。在我的tenor_subform.js中,我有:
$(document).ready(function() {
$(".add_tenor a").
data("association-insertion-method", 'append').
data("association-insertion-node", function(link){
return link.closest('.row').next('.row').find('.tenor_form')
});
});
控制器
在我的组织控制器中,我有:
def new
@organisation = Organisation.new
@organisation.bips
end
注意:我不确定是否需要在我的新操作中添加另一行来创建organisation.bip.tenor实例。我也不确定我是否应该在organisation.rb上通过关联添加has_one来引用男高音。
def organisation_params
params.fetch(:organisation, {}).permit(:title, :comment,
bips_attributes: [:id, :status, :_destroy,
tenor_attributes: [:id,:commencement, :expiry, :_destroy]
],
在我的男高音控制器中,我有:
def tenor_params
params.require(:tenor).permit( :commencement, :expiry)
end
错误
我不确定是否需要向组织控制器添加次要操作(bip的最终父级,而后者又是次要的父级)。
当我保存所有这些并尝试时,我收到一条错误消息:
unknown attribute 'tenor_id' for Tenor.
当我看到其他SO帖子出现此错误时,通常是因为:id属性尚未在父类中列入白名单。我做到了。
谁能看到我做错了什么?
男高音控制器
class TenorsController < ApplicationController
before_action :set_tenor, only: [:show, :edit, :update, :destroy]
before_action :authenticate_user!
# after_action :verify_authorized
def index
@tenors = Tenor.all
# authorize @tenors
end
def show
end
def new
@tenor = Tenor.new
# authorize @tenor
end
def edit
end
def create
@tenor = Tenor.new(tenor_params)
# authorize @tenor
respond_to do |format|
if @tenor.save
format.html { redirect_to @tenor }
format.json { render :show, status: :created, location: @tenor }
else
format.html { render :new }
format.json { render json: @tenor.errors, status: :unprocessable_entity }
end
end
end
def update
respond_to do |format|
if @tenor.update(tenor_params)
format.html { redirect_to @tenor }
format.json { render :show, status: :ok, location: @tenor }
else
format.html { render :edit }
format.json { render json: @tenor.errors, status: :unprocessable_entity }
end
end
end
def destroy
@tenor.destroy
respond_to do |format|
format.html { redirect_to action: :index }
format.json { head :no_content }
end
end
private
def set_tenor
@tenor = Tenor.find(params[:id])
# authorize @tenor
end
def tenor_params
params.require(:tenor).permit(:express_interest, :commencement, :expiry, :enduring, :repeat, :frequency)
end
end