ページの作成
親となるページを選択してください。
親ページに紐づくページを子ページといいます。
例: 親=スポーツ, 子1=サッカー, 子2=野球
子ページを親ページとして更に子ページを作成することも可能です。
例: 親=サッカー, 子=サッカーのルール
親ページはいつでも変更することが可能なのでとりあえず作ってみましょう!
| この記事の要点 |
|
コントローラーの生成
# 基本
php artisan make:controller PostController
# CRUD 7 メソッド入りのリソースコントローラ
php artisan make:controller PostController --resource
# API リソース(create/edit 無し、5 メソッド)
php artisan make:controller Api/PostController --api
# モデルと連携(型ヒント自動)
php artisan make:controller PostController --resource --model=Post
# シングルアクション(__invoke のみ)
php artisan make:controller SendWelcomeMail --invokable
基本的なコントローラ
namespace App\Http\Controllers;
use App\Models\Post;
use Illuminate\Http\Request;
class PostController extends Controller
{
// 一覧
public function index()
{
$posts = Post::latest()->paginate(20);
return view('posts.index', compact('posts'));
}
// 詳細
public function show(Post $post) // ← Route Model Binding
{
return view('posts.show', compact('post'));
}
// フォーム
public function create()
{
return view('posts.create');
}
// 保存
public function store(Request $request)
{
$data = $request->validate([
'title' => 'required|string|max:255',
'body' => 'required|string',
]);
$post = Post::create($data + ['user_id' => $request->user()->id]);
return redirect()->route('posts.show', $post)
->with('status', '保存しました');
}
// 更新
public function update(Request $request, Post $post)
{
$this->authorize('update', $post);
$post->update($request->validate([
'title' => 'required|string|max:255',
'body' => 'required|string',
]));
return redirect()->route('posts.show', $post);
}
// 削除
public function destroy(Post $post)
{
$this->authorize('delete', $post);
$post->delete();
return redirect()->route('posts.index');
}
}
リソースルートとコントローラの対応
| HTTP | URI | メソッド | 名前 |
|---|---|---|---|
| GET | /posts | index | posts.index |
| GET | /posts/create | create | posts.create |
| POST | /posts | store | posts.store |
| GET | /posts/{post} | show | posts.show |
| GET | /posts/{post}/edit | edit | posts.edit |
| PUT/PATCH | /posts/{post} | update | posts.update |
| DELETE | /posts/{post} | destroy | posts.destroy |
シングルアクションコントローラ
// php artisan make:controller SendWelcomeMail --invokable
class SendWelcomeMail extends Controller
{
public function __invoke(Request $request, User $user)
{
Mail::to($user)->send(new Welcome($user));
return back()->with('status', 'sent');
}
}
// ルート定義
Route::post('/users/{user}/welcome', SendWelcomeMail::class);
依存性注入(DI)
// メソッド DI(HTTP リクエストごとに解決)
public function store(Request $request, PostService $service)
{
$post = $service->create($request->validated());
return redirect()->route('posts.show', $post);
}
// コンストラクタ DI
class PostController extends Controller
{
public function __construct(private PostService $service) {}
public function index()
{
return $this->service->latest();
}
}
バリデーション(3 通り)
// 方法 A: コントローラ内 (簡単)
public function store(Request $request)
{
$data = $request->validate([
'title' => 'required|string|max:255',
'tags' => 'array',
'tags.*'=> 'string|max:50',
]);
}
// 方法 B: FormRequest クラスに切り出し (推奨)
// php artisan make:request StorePostRequest
class StorePostRequest extends FormRequest
{
public function authorize(): bool { return true; }
public function rules(): array
{
return [
'title' => 'required|string|max:255',
'body' => 'required|string',
];
}
public function messages(): array
{
return ['title.required' => 'タイトルは必須です'];
}
}
public function store(StorePostRequest $request) {
$post = Post::create($request->validated());
}
// 方法 C: Validator ファサード(細かい制御)
$validator = Validator::make($request->all(), [...]);
if ($validator->fails()) {
return back()->withErrors($validator)->withInput();
}
ミドルウェアの適用
class PostController extends Controller
{
public function __construct()
{
$this->middleware('auth');
$this->middleware('verified')->only(['store', 'update', 'destroy']);
$this->middleware('can:update,post')->except(['index', 'show']);
}
}
// あるいはルート定義側で
Route::resource('posts', PostController::class)
->middleware(['auth', 'verified']);
レスポンスのバリエーション
// ビュー
return view('posts.show', compact('post'));
// JSON
return response()->json(['ok' => true, 'data' => $post]);
// API Resource
return new PostResource($post);
return PostResource::collection($posts);
// ステータスコード
return response()->json([...], 201);
abort(404, 'Post not found');
abort_if(! $user->is_admin, 403);
// リダイレクト
return redirect('/dashboard');
return redirect()->route('posts.show', $post);
return back()->withInput()->withErrors(['title' => 'NG']);
// ファイルダウンロード
return response()->download(storage_path('app/report.pdf'));
return response()->streamDownload(fn () => print($csv), 'report.csv');
ページネーション
public function index()
{
$posts = Post::with('user')->latest()->paginate(20);
return view('posts.index', compact('posts'));
}@foreach ($posts as $post)
@endforeach
{{ $posts->links() }} {{-- ページネーション UI --}}
{{ $posts->total() }} {{-- 合計件数 --}}
{{ $posts->currentPage() }}
FAQ
Q: コントローラを名前空間で分けたい
A: App\Http\Controllers\Admin\UserController 等。make:controller Admin/UserController でフォルダ付きで生成。
Q: 単体テストでコントローラをテストしたい
A: HTTP テスト機能で $this->get('/posts')->assertStatus(200)。コントローラ単体ではなくルート経由でテストするのが Laravel 流。
Q: API と Web で同じコントローラを使い回したい
A: アクション内で $request->wantsJson() や $request->is('api/*') で分岐、または別コントローラに分けるのが推奨。
ページの作成
親となるページを選択してください。
親ページに紐づくページを子ページといいます。
例: 親=スポーツ, 子1=サッカー, 子2=野球
子ページを親ページとして更に子ページを作成することも可能です。
例: 親=サッカー, 子=サッカーのルール
親ページはいつでも変更することが可能なのでとりあえず作ってみましょう!
同階層のページ
- インストールと設定
- クイックスタート & チュートリアル(初心者向け)
- クイックスタート & チュートリアル(中級者向け)
- ルーティング
- Bladeテンプレート(ビュー/レイアウト)
- コントローラー
- マイグレーションとテーブル定義
- データベースの設定
- Eloquentモデル (ORM)
- SQLとクエリビルダー
- バリデーション
- .envファイルの設定値へのアクセス
- 動作環境による分岐処理
- configフォルダ配下の設定値へのアクセス
- assetヘルパーを利用したpublicフォルダへのアクセス
- storageフォルダへのアクセス
- アプリケーション名の変更
- メンテナンス
- ログイン画面(認証システム)の作成
- ログインの必須化
- ログインユーザー情報の取得
- ルートの認証化
- 本番サーバーへのデプロイ方法
- 多言語化
- csrf_field
- ファイルのダウンロード
- CSVのアップロードおよび読み込み(maatwebsite/excel)
- ページタイトルの設定
- コマンド一覧
- エラー一覧
- SQLの実行ログ出力方法
- キャッシュのクリア
- Selectの結果の最初もしくは最後に任意の値を追加する方法
- ajaxでPOST通信する際の注意点
- ソーシャルログインの実装
- セッション情報の確認
- ログイン、ユーザー登録、パスワードリセット後のリダイレクト先の変更方法
- redirectやreturn viewにメッセージを付与する方法
- クッキー(cookie)の設定と取得
- クラスの再読み込み
- csrfの有効時間を変更する方法
- ViewComposerを用いてviewに共通の値を付与する方法
- View::shareを用いて共通の値を各ビューに渡す方法
- ミドルウェアを用いた処理の共通化
- Middleware内でAuth::check()などを使用する方法
- Controller以外でリダイレクトする方法
- セッションの値の取得/保存/更新/削除
- $requestの値を変更する方法
- 常時SSL化
- ページング(ページネーション)をする方法
- vue.jsとの連携
- Vue.jsと連携するSPA実行環境構築
- .envの値をvue.jsで参照する方法
- vue.jsを本番環境にリリースする方法
- could not find driver(Windows, MySQL編)
人気ページ
- 1 Eclipseで「サーバーに追加または除去できるリソースがありません。」の原因と対処法
- 2 tomcat の起動 / 停止ログと catalina.log・catalina.out の違い
- 3 JavaScript base URL 取得方法|window.location.origin と SSR/Node.js 対応
- 4 YouTube Data API v3 エラー一覧|403/400/404 の主要原因と切り分け
- 5 Spring Frameworkのアノテーション一覧
- 6 Laravel エラー一覧|500/Blade/DB 接続/ルーティングの代表エラー
- 7 3Dグラフィックスとは|モデリング/レンダリング/主要ソフトウェア (Blender / Maya)
- 8 【Spring】@Valueアノテーションとは
- 9 CATALINA_HOME の確認方法 (Linux / Mac)
- 10 【Spring】@Autowiredアノテーションとは
最近更新/作成されたページ
- Laravel キャッシュクリア完全ガイド(cache:clear / config:clear / 2026-05-18 07:42:07
- プロジェクトの作成と削除 2026-05-18 07:42:07
- インストール直後にNetbeansが反応しない 2026-05-18 07:42:07
- 動画やチャンネルの検索 2026-05-18 07:42:07
- APIキー取得方法 2026-05-18 07:42:07
- チャンネル情報の取得 2026-05-18 07:42:07
- API 入門 — Web API(REST / GraphQL / gRPC / 2026-05-18 07:42:07
- インストール(eclipseプラグイン) 2026-05-18 07:42:07
- Laravel「Dotenv values containing spaces must be surrounded 2026-05-18 07:42:07
- エラー一覧 2026-05-18 07:42:07
- curl: (51) SSL: certificate subject name '~' does not match 2026-05-18 07:42:07
- インストール方法(Windows版) 2026-05-18 07:42:07
- JSONから配列に変換 2026-05-18 07:42:07
- 処理を一定時間待つ 2026-05-18 07:42:07
- A non well formed numeric value encountered 2026-05-18 07:42:07
コメントを削除してもよろしいでしょうか?