【问题标题】:Trying to add a foreign relationship to my users model with Devise, then update it尝试使用 Devise 向我的用户模型添加外部关系,然后对其进行更新
【发布时间】:2025-12-05 00:55:01
【问题描述】:

这适用于 Rails 4.04 和 Ruby 2.1。我想让我的用户拥有地址。所以我生成了一个地址表并给它列(NUMBER,STREET,CITY,STATE)。现在,当我转到以下网址时,我希望能够编辑此信息:

webapp.com/users/edit/

但是我注意到它只显示了相同的旧信息(姓名、密码、电子邮件)。所以我转到视图并为我的新关系添加了 simple_fields,因此视图现在看起来像这样:

<%= simple_form_for(resource, as: resource_name, url: registration_path(resource_name), html: { method: :put }) do |f| %>
 <div class="form-inputs">
    <%= f.input :email, required: true, autofocus: true %>
    <%= f.input :name, required: false %>
    <%= f.simple_fields_for :addresses do |a| %>
        <%= a.input :number %>
        <%= a.input :street %>
        <%= a.input :city %>
        <%= a.input :state %>
        <%= a.input :country %>
  <% end %>
<%end%>

但是它仍然没有生成地址所需的字段。我认为这是因为我的用户目前没有任何地址附加到他们的帐户资料中(因为此迁移刚刚创建)。但是,在这种情况下,应该会生成空白字段,以便我可以添加地址信息。

我觉得我需要像这样在 Users#Edit 操作中做一些事情

@users.each do |user|
        user.address.build
    end

对吗?我如何覆盖用户控制器,因为它是由 Devise 创建的,而我实际上没有用户控制器(我寻找它但找不到它)。

更新

好的,我越来越近了。我必须创建自己的控制器来覆盖 Devise 的默认注册控制器,如这篇堆栈溢出文章的第二个答案中所述:

Override devise registrations controller

所以现在我正在进入当前看起来像这样的控制器:

class Users::RegistrationsController < Devise::RegistrationsController
  def edit
    super
  end
end

但是,当我进入我的视野时,它仍然在跳过这样开始的块:

<%= f.simple_fields_for :addresses do |a| %>

但是,如果我手动进入我的数据库并在地址表中添加一条记录并通过外键将其链接到我的 current_signed in 用户,则不会跳过该块。那么如果地址记录尚不存在,那么生成此连接的最佳方法是什么?是构建方法吗?例如

user.address.build 

在控制器中

解决方案

是的,我需要将此方法添加到我的新 registrations_controller.rb 文件中

def edit
    if resource.addresses.size == 0
      resource.addresses.build
    end
    super
  end

它现在按我预期的方式工作。

【问题讨论】:

  • 设计不附带控制器。我认为有一种手动添加它的方法。 devise 自带user modelusers table 你有没有把usersaddresses 关联起来?

标签: ruby-on-rails devise


【解决方案1】:

你需要这样做:

#app/models/user.rb
Class User < ActiveRecord::Base
   has_one :address
   before_create :build_address, unless: Proc {|x| x.address.present? } #-> not sure about the if statement
end

#app/models/address.rb
Class Address < ActiveRecord::Base
   belongs_to :user
end

--

Devise does come with Controllers

这些控制器不会显示在您的应用中(它们与 Devise gem 一起安装,但仅在生产中可见):

- confirmations_controller.rb
- omniauth_callbacks_controller.rb
- passwords_controller.rb
- registrations_controller.rb
- sessions_controller.rb
- unlocks_controller.rb

--

你不需要他们

虽然您可以覆盖这些控制器,但您不需要这样做,因为您的 edit 操作将绑定到 users 控制器:

#config/routes.rb
resources :users

#app/controllers/users_controller.rb
Class UsersController < ApplicationController
   def edit
      @user = User.find params[:id]
   end
end

#app/views/users/edit.html.erb
<%= form_for @user do |f| %>
   ...
   <%= f.fields_for :address do |a| %>
   <% end %>
<% end %>

更新

抱歉没有解释。您可以resources :usersdevise_for :users 一起使用:

#config/routes.rb
devise_for :users
resources :users, only: [:edit, :update]

如果这不起作用,您可能需要更改devisepath_names 参数,如下所示:

#config/routes.rb
devise_for :users, path_names: { sign_in: 'login', password: 'forgot', confirmation: 'confirm', unlock: 'unblock', sign_up: 'register', sign_out: 'logout'}

-

表格

表格可以有resource,但我认为你需要弄清楚你想要改变什么。如果您尝试更改 devise 对象,请使用 resource 助手;但如果您尝试直接更改 User 模型 - 我会倾向于更改它!

我认为您遇到的问题是,如果您使用resource,它将路由到Devise 控制器操作。我会尝试在您的users controlleredit 操作中设置@user,并使用常规方式进行更新

【讨论】:

  • 问题 #1 你说的是 Class UsersController
  • 我追溯了这个动作,我可以看到它通过了 Gem 中的一个特殊设计控制器(类 Devise::RegistrationsController 时使用的控制器
  • 更新我的原帖
  • 我没有完全使用你推荐的方法,但我会选择你的答案,因为它是最彻底的,而且大部分与正确答案相关。
【解决方案2】:

看起来您没有将这些模型之间的关系添加到名为 user_idaddresses 列并添加到用户模型:

has_many :addresses

进入地址:

belongs_to :users

然后在您的视图中,您可以使用嵌套属性构建此表单,请参阅此帖子:

Rails 4 Nested Attributes Unpermitted Parameters

您可以做的另一个选择是在用户已经登录之后显示addresses 表单,然后当他更新表单时找到带有current_user 的user_id 并使用此id 但在Rails 4 中使用强参数来构建记录建议解决您的问题。

【讨论】:

    最近更新 更多