8

我看过 RailsCast,另一个嵌套属性视频,很多 SO 帖子,并为此奋斗了一段时间,但我仍然无法弄清楚。我希望它是微小的东西。

我有两个模型,User(由 Devise 创建)和Locker(又名产品愿望清单),我正在尝试在他们注册时创建Locker一个。User我的登录表单有一个字段,用于填写他们的新名称Locker(恰当地称为:name),我试图将其分配给在新用户注册时创建的储物柜。我所受到的只是:

WARNING: Can't mass-assign protected attributes: locker

我已经尝试了两种模型的所有组合accepts_nested_attributesattr_accesible但仍然没有任何效果。我可以从日志中看到它正在由 Devise#create 方法处理,而且我知道 Devise 不够聪明,无法按照我的意愿创建模型:)

这是我的两个模型的相关部分:

# user.rb    
class User < ActiveRecord::Base
  attr_accessible :username, :email, :password, :password_confirmation, :remember_me, :locker_attributes

  # Associations
  has_many :lockers
  has_many :lockups, :through => :lockers

  # Model nesting access
  accepts_nested_attributes_for :lockers
end

# locker.rb
class Locker < ActiveRecord::Base
  belongs_to :user
  has_many :lockups
  has_many :products, :through => :lockups 

  attr_accessible :name, :description
end

# lockers_controller.rb (create)
    @locker = current_user.lockers.build(params[:locker])
    @locker.save

我假设我需要重写 Devise 的create方法才能以某种方式让它工作,但我对 rails 很陌生,并且已经习惯了它的黑匣子“魔法”性质。

如果有人可以帮助我,我将非常感激。已经在这上面花了太多时间了:)

编辑:我意识到我在我的问题中遗漏了一些东西。我的Locker模型具有三个属性 - namedescription(不是强制性的),并将user_id其链接回User. 我的注册表单只需要name,所以我没有遍历嵌套表单中的所有属性。这也与我的问题有关吗?

编辑 2:我还想出了如何覆盖 Devise 的RegistrationsController#create方法,我只是不知道该放什么。设计的整个resource事情对我来说没有意义,浏览他们的源代码对RegistrationsController我也没有多大帮助。

并且对于奖励积分:当用户提交带有无效数据的登录表单时,该Locker字段总是返回空白,而常规的设计字段,用户名和电子邮件,被填写。这也可以很容易地解决吗?如果是这样,怎么做?

4

2 回答 2

2

首先,你有一个错字:

attr_accessible :locker_attributes

应该是复数:

attr_accessible :lockers_attributes

那么,标准的使用方式nested_attributes是:

<%= form_for @user do |f| %>
  <%# fields_for will iterate over all user.lockers and 
      build fields for each one of them using the block below,
      with html name attributes like user[lockers_attributes][0][name].
      it will also generate a hidden field user[lockers_attributes][0][id]
      if the locker is already persisted, which allows nested_attributes
      to know if the locker already exists of if it must create a new one
  %>
  <% f.fields_for :lockers do |locker_fields| %>
    <%= locker_fields.label      :name %>
    <%= locker_fields.text_field :name %>
  <% end %>
<% end %>

在你的控制器中:

def new
  @user = User.new
  @user.lockers.build
end

def create
  # no need to use build here : params[:user] contains a
  # :lockers_attributes key, which has an array of lockers attributes as value ;
  # it gets assigned to the user using user.lockers_attributes=, 
  # a method created by nested_attributes
  @user = User.new( params[:user] )
end

作为旁注,您可以避免以不同的方式在控制器中为新用户构建新的储物柜:

  • 在上创建工厂方法User,或覆盖new,或使用after_initialize回调来确保每个实例化的新用户都会自动构建一个储物柜。

  • 将特定对象传递给fields_for

    <% f.fields_for :lockers, f.object.lockers.new do |new_locker_fields| %>
    
于 2013-05-25T09:07:17.273 回答
2

有人帮助我以更“Rails 4'y”的方式找出解决方案strong attributes以及如何覆盖 Devise sign_up_params(以捕获来自我的注册表单的所有数据)。

  def sign_up_params
    params.require(:user).permit(:username, :email, :password, :lockers_attributes)
 end

Gemfile添加:gem 'strong_parameters'

注释掉attr_accessibleuser.rb文件中的语句,因为显然强参数消除了attr_accessible声明的需要。

 # attr_accessible :username, :email, :password, :password_confirmation, :lockers

以及在提交表单之前构建Locker的正确方法:在嵌套表单的开头:

<%= l.input :name, :required => true, label: "Locker name", :placeholder => "Name your first locker" %>

再次感谢你的帮助。我知道如果不查看整个代码库就很难回答这样的问题。

于 2013-05-26T16:22:02.623 回答