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

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

詳細はこちら
Ruby on Rails

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

Q&A

解決済

1回答

892閲覧

キャラクターを選択すると、そのキャラクターの全ての技が表示されるように実装したい!

hikaru-udon

総合スコア6

Ruby on Rails

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

0グッド

1クリップ

投稿2021/02/02 14:44

#前提
CharactersテーブルとMovesテーブルの二つのテーブルを用意し、Charactersの主キーをMovesで受け取り、そのidをもとにBにあるデータを全て取得し、表示させる、というような設計にしております。

#試してみたこと
Movesテーブルに外部キー(Charactersの主キー)を設定し、保存しています。
それをfindメソッドで取得してeachメソッドで表示させられるのではと思い、記述してみたものの、うまく行きません。
findの引数に数字を入れると取得はできるのですが、これではキャラクターごとに技データを取得することができないです。

こういった場合、
どのように記述できるでしょうか?
また、現時点でのコードでも間違いがありましたらご指摘いただけたらと思います。

-controller/characters

rails

1class CharactersController < ApplicationController 2 def index 3 character = Character.all 4 end 5end 6
  • views/character/index

rails

1<div class = "main"> 2 <div class="title-contents"> 3 <h1>フレームデータ</h1> 4 </div> 5 <div class="main-contents"> 6 <div class = "select-character"> 7 <div class = "contents-name"> 8 キャラクター選択 9 </div> 10 <div class='characters'> 11 <ul> 12 <li><%= link_to "キャラクター1", character_moves_path(:character_id) %></li> 13 <li><%= link_to "キャラクター2", character_moves_path(:character_id) %></li> 14 <li><%= link_to "キャラクター3", character_moves_path(:character_id) %></li> 15 </ul> 16 </div> 17 </div> 18 </div> 19</div>
  • controller/moves

rails

1class MovesController < ApplicationController 2 3 def index 4 @moves = Move.find(1) 5 end 6end
  • views/move/index

rails

1<table> 2 <tr> 3 <th>技名</th> 4 <th>コマンド</th> 5 <th>ダメージ</th> 6 <th>発生F</th> 7 <th>ガードF</th> 8 </tr> 9 <% @moves.each do |moves| %> 10 <tr> 11 <td><%= moves.moves_name %></td> 12 <td><%= moves.command %></td> 13 <td><%= moves.damage %></td> 14 <td><%= moves.startup %></td> 15 <td><%= moves.block %></td> 16 </tr> 17 <% end %> 18</table>

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

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

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

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

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

guest

回答1

0

ベストアンサー

app/models/character.rb と move.rb が以下のようになっているか確認してください。

ruby

1class Character < ApplicationRecord 2 has_many :moves 3end

ruby

1class Move < ApplicationRecord 2 belongs_to :character 3end

このとき、id が 1 の Character の moves は以下のようにして参照できます。

ruby

1Character.find(1).moves # => [move(character_id=1), move(character_id=1), move(character_id=1), ...]

コントローラーでは以下のようになります。

ruby

1# GET /characters/1 2def show 3 @character = Character.find(params[:id]) 4 @moves = @character.moves 5end

コードの問題は、

ruby

1character = Character.all

ruby

1@characters = Character.all

コントローラーの変数をビューで参照するにはインスタンス変数(@からはじまるやつ)の必要があります。
また、all が返すのはコレクションですので変数名は複数形にしましょう

ruby

1@moves = Move.find(1)

ruby

1@move = Move.find(1)

find が返すのは単一のオブジェクトです。変数の名前は単数形にしましょう。
これは each に応答しないので今のビューではエラーになるでしょう。

以下コメントへの返信です

ルーティングを rails の一般的な記述にしている事を前提にしています。

コントローラーが以下のようになっているのなら

ruby

1def index 2 @characters = Character.all 3end

次のように書けます。

erb

1<ul> 2<% @characters.each do |character| %> 3 <% # character に name というカラムがあると推定 %> 4 <li><%= link_to character.name, character_moves_path(character) %></li> 5<% end %> 6</ul>

生成されるパスは /charcters/:id/moves です。
次のようなルーティングが設定されていないとエラーになります。
対応するコントローラー、アクションは moves#index です。

resources :characters do resources :moves end

character_path(character) に変更すれば /characters/:id になります。
対応するコントローラー、アクションは characters#show です。

投稿2021/02/02 19:46

編集2021/02/03 02:57
neko_daisuki

総合スコア2090

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

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

hikaru-udon

2021/02/03 02:21

レビューありがとうございます!! アソシエーション確認しました、組めました! 命名規則など、本当に勉強になります! ご指摘していただいた通りに記述したのですが、 ActiveRecord::RecordNotFound in MovesController#index Couldn't find Character without an ID のエラーが出ます。 おそらく画面遷移時にcharacter_idが渡せていないのが原因だということが推測できました。 現在キャラクターの選択をimage_tagを用いたボタン画像で行えるよう実装しているのですが、ここにそれぞれのcharacterのidを持たせるにはどうすれば良いでしょうか?(上記コードの<li>要素のところです)
hikaru-udon

2021/02/03 06:06

ルーティングは提示していただいたものと同じでした! charactersコントローラでindexを記述し、html.erbを編集し、「link_to インスタンス変数名.name」でキャラクター名を表示できるようになりました。 ですが、クリックして飛ぼうとしたところ、先ほどと同じくエラーが出ました。 エラー文:Couldn't find Character with 'id'=character_id 指摘コード: def index @character = Character.find(params[:character_id]) @moves = @character.moves end binding.pryで止めて確認したところ、 「<ActionController::Parameters {"controller"=>"moves", "action"=>"index", "character_id"=>"character_id"} permitted: false>」 と出ました。falseが出ている理由がわからないです。 characterのerbかmovesコントローラのどちらかが記述ミスしているという解釈でよろしいのでしょうか?
neko_daisuki

2021/02/03 14:39

link_to character.name, character_moves_path(:character_id) となっていませんか? :character_id ではなく、 character です。
neko_daisuki

2021/02/03 14:56

xxx_path について character_path("ここ") や、character_moves_path("ここ") の ここに指定するのは通常 ID です。 前者では/characters/:id の :id、後者は/characters/:character_id/moves の :character_id の部分になります。 params[:character_id] について params[:character_id]は URL の一部です。 いまは、/characters/:character_id/moves の:character_idの部分がparams[:character_id]になります。 つまり xxx_path("ここ") で指定した値になります。 find について find は ID を検索します。見つからないとエラーになります。 params[:character_id] の値は "character_id" になっています。 "character_id" という ID がデータベースに存在しないのでエラーになっています。 通常、ID は数値です。 permitted: falseについて これはストロングパラメーターと言うやつで、データベースを更新するときのためのやつです。 なので今はfalseで問題ありません。
hikaru-udon

2021/02/03 17:42 編集

なるほど!パスの引数はそういう仕組みなのですね!勉強になります!! 変更点を変更したところ、parameterに{"character_id"=>"1"}と表示されるようになりました!ありがとうございます!! ですが、「Couldn't find Character without an ID」とでます。。 movesテーブルには外部キーのcharacter_idカラムを作っているので技データをとって来れると思うのですが、このエラー文の場合はそれ以前お問題ですかね?
neko_daisuki

2021/02/03 18:00

@character = Character.find(params[:character_id]) ここでそのエラーが出るのでしょうか?
hikaru-udon

2021/02/04 07:04

そうなのでしたが、すみません、paramsの後のidがcharacter_idにできてませんでした。 修正したところ、無事表示されました!!すごいです!! 教えていただいたパスと引数の仕組みに関してもっと理解を深める必要があると学ぶことができました。 解決のため、たくさんご教授いただき本当にありがとうございます! とても勉強になりました!
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.36%

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

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

質問する

関連した質問