1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

LaravelでTODOアプリを作成する

Posted at

はじめに

構築環境

下記の環境でLaravelを用いてTODOアプリを作成しました。

  • Windows11
  • PHP 8.1.30
  • Laravel Framework 10.48.22
  • MySQL 8.4
  • Docker

筆者はDockerの知識があまりないため、Chat-GPTをフル活用しました。

手順概要

Windows環境にPHPやLaravelをインストールしたくなかったため、下記の手順で作成しました。

Windowsのフォルダは空の状態でDocker環境構築(コンテナ環境を優先とするマウント設定)

コンテナ内でLaravelプロジェクトを作成

コンテナ内のLaravelプロジェクトをwindowsのフォルダにコピー

Windows環境を優先するマウント設定でDocker環境構築

Laravelの各ファイルを編集

完成

作成手順

1. 環境設定

1.1 Dockerのインストール

  • Docker Desktopをダウンロード: https://www.docker.com/products/docker-desktop
  • インストーラーを実行し、指示に従ってインストール
  • WSL 2のインストールが求められた場合は、指示に従ってインストール
  • インストール完了後、Docker Desktopを起動

1.2 プロジェクトの準備

  • コマンドプロンプトを開き、プロジェクトを作成したいディレクトリに移動

  • 以下のコマンドを実行してプロジェクトディレトリを作成:

    mkdir todo-list
    cd todo-list
    

1.3 Dockerファイルの作成

  • プロジェクトルートに「docker-compose.yml」ファイルを作成し、以下の内容を追加:

    version: '3'
    services:
      app:
        build:
          context: .
          dockerfile: Dockerfile
        ports:
          - "8000:8000"
        volumes:
          - .:/app
        depends_on:
          - db
      db:
        image: mysql:5.7
        environment:
          MYSQL_DATABASE: todo_list
          MYSQL_ROOT_PASSWORD: root_password
          MYSQL_USER: user
          MYSQL_PASSWORD: password
        volumes:
          - dbdata:/var/lib/mysql
    volumes:
      dbdata:
    
  • 同じディレクトリに「Dockerfile」を作成し、以下の内容を追加:

        FROM php:8.1-fpm
    
        RUN apt-get update && apt-get install -y \
            git \
            curl \
            libpng-dev \
            libonig-dev \
            libxml2-dev \
            zip \
            unzip
    
        RUN docker-php-ext-install pdo_mysql mbstring exif pcntl bcmath gd
    
        COPY --from=composer:latest /usr/bin/composer /usr/bin/composer
    
        # 作業ディレクトリを作成
        WORKDIR /var/www/app
    
        # プロジェクトファイルをコンテナにコピー
        COPY . .
    
        # 環境変数設定:Composerをrootで実行できるようにする
        ENV COMPOSER_ALLOW_SUPERUSER=1
    
        # 既存のcomposer.jsonが存在するか確認し、存在しない場合はプロジェクトを作成
        # 存在している場合はcomposer installを実行して依存関係をインストール
        RUN if [ ! -f composer.json ]; then \
            rm -rf * && \
            composer create-project --prefer-dist laravel/laravel .; \
            else composer install; \
            fi
    
        # Laravel開発サーバーの起動
        CMD php artisan serve --host=0.0.0.0 --port=8000
    
    
    

2.1 Laravelプロジェクトの作成

docker-compose up -dでビルド&立ち上げ。立ち上げ時にコンテナ内でLaravelプロジェクトが作成される。

docker cp <コンテナ名>:/var/www/app ./appでコンテナ内のLaravelプロジェクトをWindows11環境にもコピー

例: docker cp todo-list-app-1:/var/www/app ./app

DockerfileのVolumeをコメントアウトして再度docker-compose up -d

2.2 .envファイルの設定

  • .envファイルを開き、以下の部分を編集:

    DB_CONNECTION=mysql
    DB_HOST=db
    DB_PORT=3306
    DB_DATABASE=todo_list
    DB_USERNAME=user
    DB_PASSWORD=password
    

3. Dockerコンテナの起動

  • 以下のコマンドを実行してDockerコンテナを起動:

    docker-compose up -d
    

4. モデルとマイグレーションの作成

  • 以下のコマンドを実行してTaskモデルとマイグレーションを作成:

    docker-compose exec app php artisan make:model Task -m
    
  • database/migrations/xxxx_xx_xx_xxxxxx_create_tasks_table.phpを開き、以下のように編集:

    <?php
    
    use Illuminate\Database\Migrations\Migration;
    use Illuminate\Database\Schema\Blueprint;
    use Illuminate\Support\Facades\Schema;
    
    return new class extends Migration
    {
        /**
        * Run the migrations.
        * マイグレーションの実行時にテーブルを作成します
        *
        * @return void
        */
        public function up()
        {
            // tasksテーブルを作成
            Schema::create('tasks', function (Blueprint $table) {
                $table->id();  // 自動インクリメントID
                $table->string('title');  // タスクのタイトル
                $table->text('description')->nullable();  // タスクの詳細(null許容)
                $table->integer('status')->default(value: 1);  // タスクのステータス(1=未着手, 2=進行中, 3=完了)
                $table->timestamps();  // 作成日時、更新日時を自動的に管理
            });
        }
    
        /**
        * Reverse the migrations.
        * マイグレーションのロールバック時にテーブルを削除します
        *
        * @return void
        */
        public function down()
        {
            // tasksテーブルを削除
            Schema::dropIfExists('tasks');
        }
    };
    
    
  • app\app\Models\Task.phpを作成:

    <?php
    
    namespace App\Models;
    
    use Illuminate\Database\Eloquent\Factories\HasFactory;
    use Illuminate\Database\Eloquent\Model;
    
    class Task extends Model
    {
        use HasFactory;
    
        // テーブルに対応するモデルが操作可能なカラムを指定
        protected $fillable = [
            'title',        // タスクのタイトル
            'description',  // タスクの詳細
            'status',       // タスクのステータス(1=未着手, 2=進行中, 3=完了)
        ];
    
        // ステータスをラベルに変換するアクセサ
        public function getStatusLabelAttribute()
        {
            return match ($this->status) {
                1 => '未着手',
                2 => '進行中',
                3 => '完了',
                default => '不明',
            };
        }
    }
    
  • マイグレーションを実行:

    docker-compose exec app php artisan migrate
    
  • マイグレーションを実行(既存テーブルを全て削除する場合):

    docker-compose exec app php artisan migrate:fresh
    

5. コントローラーの作成

  • 以下のコマンドを実行してTaskControllerを作成:

    docker-compose exec app php artisan make:controller TaskController --resource
    
  • app\app\Http\Controllers\TaskController.phpを開き、以下のように編集:

    <?php
    
    namespace App\Http\Controllers;
    
    use Illuminate\Http\Request;
    use App\Models\Task;
    
    class TaskController extends Controller
    {
        // タスク一覧画面を表示
        public function showIndex()
        {
            $tasks = Task::all();  // すべてのタスクを取得
            return view(view: 'tasks.index', data: compact(var_name: 'tasks'));  // ビューにデータを渡す
        }
    
        // タスク作成画面を表示
        public function showStore()
        {
            return view(view: 'tasks.store');  
        }
    
        // タスク編集画面を表示
        public function showUpdate(Task $task)
        {
            return view('tasks.update', compact(var_name: 'task'));  
        }
    
        // 新しいタスクの作成
        public function store(Request $request)
        {
            $request->validate([
                'title' => 'required|max:255',
            ]);
    
            Task::create([
                'title' => $request->title,
                'description' => $request->description,
                'status' => $request->status,
            ]);
    
            return redirect()->route('tasks.index');
        }
    
        // タスクの更新
        public function update(Request $request, Task $task)
        {
            $request->validate([
                'title' => 'required|max:255',
            ]);
        
            $task->update([
                'title' => $request->title,
                'description' => $request->description,
                'status' => $request->status,
            ]);
        
            return redirect()->route('tasks.index');
        }
    
        // タスクの削除
        public function delete(Task $task)
        {
            $task->delete();
    
            return redirect()->route('tasks.index');
        }
    }
    
    
    

6. ルーティングの設定

  • routes/web.phpを開き、以下のように編集:

    <?php
    
    use Illuminate\Support\Facades\Route;
    use App\Http\Controllers\TaskController;
    
    
    // タスク管理用のルート
    Route::prefix('tasks')->name('tasks.')->controller(TaskController::class)->group(function () {
        Route::get('/index', 'showIndex')->name('index');
        Route::get('/store', 'showStore')->name('store');
        Route::post('/store', 'store')->name('store');
        Route::get('/update/{task}', 'showUpdate')->name('update');
        Route::put('/update/{task}', 'update')->name('update');
        Route::delete('/delete/{task}', 'delete')->name('delete');
    });
    

7. ビューの作成

  • 共通レイアウトの作成

    • app\resources\views\layouts\app.blade.phpディレクトリを作成し、以下のファイルを作成します:
          <!DOCTYPE html>
          <html lang="ja">
          <head>
              <meta charset="UTF-8">
              <meta name="viewport" content="width=device-width, initial-scale=1.0">
              <title>タスク管理システム</title>
              <link rel="stylesheet" href="https://stackpath.bootstrapcdn.com/bootstrap/4.5.2/css/bootstrap.min.css">
          </head>
          <body class="bg-light">
              <!-- ヘッダー部分 -->
              <header class="bg-dark text-white py-4 mb-4">
                  <div class="container text-center">
                      <h1 class="display-4">タスク管理システム</h1>
                  </div>
              </header>
              <!-- メインコンテンツ部分 -->
              <main class="container">
                  <div class="bg-white p-4 rounded shadow-sm">
                      @yield('content')
                  </div>
              </main>
          </body>
          </html>
      
  • ビューファイルの作成

    • app\resources\views\tasksディレクトリを作成し、以下のファイルを作成します:

      • index.blade.php(タスク一覧画面)

        @extends('layouts.app')
        
        @section('content')
            <h1>タスク一覧</h1>
            <a href="{{ route('tasks.store') }}" class="btn btn-primary mb-3">新規タスク作成</a>
            
            @if($tasks->count() > 0)
                <table class="table">
                    <thead>
                        <tr>
                            <th>タイトル</th>
                            <th>状態</th>
                            <th>操作</th>
                        </tr>
                    </thead>
                    <tbody>
                        @foreach($tasks as $task)
                            <tr>
                                <td>{{ $task->title }}</td>
                                <!-- statusはアクセサを使用して変換された値を表示 -->
                                <td>{{ $task->status_label }}</td>
                                <td>
                                    <a href="{{ route('tasks.update', $task->id) }}" class="btn btn-sm btn-warning">編集</a>
                                    <form action="{{ route('tasks.delete', $task->id) }}" method="POST" style="display: inline-block;">
                                        @csrf
                                        @method('DELETE')
                                        <button type="submit" class="btn btn-sm btn-danger" onclick="return confirm('本当に削除しますか?')">削除</button>
                                    </form>
                                </td>
                            </tr>
                        @endforeach
                    </tbody>
                </table>
            @else
                <p>タスクはありません</p>
            @endif
        @endsection
        
      • store.blade.php(タスク作成画面)

        @extends('layouts.app')
        
        @section('content')
            <h1>新規タスク作成</h1>
            <form action="{{ route('tasks.store') }}" method="POST">
                @csrf
                <div class="form-group">
                    <label for="title">タイトル</label>
                    <input type="text" class="form-control" id="title" name="title" required>
                </div>
                <div class="form-group">
                    <label for="description">説明</label>
                    <textarea class="form-control" id="description" name="description" rows="3"></textarea>
                </div>
                <div class="form-group">
                    <label for="status">状態</label>
                    <select class="form-control" id="status" name="status">
                        <option value="1">未着手</option>
                        <option value="2">進行中</option>
                        <option value="3">完了</option>
                    </select>
                </div>
                <button type="submit" class="btn btn-primary">作成</button>
                <a href="{{ route('tasks.index') }}" class="btn btn-secondary">戻る</a>
            </form>
        @endsection
        
        
      • update.blade.php(タスク編集画面)

        @extends('layouts.app')
        
        @section('content')
            <h1>タスク編集</h1>
            <form action="{{ route('tasks.update', $task->id) }}" method="POST">
                @csrf
                @method('PUT')
                <div class="form-group">
                    <label for="title">タイトル</label>
                    <input type="text" class="form-control" id="title" name="title" value="{{ $task->title }}" required>
                </div>
                <div class="form-group">
                    <label for="description">説明</label>
                    <textarea class="form-control" id="description" name="description" rows="3">{{ $task->description }}</textarea>
                </div>
                <div class="form-group">
                    <label for="status">状態</label>
                    <select class="form-control" id="status" name="status">
                        <option value="1" {{ $task->status == 1 ? 'selected' : '' }}>未着手</option>
                        <option value="2" {{ $task->status == 2 ? 'selected' : '' }}>進行中</option>
                        <option value="3" {{ $task->status == 3 ? 'selected' : '' }}>完了</option>
                    </select>
                </div>
                <button type="submit" class="btn btn-primary">更新</button>
                <a href="{{ route('tasks.index') }}" class="btn btn-secondary">キャンセル</a>
            </form>
        @endsection
        
        

これらのファイルは、HTMLとLaravelのBladeテンプレート構文を使用して、フォームと一覧表示を実装しています。各ファイルは@extendsディレクティブを使用してレイアウトを継承し、@sectionディレクティブを使用してコンテンツを定義しています。

8. アプリケーションの動作確認

  • ブラウザでhttp://localhost:8000/tasks/indexにアクセスし、TODOアプリが正常に動作することを確認。
  • そのほかのページが動作することも確認。

9. ログの確認

  • コードの変更は自動的にDockerコンテナに反映されます

  • ログの確認:

    docker-compose logs app
    

10. Dockerコンテナの停止

  • 開発終了時、以下のコマンドでDockerコンテナを停止:

    docker-compose down
    

時間があれば盛り込みたい内容

  • ログの追加
  • モデルやサービスに処理を分散させる
  • シーダーの追加
  • 画像付きの説明にする

所感

Chat-GPTはDocker周りの解答を意外とミスしており、非常に時間がかかった。
Laravel関連の解答はこちらが事前に情報を提示すれば意図通りの解答をすることが多い。
Docker関連の操作はコマンドよりVscodeから操作したほうが楽。

1
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?