🎄teratailクリスマスプレゼントキャンペーン2024🎄』開催中!

\teratail特別グッズやAmazonギフトカード最大2,000円分が当たる!/

詳細はこちら
Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Q&A

解決済

1回答

1483閲覧

複数レコードの一括保存(チェックボックスは使わず、フォームのnameに記述があるもののみ保存)

kao.

総合スコア2

Ruby

Rubyはプログラミング言語のひとつで、オープンソース、オブジェクト指向のプログラミング開発に対応しています。

Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

0グッド

0クリップ

投稿2020/12/09 03:25

編集2020/12/10 01:52

前提・実現したいこと

Railsで冷蔵庫管理システムを作ろうと挑戦しています。個人冷蔵庫(refrige)に食材(food)を複数入力し複数のレコードを、一括で保存したいと考えています。
■■最終的には、フォームの全部に入力している場合にのみ保存ではなく。食材名(:food_name)の欄への記述があるもののみ保存したいです。■■
■■画面でエラーは出ず、保存されない状態です。現在はパラメーターまではデータが届いているようですが、保存されません。@formにfood_idとrefrige_idが渡っていないのが問題です■■
日付がnilになっていた件は解決しました!!(カレンダーから選択にしたらnilが解消しました)

発生している問題・エラーメッセージ

●food_collection_paramsの中身
12: def create
13: byebug
=> 14: @form = Form::FoodCollection.new(food_collection_params)
15: if @form.save
16: redirect_to refrige_foods_path
17: else

(byebug) food_collection_params
<ActionController::Parameters {"foods_attributes"=><ActionController::Parameters {"0"=><ActionController::Parameters {"food_name"=>"ああああ", "product_name"=>"あああああ", "category_id"=>"3", "purchase_date"=>"111111", "sell_by"=>"33333", "shop"=>"ああああ"} permitted: true>, "1"=><ActionController::Parameters {"food_name"=>"いいいい", "product_name"=>"いいいいい", "category_id"=>"6", "purchase_date"=>"222222", "sell_by"=>"22222", "shop"=>"いいいい"} permitted: true>} permitted: true>, "refrige_id"=>"1"} permitted: true>

●@formの中身
12: def create
13: byebug
14: @form = Form::FoodCollection.new(food_collection_params)
=> 15: if @form.save
16: redirect_to refrige_foods_path
17: else
18: flash.now[:alert]
19: render :new
(byebug) @form
<Form::FoodCollection:0x00007fd2c17663d0 @foods=[#<Food id: nil, food_name: "ああああ", product_name: "あああああ", category_id: 3, purchase_date: "0011-11-11", sell_by: nil, shop: "ああああ", refrige_id: nil, created_at: nil, updated_at: nil>, #<Food id: nil, food_name: "いいいい", product_name: "いいいいい", category_id: 6, purchase_date: nil, sell_by: nil, shop: "いいいい", refrige_id: nil, created_at: nil, updated_at: nil>], @refrige_id="1">

#該当コード

該当のソースコード

●foodテーブル●

class CreateFoods < ActiveRecord::Migration[6.0] def change create_table :foods do |t| t.string :food_name, null: false t.string :product_name t.integer :category_id, null: false t.date :purchase_date, null: false t.date :sell_by t.string :shop t.references :refrige, foreign_key: true t.timestamps end end end html

●foodモデル●

class Food < ApplicationRecord extend ActiveHash::Associations::ActiveRecordExtensions belongs_to :category with_options presence: true do validates :food_name validates :purchase_date validates :refrige_id end validates :genre_id, numericality: { other_than: 1 } belongs_to :refrige, validate: false end

●foodcollectionモデル●

class Form::FoodCollection < Form::Base FORM_COUNT = 2 attr_accessor :foods, :refrige_id def initialize(attributes = {}) super attributes self.foods = FORM_COUNT.times.map { Food.new } unless foods.present?  end def foods_attributes=(attributes) self.foods = attributes.map { |_, v| Food.new(v) } end def save Food.transaction do foods.map do |food| food.save if food.food_name end rescue => e false end end

●foods コントローラー●

def new @form = Form::FoodCollection.new end def create byebug @form = Form::FoodCollection.new(food_collection_params) if @form.save redirect_to refrige_foods_path else render :new end end private def food_collection_params params.require(:form_food_collection) .permit(foods_attributes: [:food_name, :product_name, :category_id, :purchase_date, :sell_by, :shop],).merge(refrige_id: params[:refrige_id]) end

試したこと

@formにfood_idとrefrige_idがnilになっている事。
attr_accessor :foods, :refrige_id への記述。ストロングパラメーターにも.merge(refrige_id: params[:refrige_id])の記述をしましたが変わりません。
まとめずに1レコードずつの保存はできました。@formにfood_idとrefrige_idの渡し方分かる方いらっしゃいませんでしょうか。

補足情報(FW/ツールのバージョンなど)

こちらの記事を参考にさせていただきました。
ど素人です。不明な点や、至らない記述があれば追加します。よろしくお願いします。
【Rails 5】モデルを一括登録する手順

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

Form::FoodCollection の foods それぞれに refrige_id がセットされてないので
保存に失敗しているように見えます。

refrige_id をセットするには、以下のどちらかを行います
0. Form::FoodCollection の中で refrige_id をセットする
0. Form::FoodCollection にnewで渡すparamsの形を整える

1. Form::FoodCollection の中で refrige_id をセットする

ruby

1def save 2 foods.map do |food| 3 food.refrige_id = self.refrige_id 4 food.save if food.food_name 5 end 6end

(transaction を rescueすると巻き戻らないし、saveは例外を出さないのでsave!を使いましょう)

2. Form::FoodCollection にnewで渡すparams の形を整える

ruby

1# 今はこうなっています。 2{ 3 0 => {food}, 4 1 => {food}, 5 "refrige_id"=>"1" 6} 7 8#これを以下のようにしたい 9{ 10 0 => {food, "refrige_id"=>"1"}, 11 1 => {food, "refrige_id"=>"1"}, 12}

たぶんこうすると上の形に出来ますが、ForbiddenAttributesErrorが出るかもしれません。

def food_collection_params params.require(:form_food_collection) .permit(foods_attributes: [:food_name, :product_name, :category_id, :purchase_date, :sell_by, :shop]) .map{ |key, value| [key. value.merge(refrige_id: params[:refrige_id]] }.to_h end

投稿2020/12/09 16:56

neko_daisuki

総合スコア2090

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

kao.

2020/12/10 00:47 編集

早速回答していただきありがとうございました( ; ; ) ⑴の方法を試し無事保存できました◎ def save Food.transaction do foods.map do |food| food.refrige_id = self.refrige_id food.save if food.food_name end end end しかし、保存できてもできなくても、保存できた場合の画面に遷移してしまいます。 if @form.save redirect_to refrige_foods_path neko_daisukiさんがおっしゃってくれたように ※(transaction を rescueすると巻き戻らないし、saveは例外を出さないのでsave!を使いましょう) の問題だと思います。がどのように記述したらfoodsコントローラーのcreateの動きをしてくれるのか分かりません。 neko_daisukiさんのヒントを参考に色々試してみる予定ですが、解決策があれば詳しく教えていただきたいです????‍♀️
neko_daisuki

2020/12/10 01:25 編集

@form.save が偽を返せば render :new されます。 今は foods.map が偽になることはないので必ず真になってますね。 どのような場合に偽にするのか考える必要がありますが、 ひとつでも保存に失敗した場合に全部の保存を取り消し、戻り値を偽にするには以下のようにします。 大丈夫だとは思いますが責任はもてませんのでよくテストしてください。 Food.transaction do foods.each do |food| return if food.food_name.blank? food.refrige_id = self.refrige_id unless food.save raise ActiveRecord::Rollback end end end transaction の挙動は以下の記事が参考になりました https://qiita.com/kazutosato/items/fbc147bc907c246e3868
kao.

2020/12/10 02:16

度々ありがとうございます????‍♀️ 知識の無い状態で、一括保存に挑戦したため、ほとんどが引用でコードの記述に対しての理解不足が大きな問題でした。丁寧に教えていただき有難うございます。試してみます!neko_daisukiさんからいただいた内容を参考にしながら、きちんと理解できるよう頑張りたいと思います。 本当に有難うございました!!!!
kao.

2020/12/10 02:33

上手く解決できなければ、もう一度新しく質問しようと思います! 質問していた保存に関しては、無事解決できましたので、ベストアンサーを付けさせていただきました????
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.36%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問