Как настроить форму заказа с опциями, относящимися к категории товаров - PullRequest
0 голосов
/ 04 ноября 2019

Цель

Я хотел бы создать форму заказа, где пользователи могут заказать один продукт. После заполнения product_category пользователь может выбрать

  • продукт, принадлежащий к product_category
  • количество каждой опции, принадлежащей product_category.

Текущее состояние

То, как я сейчас настраиваю свой код, вызывает проблемы, когда форму нужно создавать заново:

  • , когда проблема с проверкойtriggered, (1) product_category, (2) product и (3) опции пусты, но резервирование также сохраняется, создавая ситуацию, когда резервирование сохраняется дважды.

=> Я знаю, что это потому, что я сначала сохраняю резервирование в контроллере, а затем опции, но я не знаю, как решить эту проблему (например, оно сохраняется, когда проверка выполняетсясрабатывает и после того, как пользователь правильно заполнил форму).

Код

Модели

class Order < ApplicationRecord
  belongs_to :store
  belongs_to :product

  has_many :order_options, dependent: :destroy
  has_many :options, through: :order_options
  accepts_nested_attributes_for :order_options
end

class OrderOption < ApplicationRecord
  belongs_to :option
  belongs_to :order
  accepts_nested_attributes_for :option
end

class Option < ApplicationRecord
  belongs_to :product_category
  has_many :order_options, dependent: :destroy
  has_many :orders, through: :order_options
end

class ProductCategory < ApplicationRecord
  belongs_to :store
  has_many :products, dependent: :destroy
  accepts_nested_attributes_for :products, allow_destroy: true
  has_many :options, dependent: :destroy
  accepts_nested_attributes_for :options, allow_destroy: true
end

Order_controller

class OrdersController < ApplicationController
  # skip_before_action :authenticate_user!
  def new
    @user = current_user
    @store = Store.find(params[:store_id])
    @order = Order.new
    @order.build_order_contact
    @product_category_list = @store.product_categories
    @all_options = @store.options
    @products = []
    @options = []
    if params[:product_category].present?
      @products = ProductCategory.find(params[:product_category]).products
      @options = ProductCategory.find(params[:product_category]).options
    else
    end
    if request.xhr?
      respond_to do |format|
        format.json {
        render json: {products: @products, options: @options}
      }
        format.js
      end
    end

    authorize @order
  end

  def create
    @user = current_user
    @store = Store.find(params[:store_id])
    @order = Order.new(order_params)
    @order.store = @store
    authorize @order
    if @order.save
      params[:order_options_attributes].each do |order_option|
        if @option = Option.find_by(id: order_option[:option_id])
          @option_quantity = order_option[:option_quantity]
          @order.options << @option
          order_option = @order.order_options.where(option: @option)
          order_option.update(option_quantity: @option_quantity)
        end
      end
      redirect_to store_path(@store)
    else
      @product_category_list = @store.product_categories
      render 'new'
    end
  end

views / orders / new.js

$("#product_options").html("<%= escape_javascript(render partial: 'option_fields', collection: @options) %>");


$("#dynamic-products").empty();
<% @products.each do |pro| %>
    $("#dynamic-products").append('<option value="<%= pro.id %>"><%= pro.name %></option>')
<% end %>

views / orders / new.html.erb

<%= simple_form_for [@store, @order] do |f|%>
  <%= f.simple_fields_for :products do |product| %>
    <%= product.input :product_category, collection: @product_category_list, prompt: "Select type of product", label:false,
      input_html:{
      id: "product_category"
    }%>

  <%= f.association :product, collection: @products,  input_html:{
    value: @products.object_id,
    id: "dynamic-products"
  } %>

  <div class="product_category-options" id="product_options">
  </div>
  <% end %>
<% end %>

<script >
  // dynamic products and options for change category
  $(document).on("change", "#product_category", function(){
    var product_category = $(this).val();


    $.ajax({
      url: "/stores/<%= @store.id %>/orders/new",
      method: "GET",
      // dataType: "json",
      dataType: "script",
      data: {product_category: product_category},
      error: function (xhr, status, error) {
        console.error('AJAX Error: ' + status + error);
      },
      success: function (response) {
    }
  });
  });
  // dynamic products and option for releading form (e.g. new)
   $(document).ready(function(){
    var product_category = $("#product_category").val();

    $.ajax({
      url: "/stores/<%= @store.id %>/orders/new",
      method: "GET",
      dataType: "json",
      data: {product_category: product_category},
      error: function (xhr, status, error) {
        console.error('AJAX Error: ' + status + error);
      },
      success: function (response) {
    }
  });
  });

</script>

views / orders / _option_fields.html.erb

<div class="product_option order-form-quantity-row border-bottom col col-sm-10">
  <div class="product_option_name order-form-quantity-name">
    <strong>  <%= option_fields.name %></strong>
  </div>

  <div class="order-form-input">
    <%= hidden_field_tag("order_options_attributes[]option_id", option_fields.id ) %>
    <%= select_tag("order_options_attributes[]option_quantity", options_for_select((0..9)), {class:'form-control col col-sm-12'} ) %>

  </div>
</div>

1 Ответ

1 голос
/ 04 ноября 2019

Это очень сложно и неправильно. Все, что вам действительно нужно, это что-то вроде:

<%= simple_form_for([@store, @order]) do |f| %>
  <% f.simple_fields_for(:order_options) do |ff| %>
    <%= ff.association :option %>
    <%= ff.input :option_quantity %>
  <% end %>
   # ...
<% end %>

class OrdersController
  # Use callbacks to DRY your code
  before_action :set_store, only: [:new, :create, :index]

  def new
    @order = @store.order.new
    # seed the record to create the inputs
    5.times { @order.order_options.build }
    authorize @order
  end

  def create
    @order = @store.orders.new(order_params) do |order|
       order.user = current_user
    end

    if @order.save
       redirect_to @order.store
    else
       render :new
    end
  end

  def set_store
    @store = Store.find(params[:store_id])
  end

  def order_params
    params.require(:order)
          .permit(:foo, :bar, 
             order_options_attributes: [:option_id, :option_quantity]
          )
  end 
end

Вам не нужно принимать вложенные атрибуты для опции, если вы не позволяете пользователям создавать их на лету, что не похоже нахорошая идея, поскольку у вас уже есть 100 уровней сложности в одном компоненте.

Вам также не нужно когда-либо делать params[:order_options_attributes].each do |order_option| и выполнять итерацию по вложенным атрибутам. На самом деле никогда не делайте этого, так как это сводит на нет всю цель использования вложенных атрибутов.

Когда вы используете установщик order_options_attributes=, созданный accepts_nested_attributes Rails будет обрабатывать назначение атрибутов новым экземплярам. of order_options и сделает это до сохранения записи. Когда вы вызываете функцию сохранения, в транзакции все сохраняется сразу, что позволяет избежать большинства проблем, с которыми вы сталкиваетесь.

Вы можете использовать validates_associated для запуска проверок параметров order_options перед сохранением.

Если вытогда хотите использовать AJAX, чтобы украсить его, не стесняйтесь. Но вы действительно должны начать с простой и синхронной настройки, чтобы понять, как работают вложенные атрибуты.

В целом этот код, похоже, страдает от быстрого перехода. Начните с настройки только основ (то есть просто создавая заказ на продукт). Проверьте это - рефакторинг - и затем добавьте больше возможностей. Если вы пытаетесь сделать все сразу, вы обычно получаете пожар в мусорном баке.

...