覆盖设计注册控制器

Posted

技术标签:

【中文标题】覆盖设计注册控制器【英文标题】:Override devise registrations controller 【发布时间】:2011-04-02 13:21:22 【问题描述】:

我在注册表单中添加了一个基于不同模型的字段,请参阅How do I use nested attributes with the devise model 了解详细信息。这部分工作正常。

现在的问题是,当我保存时,它在注册控制器的创建操作中失败,该操作由设计提供,在该字段(公司)上带有 Activerecord::UnknownAttributeError

我假设我需要覆盖注册控制器,还是有更好/更简单的方法来解决这个问题?

【问题讨论】:

我实际上在jacopretorius.net/2014/03/…jacopretorius.net/2014/03/…上写了整篇博文 【参考方案1】:

非常简单的方法 只需转到终端和以下类型

rails g devise:controllers users //This will create devise controllers in controllers/users folder

接下来使用自定义视图

rails g devise:views users //This will create devise views in views/users folder

现在在你的 route.rb 文件中

devise_for :users, controllers: 
           :sessions => "users/sessions",
           :registrations => "users/registrations" 

您也可以添加其他控制器。这将设计使用用户文件夹中的控制器和用户文件夹中的视图。

现在您可以根据需要自定义视图并将逻辑添加到控制器/用户文件夹中的控制器。 享受吧!

【讨论】:

【参考方案2】:

您可以为设计定制生成视图和控制器。

使用

rails g devise:controllers users -c=registrations

rails g devise:views 

它将特定的控制器和视图从 gem 复制到您的应用程序。

接下来,告诉路由器使用这个控制器:

devise_for :users, :controllers => :registrations => "users/registrations"

【讨论】:

【参考方案3】:

在您的表单中,您是否通过不属于您的用户模型或任何嵌套模型的批量分配传递了任何其他属性?

如果是这样,我相信在这种情况下会触发 ActiveRecord::UnknownAttributeError。

否则,我认为您可以通过生成如下内容来创建自己的控制器:

# app/controllers/registrations_controller.rb
class RegistrationsController < Devise::RegistrationsController
  def new
    super
  end

  def create
    # add custom create logic here
  end

  def update
    super
  end
end 

然后告诉 devise 使用该控制器而不是默认控制器:

# app/config/routes.rb
devise_for :users, :controllers => :registrations => "registrations"

【讨论】:

但是如何确保设计在设计目录中查找视图?我正在尝试这个,但设计诸如“sign_in_and_redirect(resource_name,resource)”之类的方法正在寻找模板的视图。 如果你想自定义你的设计视图,你只需要先生成它们,设计会在从 gem 加载视图之前检查你的视图文件夹。在 Rails 3 中是:rails generate devise:views,在 Rails 2 中(我认为)是:script/generate devise:views 上述技巧不适用于设计 1.0.8,该版本适用于 rails 2。 如果您像这样覆盖 Devise 控制器,请确保将所有视图从 app/views/devise/registrations 复制到 app/views/registrations/ (更改您要覆盖的控制器)。 或者,您可以将设计视图保留在原处,并在您的 config/application.rb 中添加 paths.app.views &lt;&lt; "app/views/devise"【参考方案4】:

使用命名空间覆盖 Devise 控制器和视图的更好、更有条理的方法:

创建以下文件夹:

app/controllers/my_devise
app/views/my_devise

将您要覆盖的所有控制器放入 app/controllers/my_devise 并将 MyDevise 命名空间添加到控制器类名称中。 Registrations 例子:

# app/controllers/my_devise/registrations_controller.rb
class MyDevise::RegistrationsController < Devise::RegistrationsController

  ...

  def create
    # add custom create logic here
  end

  ...    

end 

相应地更改您的路线:

devise_for :users,
           :controllers  => 
             :registrations => 'my_devise/registrations',
             # ...
           

将所有需要的视图从 Devise gem 文件夹复制到 app/views/my_devise 或使用 rails generate devise:views,删除您未覆盖的视图并将 devise 文件夹重命名为 my_devise

这样您就可以将所有内容整齐地组织在两个文件夹中。

【讨论】:

这类似于我正在采用的方法,但我不知道在我覆盖的 Devise 的 create 方法中放入什么自定义逻辑。我修改后的脚手架创建的控制器效果很好,但是如何使其与 Devise 的 resource 业务一起使用? @Vincent 谢谢 - 如果我只想覆盖一个方法,我是否只写我想要覆盖的方法 - 其他一切都会正常工作吗?非常感谢您的帮助 MyDevise::RegistrationsController &lt; Devise::RegistrationsController 创建循环依赖错误。我做错了吗?【参考方案5】:

我相信有比重写 RegistrationsController 更好的解决方案。我做了完全相同的事情(我只有组织而不是公司)。

如果您在模型和视图级别正确设置嵌套表单,一切都会像魅力一样发挥作用。

我的用户模型:

class User < ActiveRecord::Base
  # Include default devise modules. Others available are:
  # :token_authenticatable, :confirmable, :lockable and :timeoutable
  devise :database_authenticatable, :registerable,
     :recoverable, :rememberable, :trackable, :validatable

  has_many :owned_organizations, :class_name => 'Organization', :foreign_key => :owner_id

  has_many :organization_memberships
  has_many :organizations, :through => :organization_memberships

  # Setup accessible (or protected) attributes for your model
  attr_accessible :email, :password, :password_confirmation, :remember_me, :name, :username, :owned_organizations_attributes

  accepts_nested_attributes_for :owned_organizations
  ...
end

我的组织模式:

class Organization < ActiveRecord::Base
  belongs_to :owner, :class_name => 'User'
  has_many :organization_memberships
  has_many :users, :through => :organization_memberships
  has_many :contracts

  attr_accessor :plan_name

  after_create :set_owner_membership, :set_contract
  ...
end

我的观点:'devise/registrations/new.html.erb'

<h2>Sign up</h2>

<% resource.owned_organizations.build if resource.owned_organizations.empty? %>
<%= form_for(resource, :as => resource_name, :url => registration_path(resource_name)) do |f| %>
  <%= devise_error_messages! %>

  <p><%= f.label :name %><br />
    <%= f.text_field :name %></p>

  <p><%= f.label :email %><br />
    <%= f.text_field :email %></p>

  <p><%= f.label :username %><br />
    <%= f.text_field :username %></p>

  <p><%= f.label :password %><br />
    <%= f.password_field :password %></p>

  <p><%= f.label :password_confirmation %><br />
    <%= f.password_field :password_confirmation %></p>

  <%= f.fields_for :owned_organizations do |organization_form| %>

    <p><%= organization_form.label :name %><br />
      <%= organization_form.text_field :name %></p>

    <p><%= organization_form.label :subdomain %><br />
      <%= organization_form.text_field :subdomain %></p>

    <%= organization_form.hidden_field :plan_name, :value => params[:plan] %>

  <% end %>

  <p><%= f.submit "Sign up" %></p>
<% end %>

<%= render :partial => "devise/shared/links" %>

【讨论】:

将构建逻辑从视图移动到模型会更简洁,请参阅***.com/questions/3544265#3764837 我生成了设计控制器,现在当用户单击注册时触发了控制器操作创建。有没有办法(比如覆盖/一些示例代码)我可以使用 Devise 加密密码并对密码和其他字段进行后端检查?并将其保存到模型数据库中? 如何访问视图中的局部变量resource而不是类实例变量@resource

以上是关于覆盖设计注册控制器的主要内容,如果未能解决你的问题,请参考以下文章

设计 - 用回形针注册(缺少图像)

Rails - 如何覆盖设计SessionsController以在用户登录时执行特定任务?

joomla com_users 控制器覆盖

路由覆盖控制器路由

设计注册控制器的ActionController :: UrlGenerationError

使用设计注册