質問をすることでしか得られない、回答やアドバイスがある。

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

ただいまの
回答率

90.47%

  • Ruby on Rails

    9064questions

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

関連付けたモデルのデータをフォーム送信する方法が分かりません(rails)

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 1,347

yamady

score 163

お世話になっております。
Ruby on Railsで「ユーザーと紐付いている別モデル」にフォーム送信するようにしたいと考えているのですが、どうやってやるべきなのかがなかなか見えず困っています・・・

具体的に言うと、Facebookカバー写真を設置するために「UserImage」モデルを作成しており、これにアップロードするための方法が分からないということです。
CarrierWaveを使用しているため、ユーザーモデルにカバー写真が紐付けられていません。

開発環境
・Rails 5.0.0.1
・Carrier Wave
・Devise

該当するソースコード

views/devise/registration/edit.html.erb

<% provide(:title, "アカウントの編集") %>
<%= form_for(resource, as: resource_name, url: registration_path(resource_name), html: { method: :put }) do |f| %>
<%= devise_error_messages! %>
<section class="mypage-profile">
  <div class="container mypage-form title">
    <div class="title-text">
      <h2>プロフィールの編集</h2>
    </div>
    <form>
      <div class="form-group">
        <%= f.label :name, "ユーザー名" %>
        <%= f.text_field :name, :placeholder => "ユーザー名", :value => @user.name, class: "form-control" %>
      </div>

      <!-- ここがUserImageモデルに紐付いているカラム -->
      <div class="form-group">
        <%=f.label :image,"ユーザー画像" %>
        <%= f.file_field :image %>
      </div>
      <div class="form-group">
        <%=f.label :image,"カバー画像" %>
        <%= f.file_field :cover_image %>
      </div>
      <!-- /ここがUserImageモデルに紐付いているカラム -->

      <div class="form-group">
        <%= f.label :description, "自己紹介" %>
        <%= f.text_area :description, :placeholder => "自己紹介", :size => "10x10", :value => @user.description, class: "form-control" %>
      </div>
      <%= f.submit "登録内容を変更する", class: "btn btn-form" %><br />
    </form>
    <% end %>
  </div>
</section>

user.rb

class User < ApplicationRecord
  # Include default devise modules. Others available are:
  # :confirmable, :lockable, :timeoutable and :omniauthable
  devise :database_authenticatable, :registerable,
         :recoverable, :rememberable, :trackable, :validatable,
         :omniauthable
  has_many :reviews, dependent: :destroy
  has_many :user_images

  class << self
    def find_for_facebook_oauth(auth)
      where(provider: auth.provider, uid: auth.uid).first_or_create do |user|
        user.email = auth.info.email
        user.password = Devise.friendly_token[0,20]
        user.name = auth.info.name
      end
    end

    def create_unique_string
      SecureRandom.uuid
    end

    def create_unique_email
      User.create_unique_string + "@example.com"
    end
  end

  private

  def user_params
    params.require(:user).permit(:name, :email)
  end
end

user_image.rb

class UserImage < ApplicationRecord
  belongs_to :user

  mount_uploader :image, ThumbnailUploader
  mount_uploader :cover_image, ThumbnailUploader
end

users_controller.rb

class UsersController < ApplicationController

  def new
    @user_image = User.new(create_params)
    2.times { @user.user_images.build }
  end

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

  private

  def user_params
    params(:user).permit(:name, :email, :background, :description)
  end

  def create_params
    params.require(:users).permit(user_images_attributes: [:image, :cover_image])
  end
end

試してみたこと

form_for の中にform_forしてみましたが、やり方が悪いのかできませんでした・・・汗

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

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+1

nested_attribute_for と fields_forを使いましょう。
後、Carrier Waveの使い方が間違っている気がします。
Carrier Waveは一つのモデルに対し一つしか画像を添付できません。
なので、user_imageに別カラム(例えばhoge:integer)を作り
hogeが0ならカバー1ならユーザーみたいにする必要があると思います。
で以下が修正した内容です。
動作は未確認です。あとは工夫して見てください。

  def new
    @user_image = User.new(create_params)
    2.times {|i| @user.user_images.build(hoge: i)}
  end
class User < ApplicationRecord
  # Include default devise modules. Others available are:
  # :confirmable, :lockable, :timeoutable and :omniauthable
  devise :database_authenticatable, :registerable,
         :recoverable, :rememberable, :trackable, :validatable,
         :omniauthable
  has_many :reviews, dependent: :destroy
  has_many :user_images
  accepts_nested_attributes_for :user_images#追加
  class << self
    def find_for_facebook_oauth(auth)
      where(provider: auth.provider, uid: auth.uid).first_or_create do |user|
        user.email = auth.info.email
        user.password = Devise.friendly_token[0,20]
        user.name = auth.info.name
      end
    end

    def create_unique_string
      SecureRandom.uuid
    end

    def create_unique_email
      User.create_unique_string + "@example.com"
    end
  end

  private

  def user_params
    params.require(:user).permit(:name, :email,user_images_attributes: [:image])#追加
  end
end
<% provide(:title, "アカウントの編集") %>
<%= form_for(resource, as: resource_name, url: registration_path(resource_name), html: { method: :put }) do |f| %>
<%= devise_error_messages! %>
<section class="mypage-profile">
  <div class="container mypage-form title">
    <div class="title-text">
      <h2>プロフィールの編集</h2>
    </div>
    <form>
      <div class="form-group">
        <%= f.label :name, "ユーザー名" %>
        <%= f.text_field :name, :placeholder => "ユーザー名", :value => @user.name, class: "form-control" %>
        <%= f.hidden :hoge %>
      </div>

      <!-- ここがUserImageモデルに紐付いているカラム -->
<%= f.fields_for :user_images_attributes do |a| %>
      <div class="form-group">
        <%= a.label :image, a.object.hoge==0 ? 'カバー画像':'ユーザー画像' %>
        <%= a.file_field :image %>
      </div>
<% end %>
      <!-- /ここがUserImageモデルに紐付いているカラム -->

      <div class="form-group">
        <%= f.label :description, "自己紹介" %>
        <%= f.text_area :description, :placeholder => "自己紹介", :size => "10x10", :value => @user.description, class: "form-control" %>
      </div>
      <%= f.submit "登録内容を変更する", class: "btn btn-form" %><br />
    </form>
    <% end %>
  </div>
</section>

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2017/06/27 23:01

    mokeさま、詳しいご説明をいつも本当にありがとうございます。非常に参考になる回答となりました。ここまでご説明いただけると、立ち返ってできそうです!

    キャンセル

  • 2017/06/27 23:33 編集

    どういたしまして、定期的に似たような質問に答えている気がしますが。
    Carrier Waveは一つのモデルに対し一つしか画像を添付できません。
    が基本なので、わからなくなったら基本に立ち返って下さいね
    頑張って下さい👍

    キャンセル

  • 2017/06/28 18:47

    ありがとうございます(そして、同じような質問すみません><
    基本に立ち返るくせをつけていこうと思います!!
    どうぞよろしくお願いいたします。

    キャンセル

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

  • ただいまの回答率 90.47%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる

同じタグがついた質問を見る

  • Ruby on Rails

    9064questions

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