現在Ruby on Rails にて、Twitterのような投稿サービスを作っております。求職用のポートフォリオとしての制作です。
元々progateで勉強をしていたため入力フォームを「form_tag」で理解していました。そして今「form_for」or「form_with」へログインページのフォーム部分を入れ替えているところです。その最中、form_tagでは動いていたログイン機能で、エラーが出るようになってしまいました。
Rails.application.routes.draw do get "home/top" => 'home#top' get "posts/index" => "posts#index" get "posts/new" => "posts#new" post "posts/create" => "posts#create", as: :posts get "posts/:id/edit" => "posts#edit" get "posts/:id" => "posts#show" patch 'posts/:id' => 'posts#update', as: :post post "posts/:id/destroy" => "posts#destroy" get "signnup" => "users#new" post "users/create" => "users#create", as: :users post "login_guest" => "users#login_guest" get "login" => "users#login_form" post "login" => "users#login" post "logout" => "users#logout" get "users/index" => "users#index" get "users/:id" => "users#show" get "users/:id/edit" => "users#edit" patch "users/:id/update" => "users#update", as: :user post "users/:id/destroy" => "users#destroy" end
users.controller class UsersController < ApplicationController before_action :ensure_correct_user, {only: [:edit, :update]} before_action :authenticate_user, {only: [:edit, :update]} before_action :forbid_login_user, {only: [:new, :create, :login_form, :login]} def ensure_correct_user if @current_user.id != params[:id].to_i flash[:notice] = "できません" redirect_to("/login") end end def index @users = User.all.order(created_at: :desc) end def show @user = User.find_by(id: params[:id]) end def new @user = User.new end def login_guest @user = User.find_by(email: "guest_user@example.com", password: "12345") if @user session[:user_id] = @user.id flash[:notice] = "ゲストユーザーとしてログインしました" redirect_to("/posts/index") end end def create @user = User.new(user_params) if @user.save session[:user_id] = @user.id flash[:notice] = "ユーザー登録完了" redirect_to("/users/#{@user.id}") else flash[:notice] = "ユーザー登録失敗" render("users/new") end end def login_form end def login @user = User.find_by(email: params[:email]) if @user && @user.authenticate(params[:password]) session[:user_id] = @user.id flash[:notice] = "ログイン成功" redirect_to("/posts/index") else @error_message = "メールアドレスかパスワードが違います" @email = params[:email] @password = params[:password] render("users/login_form") end end def logout session[:user_id] = nil flash[:notice] = "ログアウトしました" redirect_to("/login") end def edit @user = User.find_by(id: params[:id]) end def update @user = User.find_by(id: params[:id]) if @user.update(user_params) flash[:notice] = "ユーザー情報を変更しました" redirect_to("/users/#{@user.id}") else render("/users/edit") end end def destroy @user = User.find_by(id: params[:id]) @user.destroy redirect_to("/users/index") flash[:notice] = "削除しました" end private def user_params params.require(:user).permit(:name, :email, :password, :image) end end
application_controller class ApplicationController < ActionController::Base before_action :set_current_user def top end def set_current_user @current_user = User.find_by(id: session[:user_id]) end def authenticate_user end def forbid_login_user if @current_user flash[:notice] = "すでにログインしています" redirect_to("/posts/index") end end end
home_controller class HomeController < ApplicationController before_action :forbid_login_user, {only: [:new, :create, :login_form, :login]} def top end end
users/login_form.html.erb <div class = "form-error"> <% if @error_message %> <%= @error_message %> <% end %> </div> <div class="user_login_item"> <h1>既存ユーザーログイン</h1> <div class="user_login"> <%= form_for @user do |f| %> <%= f.label :email, "メールアドレス" %><br> <%= f.text_field :email %><br><br> <%= f.label :password, "パスワード" %><br> <%= f.text_field :password %><br><br> <%= f.submit "ログイン" %> <% end %> </div> </div>
schema.rb ActiveRecord::Schema.define(version: 2020_01_18_054634) do create_table "destroys", force: :cascade do |t| t.datetime "created_at", precision: 6, null: false t.datetime "updated_at", precision: 6, null: false end create_table "posts", force: :cascade do |t| t.text "content" t.datetime "created_at", precision: 6, null: false t.datetime "updated_at", precision: 6, null: false t.string "image" t.integer "user_id" t.string "title" t.float "rate" end create_table "sessions_users", force: :cascade do |t| t.string "name" t.string "email" t.string "password" t.string "image" t.datetime "created_at", precision: 6, null: false t.datetime "updated_at", precision: 6, null: false end create_table "users", force: :cascade do |t| t.string "name" t.string "email" t.string "image" t.datetime "created_at", precision: 6, null: false t.datetime "updated_at", precision: 6, null: false t.string "password_digest" end end
models/User class User < ApplicationRecord has_secure_password mount_uploader :image,ImageUploader validates :name, {presence: true} validates :email, {presence: true} validates :password, {presence: true} def posts return Post.where(user_id: self.id) end paginates_per 8 end
user/edit <div class="user-index"> <h1>ユーザー情報の編集</h1> <% @user.errors.full_messages.each do |message| %> <div class="form-error"> <%= message %> </div> <% end %> <div class="user-edit-container"> <%= form_for @user do |f| %> <div class="user-edit-item"> <%= f.label :name, "ユーザー名(※必須)", class: "form" %><br> <%= f.text_field :name %><br> <%= f.label :image, "画像", class: "form" %><br> <%= f.file_field :image %><br> <%= f.label :email, "メールアドレス(※必須)", class: "form" %><br> <%= f.text_field :email %><br> <%= f.label :password, "パスワード(※必須)", class: "form" %><br> <%= f.text_field :password %><br> <%= f.submit "新規登録", class: "form" %> </div> <% end %> </div> </div>
自分で試したこととしては、form_forの@userがnilなので、login_formアクション内で@userオブジェクトを生成したりもしたのですが、特に変わらず、、
(具体的には、
・@user = User.find_by(email: params[:email], password: params[:password])
・@user = User.find_by(id: params[:id])
など。)
sessionコントローラーなどは用いずに実装したいと考えています。(できないのであれば仕方ないとは考えています)。
また、form_tagで実装すると機能するので、form_tagのままでいいかなとも思いつつ、全体としてform_forで揃えるべきだよな、、とも考えています。
login_formアクション内で@userオブジェクトに何を定義すればいいのでしょうか?もしお分かりの方がいらっしゃいましたら、アドバイスいただけると幸いです。
回答3件
あなたの回答
tips
プレビュー
バッドをするには、ログインかつ
こちらの条件を満たす必要があります。
2020/02/18 05:43
2020/02/18 05:57
2020/02/18 06:31