4

我对 Ruby on Rails 相当陌生,并立即开始使用 rails 4。

我已经成功地嵌套了一个食谱和一个成分模型,以便它们可以以相同的形式添加。接下来,我想将数量嵌套在成分中,以便也可以在同一表格中添加。直到即将将成分的数量插入数据库之前,一切似乎都运行良好,因此我相信 recipes_controller 中的强参数有问题。但我会在下面发布完整的代码。我正在为表单使用 simple_form。

感谢您的帮助!

这是我的模型:

class Recipe < ActiveRecord::Base
  has_many :comments, dependent: :destroy
  has_many :ingredients, dependent: :destroy
  accepts_nested_attributes_for :ingredients, :reject_if => lambda { |a| a[:name].blank? }, :allow_destroy => true
  validates :title, presence: true
  validates :desc, presence: true
end


class Ingredient < ActiveRecord::Base
  belongs_to :recipe
  has_many :quantities, dependent: :destroy
  accepts_nested_attributes_for :quantities, :reject_if => lambda { |a| a[:name].blank? }, :allow_destroy => true
end

class Quantity < ActiveRecord::Base
  belongs_to :ingredient
end

这是recipes_controller

class RecipesController < ApplicationController

  def new
    @recipe = Recipe.new
    3.times do 
      ingredient = @recipe.ingredients.build
      1.times {ingredient.quantities.build }
    end
  end

  def create
    @recipe = Recipe.new(params[:recipe].permit(:title, :desc, ingredients_attributes: [:id, :recipe_id, :name, :_destroy, quantities_attributes: [:id, :ingredient_id, :amount, :unit, :_destroy]]))

    if @recipe.save
      redirect_to @recipe
    else
      render "new"
    end
  end

  def show
    @recipe = Recipe.find(params[:id])
  end


  def edit
    @recipe = Recipe.find(params[:id])
  end


  def update
    @recipe = Recipe.find(params[:id])

    if @recipe.update(params[:recipe].permit(:title, :desc))
      redirect_to @recipe
    else
      render 'edit'
    end
  end


  def destroy
    @recipe = Recipe.find(params[:id])
    @recipe.destroy

    redirect_to recipes_path
  end





  def index
    @recipes = Recipe.all
  end

  private
    def post_params
      params.require(:recipe).permit(:title, :desc, ingredients_attributes: [:id, :recipe_id, :name, :_destroy, quantities_attributes: [:id, :ingredient_id, :amount, :unit, :_destroy]])
    end
end

然后我使用简单的表格通过部分创建配方、成分和数量的表格。

_形式:

<%= simple_form_for @recipe do |f| %>
    <%= f.error_notification %>


    <%= f.input :title %>
    <%= f.input :desc %>



    <%= f.simple_fields_for :ingredients do |builder| %>

    <%= render "ingredient_fields", :f => builder %>


    <% end %>

    <p class="links">
    <%= link_to_add_association 'add ingredient', f, :ingredients %>
    <p class="links">
    <%= f.error :base%>
    <%= f.submit %>

<% end %>

从 _ingredients_fields 渲染:

<div class="nested-fields">
    <%= f.input :name, label: "Ingredient" %>



    <%= f.simple_fields_for :quantities do |builder| %>

    <%= render "quantities_fields", :f => builder %>

    <% end %>

    <%= link_to_remove_association "remove", f %>
</div>

从 _quantities_fields 渲染:[已编辑]

<div class="nested-fields">
    <%= f.input :amount %>
    <%= f.input :unit %>
</div>

尝试添加新配方会导致以下日志语句:

Started POST "/recipes" for 127.0.0.1 at 2013-10-29 14:15:40 +0100
Processing by RecipesController#create as HTML
  Parameters: {"utf8"=>"✓", "authenticity_token"=>"t6LKgDLwAxaU9xo2ipyCM+j1yfVF9WrI8AoGTX+gRkw=", "recipe"=>{"title"=>"Pancakes", "desc"=>"Tasty", "ingredients_attributes"=>{"0"=>{"name"=>"Milk", "quantities_attributes"=>{"0"=>{"amount"=>"1", "unit"=>"Cup"}}, "_destroy"=>"false"}}}, "commit"=>"Create Recipe"}
  [1m[35m (0.1ms)[0m  begin transaction
  [1m[36mSQL (3.5ms)[0m  [1mINSERT INTO "recipes" ("created_at", "desc", "title", "updated_at") VALUES (?, ?, ?, ?)[0m  [["created_at", Tue, 29 Oct 2013 13:15:40 UTC +00:00], ["desc", "Tasty"], ["title", "Pancakes"], ["updated_at", Tue, 29 Oct 2013 13:15:40 UTC +00:00]]
  [1m[35mSQL (0.4ms)[0m  INSERT INTO "ingredients" ("created_at", "name", "recipe_id", "updated_at") VALUES (?, ?, ?, ?)  [["created_at", Tue, 29 Oct 2013 13:15:40 UTC +00:00], ["name", "Milk"], ["recipe_id", 27], ["updated_at", Tue, 29 Oct 2013 13:15:40 UTC +00:00]]
  [1m[36m (7.8ms)[0m  [1mcommit transaction[0m
Redirected to http://www.projectcookbook.dev/recipes/27
Completed 302 Found in 22ms (ActiveRecord: 11.7ms)
4

2 回答 2

2

您对 _quantities 和 _ingredients 部分使用了类似的渲染,这是错误的。在 _quantities_field 你不需要

<%= f.simple_fields_for :quantities do |builder| %>
<%= render "quantities_fields", :f => builder %>
<% end %>

并且应该调整

<%= f.input :name, label: "Quantity" %>

在 _quantities_fields 中。

UPD

我认为问题出:reject_if在成分模型的 -clause 中。它应该是

:reject_if => lambda { |a| a[:amount].blank? }

bc 在这里您为数量指定条件,而不是为成分

关于代码样式:1)在控制器中,最好使用私有方法的相关名称作为强参数:recipe_params而不是post_params然后使用它来创建新配方@recipe = Recipe.new(recipe_params)

2) 如果两个配方使用相似的配方,则配方、成分和数量之间的当前关联将导致成分重复。原因是belongs_to,它定义了单一关联。尝试另一种方法(如下)。

顺便提一句。最近我已经回答了类似的问题。看看:如何在嵌套的 Rails 表单中引用模型的现有实例?

于 2013-10-29T13:57:58.700 回答
0

我觉得你在这部分缺少

<%= simple_form_for @recipe do |f| %>

它应该是

<%= simple_nested_form_for @recipe do |f| %>

于 2014-10-27T06:13:41.560 回答