我想将@trade_wizard
(它有自己的控制器,WizardsController
)的多步骤表单的第一步渲染为部分内部ItemsController#show
,但我不知道如何构建它,而不是将代码从一个控制器加倍到另一个控制器。
我正在渲染Item的显示页面中的第一步:
<%= render "/wizards/step1" %>
@trade_wizard在实例化@trade的特殊模型中处理,然后从每个步骤连续继承验证:
module Wizard
module Trade
STEPS = %w(step1 step2 step3).freeze
class Base
include ActiveModel::Model
attr_accessor :trade
delegate *::Trade.attribute_names.map { |attr| [attr, "#{attr}="] }.flatten, to: :trade
def initialize(trade_attributes)
@trade = ::Trade.new(trade_attributes)
end
end
class Step1 < Base
validates :trade_requester_id, :trade_recipient_id, :wanted_item_id, presence: true
validates :shares, numericality: { only_integer: true, greater_than_or_equal_to: 0,
less_than_or_equal_to: :max_shares }
def max_shares
@trade.wanted_item.shares
end
end
class Step2 < Step1
validates :collateral_item_id, presence: true
end
class Step3 < Step2
validates :agreement, presence: true
end
end
end
然后我的WizardsController在每一步上运行验证并保存对象:
class WizardsController < ApplicationController
before_action :load_trade_wizard, except: %i(validate_step)
def validate_step
current_step = params[:current_step]
@trade_wizard = wizard_trade_for_step(current_step)
@trade_wizard.trade.attributes = trade_wizard_params
session[:trade_attributes] = @trade_wizard.trade.attributes
if @trade_wizard.valid?
next_step = wizard_trade_next_step(current_step)
create and return unless next_step
redirect_to action: next_step
else
render current_step
end
end
def create
if @trade_wizard.trade.save
session[:trade_attributes] = nil
redirect_to root_path, notice: 'Trade succesfully created!'
else
redirect_to({ action: Wizard::Trade::STEPS.first }, alert: 'There were a problem when creating the trade.')
end
end
private
def load_trade_wizard
@trade_wizard = wizard_trade_for_step(action_name)
end
def wizard_trade_next_step(step)
Wizard::Trade::STEPS[Wizard::Trade::STEPS.index(step) + 1]
end
def wizard_trade_for_step(step)
raise InvalidStep unless step.in?(Wizard::Trade::STEPS)
"Wizard::Trade::#{step.camelize}".constantize.new(session[:trade_attributes])
end
def trade_wizard_params
params.require(:trade_wizard).permit(:trade_requester_id, :trade_recipient_id, :wanted_item_id, :collateral_item_id, :shares, :agreement)
end
class InvalidStep < StandardError; end
end
在我的路线中我有
resource :wizard do
get :step1
get :step2
get :step3
post :validate_step
end
我在此设置中遇到的错误是First argument in form cannot contain nil or be empty
。我知道为什么会发生这种情况 - 我需要在ItemsController#show中定义@trade_wizard
,我还没有这样做,因为这只会导致我从WizardsController复制代码。我不需要任何人为我做我的工作,我只需要一个指针,说明我如何能够摆脱这个问题。
答案 0 :(得分:2)
控制器设计为独立的,它们不能相互依赖。这与视图不同,可能正如您所做的那样,可以通过部分重用和组合。
如果您需要重用控制器中的行为(与一个控制器不同,取决于另一个控制器),您可以使用继承,或者按照Rails方式concerns。
在这种情况下,我会在包含@trade_wizard
局部视图的任何控制器中设置wizards/step1
变量。
答案 1 :(得分:2)
从elc告诉我将使用ajax来隐藏和显示步骤,并结合嵌套表单。
您创建的Wizard
模型包含许多步骤,并接受steps
作为nested attributes
。您可以在rails guide
nested forms
class Wizard < ActiveRecord:Base
has_many :steps
accepts_nested_attributes_for :steps
end
Step
模型属于向导
class Step < ActiveRecord:Base
belongs_to :wizard
end
这是你的表格
<%= form_for @wizard, class: 'hidden' do |f| %>
Addresses:
<ul>
<%= f.fields_for :steps do |step| %>
// include your fields
<% end %>
</ul>
<% end %>
此表单向post
执行/wizards
请求,添加一些ajax逻辑,允许隐藏您在steps
中创建文件的部分app/views/wizards
表单create.js.erb
并在那里写下js
逻辑,其中包含控制器中使用的任何变量,因为它是erb
文件。
这取决于您想要如何编写此内容,但您可以在wizards#create
操作中包含此逻辑
在某些情况下,您可能希望执行js以显示下一个表单,其他情况下您要保存该对象并呈现新视图。概念是http是无状态的,因此对于每个请求,您将重新创建@wizard实例,但是隐藏时从表单填充的字段仍将作为强对象重新提交
# app/controllers/wizards_controller.rb
# ......
def create
@wizard = Wizard.new(params[:wizard])
respond_to do |format|
// you can set conditions and perform different AJAX responses based on the request you received.
format.js
format.html { render action: "new" }
end
end
我会写更多但我需要去