replica410
@replica410 (陽 有村)

Are you sure you want to delete the question?

If your question is resolved, you may close it.

Leaving a resolved question undeleted may help others!

We hope you find it useful!

Laravelでのルーティング時のエラーについて

解決したいこと

Laravelでのフリーマーケットサイトのルーティングを行なっています。
今しているのが、自分が商品名や商品画像を設定し「出品」を押して投稿するという動作なのですが、出品ボタンを押してから更新されるまでの間にエラーが出てしまい困っています。
調べた際にこのエラーが出た時の原因で名前の書き間違いなどが出てきたのですが、自力でチェックしたところ原因が見つけられず困っているので、是非皆さんのお力を貸して欲しいと思い質問させて頂きました。
不慣れなので、情報として足りない部分(このファイルのコードも載せていないと原因が分からない等)があればご指摘下さい。

発生している問題・エラー


BadMethodCallException
Method App\Http\Controllers\ItemsController::index does not exist.

```web.php(ルーティング)
<?php

/*
|--------------------------------------------------------------------------
| Web Routes
|--------------------------------------------------------------------------
|
| Here is where you can register web routes for your application. These
| routes are loaded by the RouteServiceProvider within a group which
| contains the "web" middleware group. Now create something great!
|
*/
Auth::routes();
// トップページをログイン画面後に表示させたい場合はmiddlewareを書かないとエラーが出る
Route::get('/', function () {
    return view('layouts.top');
})->middleware('auth');
// お気に入り一覧
Route::resource('likes', 'LikesController')->only([
    'index', 'store', 'destroy'
]);

Route::resource('items', 'ItemsController');

Route::resource('profile', 'ProfileController')->only([
    'store', 'destroy'
]);
Route::get('users/{id}/exhibitions', 'ItemsController@exhibitions')->name('users.exhibitions');


Route::get('/items/{id}/edit_image', 'ItemsController@editImage')->name('items.edit_image');

Route::patch('/items/{id}/edit_image', 'ItemsController@editImage')->name('items.update_image');

Route::get('/profile/{id}/edit', 'ProfileController@edit')->name('profile.edit');
Route::patch('/profile/{id}', 'ProfileController@update')->name('profile.update');
Route::get('/profile/{id}/edit_image', 'ProfileController@editImage')->name('profile.edit_image');
Route::patch('/profile/{id}/edit_image', 'ProfileController@updateImage')->name('profile.update_image');

Route::resource('profile', 'ProfileController')->only([
  'show',
]);

ItemsController.php(出品関連のコントローラー)
<?php

namespace App\Http\Controllers;

use Illuminate\Http\Request;
use App\Items;
use App\User;
use App\Http\Requests\ItemsRequest;
use App\Http\Requests\ItemsImageRequest;
use App\Http\Requests\ProfileImageRequest;
use App\Services\FileUploadService;

class ItemsController extends Controller
{
    public function __construct()
    {
        $this->middleware('auth');
    }

    public function exhibitions()
    {
        $items = Items::where('user_id', \Auth::user()->id)->get();
        return view('users.exhibitions',[
            'title' => '出品商品一覧',
            'items' => $items,
        ]);
    }
    
    //新規出品
    public function create()
    {
        return view('items.create',[
        'title' => '商品を出品' 
        ]);
    }

    public function store(ItemsRequest $request, FileUploadService $service)
    {
        // 画像投稿処理
        $path = $this->saveImage($request->file('image'));
        $path= '';
        $image = $request->file('image');
        if(isset($image) === true ){
            // publicディスク(storage/app/public/)のphotosディレクトリに保存
            $path = $image->store('photos', 'public');
        }
        
        Items::create([
            'user' => \Auth::user(),
            'categories' => Categories::all(),
            'description' => $request->description,
            'image' => $path, //ファイルパスを保存
        ]);
        session()->flash('success', '商品を出品しました');
        return redirect()->route('users.exhibitions');
    }

     // 購入詳細
    public function show($id)
    {
        //
    }

     // 商品情報の編集
    public function edit(Items $items)
    {
        return view('items.edit',[
            'title' => '出品編集',
            'items' => $items,
        ]);
    }

    public function update(Request $request, $id)
    {
        $items = Items::find($id);
        $items->update($request->only(['description']));
        session()->flash('success', '出品しました');
        return redirect()->route('items.exhibitions');
    }

    public function destroy($id)
    {
        $items = Items::find($id);
        $items ->delete();
        \Session::flash('success', '商品を削除しました');
        return redirect()->route('items.exhibitions');
    }
    
    public function editImage($id)
    {
        $items = Items::find($id);
        return view('items.edit_image',[
            'title' => '画像変更処理',
            'items' => $items, 
        ]);
    }
    
    public function updateImage($id, ItemsImageRequest $request, FileUploadService $service)
        {
          
        //画像投稿処理
        $path = '';
        $image = $request->file('image');
          
        if(isset($image) === true){
              // publicディスク(storage/app/)のphotosディレクトリに保存
        $path = $image->store('photos', 'public');
        }
          
        $items = Items::find($id);
          
          // 変更前の画像の削除
        if($items->image !== ''){
              // publicディスクから、該当の投稿画像($user->image)を削除
              \Storage::disk('public')->delete(\Storage::url($items->image));
        }
          
        $items->update([
            'image' => $path, //ファイル名を保存    
        ]);
          
        session()->flash('success', '画像を変更しました!');
        return redirect()->route('users.exhibitions', $id);
        }
      
        private function saveImage($image){
                // 画像投稿処理
        $path = '';
        if(isset($image) === true){
               // publicディスク(storage/app/)のphotosディレクトリに保存
           $path = $image->store('photos', 'public');
        }
            return $path;; // 画像が存在しない場合は空文字
        }

}

exhibitions.blade.php(出品一覧画面ビュー)
@extends('layouts.top')
 
@section('title', $title)
 
@section('content')
  <h1>{{ $title }}</h1>
  <a href="{{route('items.create')}}">新規出品</a>
  <ul>
      @forelse($items as $item)
      <li class="items">
          <div class="items_content">
            <div class="items_body_heading">
                <div class="items_body_main_img">
                    @if($items->image !== '')
                        <img src="{{ asset('storage/' . $items->image) }}">
                    @else
                        <img src="{{ asset('images/no_image.png') }}">
                    @endif
                </div>
                    <div class="items_body_main_comment">
                        {{ $items->description }}
                    </div>
            </div>
            <div class="items_body_main">
                商品名:{{ $items->name }}
                {{ $items->price }}
            </div>
            <div class="items_category">
                カテゴリ:{{ $items->category_id }}
                ({{ $items->created_at }})
            </div>
            <div class="items_body_footer">
                [<a href="{{ route('items.edit', $items) }}">編集</a>]
                  <form class="delete" method="post" action="{{ route('items.destroy', $items) }}">
                    @csrf
                    @method('DELETE')
                    <input type="submit" value="削除">
            </div>
          </div>
      </li>
      @empty
          <li>出品している商品はありません。</li>
      @endforelse
  </ul>
@endsection
create.blade.php(新規出品画面ビュー)
@extends('layouts.top')

@section('title', $title)

@section('content')
  <h1>{{ $title }}</h1>
  <form method="POST" 
  action="{{ action('ItemsController@store') }}"
  enctype="multipart/form-data">
      @csrf
      <div>
          <label>
            商品名:
            <input type="text" name="name">
          </label>
      </div>
      <div>
          <label>
              商品説明:
              <input type="text" name="description">
          </label>
      </div>
      <div>
          <label>
              価格:
              <input type="text" name="price">
          </label>
      </div>
      <div>
          <label>
              カテゴリー:
          </label>
      </div>
      <div>
        <label>
          画像を選択:
          <input type="file" name="image">
        </label>
      </div>
      <input type="submit" value="出品">
  </form>
@endsection

自分で試したこと

名前の書き間違いなどの確認、ルーティングのチェック

0

1Answer

ルーティング内でItemsControllerに対してRoute::resourceしてますね。
Route::resource('アドレス', 'コントローラ')で一気に各種機能にルーティングを設定してくれるという便利なものですが、もしコントローラ内にそれぞれの機能に対応する関数が無かった場合、このようなエラーが出てしまいます。
今回はItemsControllerindex関数が含まれていないというのが直接的な原因でしょう。
今後どのようにしたいかにも依りますが、index関数をそれらしく作成するか、あるいは既にコード内に存在しているもいののように、['only' => [【機能の文字列】]]を付けてindex関数が無くても良いようにRoute::resourceでの機能を制限するか、それが解決方法としては早いものになるかと思われます。

余談ですが、関連しそうなコードを大凡載せて頂けていることで、(多分)速攻で問題点が分かりました。ありがとうございます!

1Like

Comments

  1. @replica410

    Questioner

    ありがとうございます。
    アドバイス頂いた通り、出品一覧で作成したexhibitionsファイルをindexに名前を変更しresourceに含まれたルートを全て使うようにしたら無事出来ました。
    それとは別のエラーが出てしまいまだ画面は表示出来ていませんが、それは別で聞こうかと思います。

Your answer might help someone💌