这个让我失望! :(
我正在尝试为我的用户模型创建一个嵌套模型表单,其中包含一个复选框列表,其中可以检查或取消选中多个商店以通过模型人员配置来管理商店。
class Staffing < ActiveRecord::Base
# Associations
belongs_to :user
belongs_to :staffable, :polymorphic => true
belongs_to :store, :class_name => "Store",
:foreign_key => "staffable_id"
end
class User < ActiveRecord::Base
# Includes
acts_as_authentic
# Associations
has_many :staffings, :dependent => :destroy
has_many :stores, :through => :staffings
# Nested Attributes
accepts_nested_attributes_for :staffings, :allow_destroy => true
end
class Store < ActiveRecord::Base
# Associations
has_many :staffings, :as => :staffable, :dependent => :destroy
has_many :users, :through => :staffings
end
# Users Controller
def create
@user = User.new(params[:user])
flash.now[:notice] = "#{@user.name} was successfully created." if @user.save
respond_with @user
end
def update
@user = User.find(params[:id])
params[:user][:store_ids] ||= []
@user.update_attributes(params[:user])
flash.now[:notice] = "#{@user.name} was successfully updated."
respond_with @user
end
出于手头的目的,可以忽略其他人员协会。这是一个类似的模型,我最终想要与商店一起管理但是第一件事是第一件事,因为我很难过。
# User Form
- for store in Store.all
%p
= check_box_tag "user[store_ids][]", store.id, @user.stores.include?(store)
= store.nickname
发送我的参数:
{"utf8"=>"✓",
"authenticity_token"=>"blub-blub-blub=",
"user"=>{"login"=>"hey.buddy",
"email"=>"test@hey.net",
"role"=>"admin",
"hq"=>"1",
"password"=>"[FILTERED]",
"password_confirmation"=>"[FILTERED]",
"store_ids"=>["3",
"4"]}}
然后是ERROR!
Mysql2::Error: Column 'staffable_type' cannot be null: INSERT INTO `staffings` (`created_at`, `staffable_id`, `staffable_type`, `updated_at`, `user_id`) VALUES ('2010-11-17 00:30:24', 3, NULL, '2010-11-17 00:30:24', 102)
我知道我必须将staffable_type构建为'Store',但我一整天都在努力 - 这是什么诀窍?
我确实将staffable_type(和id)列设置为:null =&gt;是的,但这不能成为原因,因为无论如何都需要在控制器中理清它。
为什么以下内容在我的创建操作中不起作用:
@user.staffings.each do |s|
s.staffable_type = 'Store'
end
或:
@user.store_ids.each do |i|
s = @user.staffings.build
s.staffable_type = 'Store'
s.staffable_id = i
end
如果尝试过类似上述的许多事情都无济于事。任何帮助都会受到大力赞赏。
谢谢你的时间!
答案 0 :(得分:4)
好的,我想出了这个。我在这里回答所以我希望有一天能帮助某个人,但问题是对许多通过和拥有并属于许多协会之间的差异的基本疏忽。
如果您想要处理Has Many Through中的复选框或多选列表,则关联不会为您生成'_ids'方法,您需要自己编写一个。
请参阅Paul Barry的文章: http://paulbarry.com/articles/2007/10/24/has_many-through-checkboxes
多态的东西可能很复杂,但是如果你真的很好,可能会退后一步,确保它不是更基本的东西让你搞砸了 - 为我工作!
下面是保罗博客中关于如何处理这个问题的代码,如果你处理多态有很多通过(基本上你只需要使用属性hash并编写你的polymorphic_type属性):
attr_accessor :store_ids
after_save :update_stores
#after_save callback to handle group_ids
def update_stores
unless store_ids.nil?
self.staffings.each do |staffing|
staffing.destroy unless store_ids.include?(staffing.staffable_id.to_s)
store_ids.delete(staffing.staffable_id.to_s)
end
store_ids.each do |s|
self.staffings.create(attributes = {:staffable_id => s, :staffable_type => 'Store'}) unless s.blank?
end
reload
self.store_ids = nil
end
end