将多步骤表单向导的第一步渲染为另一个控制器的show action

时间:2018-01-21 10:57:09

标签: ruby-on-rails ruby wizard formwizard

我想将@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复制代码。我不需要任何人为我做我的工作,我只需要一个指针,说明我如何能够摆脱这个问题。

2 个答案:

答案 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

我会写更多但我需要去