Gary Vlc
Gary Vlc

Reputation: 143

How do I create a two-step form in rails?

I have a question for rails. I'm creating a form for user to register. What I want to do is that after the user press "Submit" button I want to redirect the user to another page which shows all the information from the form filled by the user just now (read-only).

This is my controller

class PermitsController < ApplicationController
  before_action :set_permit, only: [:show, :destroy]
  def index
    @permits = Permit.all
  end

  def new
    @permits = Permit.new
  end

  def create

    @permits = current_user.permits.build(permit_params)
      if @permits.save
        redirect_to invoice_path
      else
        render 'new'
      end

      end


  def destroy
    Permit.destroy_all(user_id: current_user)
    respond_to do |format|
      format.html { redirect_to root_path, notice: 'Permit was successfully canceled.' }
      format.json { head :no_content }
    end
  end

def invoice
  @permits = current_user.permits(permit_params)

end

  def show
    @user = User.find(params[:id])
    @permits = @user.permits.paginate(permit_params)
  end

  def update
    @permits = Permit.where(user_id: current_user).take
    respond_to do |format|
      if @permits.update(permit_params)
        format.html { redirect_to root_path}
        flash[:success] = "Permit successfully updated"
        format.json { render :show, status: :ok, location: @user }
      else
        format.html { render :edit }
        format.json { render json: @user.errors, status: :unprocessable_entity }
      end
    end
  end

  private
  # Use callbacks to share common setup or constraints between actions.
  def set_permit
    @permits = Permit.find(params[:id])
  end

  # Never trust parameters from the scary internet, only allow the white list through.
  def permit_params
    params.require(:permit).permit(:vehicle_type, :name, :studentid, :department, :carplate, :duration, :permitstart, :permitend)
  end
end

This is the form filled by user

<% provide(:title, 'New Permit') %>
<h1>Permit Application</h1>

<div class="row">
  <div class="col-md-6 col-md-offset-3">
    <%= form_for(@permits) do |f| %>

        <%= f.label :"Vehicle" %>
        <%= f.text_field :vehicle_type, class: 'form-control' %>

        <%= f.label :"License Plate" %>
        <%= f.text_field :carplate, class: 'form-control' %>

        <%= f.label :"Student ID" %>
        <%= f.text_field :studentid, class: 'form-control' %>

        <%= f.label :name %>
        <%= f.text_field :name, class: 'form-control' %>

        <%= f.label :"Department of applicant" %>
        <%= f.text_field :department, class: 'form-control' %>

        <%= f.label :permit_start %>
        <%= f.date_select :permitstart, class: 'form-control' %>

        <%= f.label :permit_end %>
        <%= f.date_select :permitend,  class: 'form-control'  %>


        <%= f.submit "Submit", class: "btn btn-primary" %>
    <% end %>
  </div>
</div>

Upvotes: 0

Views: 74

Answers (1)

max
max

Reputation: 102055

You could add an additional action between new and create.

# config/routes.rb
resources :permit do
  collection do
    post :confirm
  end
end

The reason that we use POST even if the route does not create a resource is that we don't want to pass any user info in the request URL.

class PermitsController < ApplicationController
  # POST /permits/confirm
  def confirm
    @fields = %i[vehicle_type, carplate, studentid, name, department, permitstart, permitend]
    @permit = current_user.permits.build(permit_params)
    render :new and return unless @permit.valid? 
  end
end

render :new and return unless @permit.valid? shortcuts the process and renders the :new form again if the input is not valid in the first place.

Since we are using POST we need a form for both the new.html.erb and confirm.html.erb all duplicating all those inputs would not be great so lets extract them to a partial:

<% # /views/permits/_inputs.html.erb %>
<% 
  input_options ||= {}
  input_options[:class] ||= 'form-control'
%>

<%= f.label :"Vehicle" %>
<%= f.text_field :vehicle_type, input_options%>

<%= f.label :"License Plate" %>
<%= f.text_field :carplate, input_options %>

<%= f.label :"Student ID" %>
<%= f.text_field :studentid, input_options %>

<%= f.label :name %>
<%= f.text_field :name, input_options %>

<%= f.label :"Department of applicant" %>
<%= f.text_field :department, input_options %>

<%= f.label :permit_start %>
<%= f.date_select :permitstart, input_options %>

<%= f.label :permit_end %>
<%= f.date_select :permitend,  input_options  %>

So lets point the new.html.erb form so that it submits to /permits/confirm:

<% provide(:title, 'New Permit') %>
<h1>Permit Application</h1>

<div class="row">
  <div class="col-md-6 col-md-offset-3">
    <%= form_for(@permits, url: '/permits/confirm_permits_path') do |f| %>
       <% render partial: :inputs %>
       <%= f.submit "Submit", class: "btn btn-primary" %>
    <% end %>
  </div>
</div>

And create a /views/permits/confirm.html.erb view:

<% provide(:title, 'Confirm Permit application') %>
<h1>Confirm Permit application</h1>

<div class="row">
  <div class="col-md-6 col-md-offset-3">
    <%= form_for(@permits) do |f| %>
       <% render partial: :inputs, input_options: { readonly: 'readonly' } %>
    <% end %>
  </div>
</div>

Upvotes: 1

Related Questions