こんにちは。Laravelでenumを使いたい時、PHPやLaravel本体ではサポートされていないのでComposer経由でパッケージを入れることになります。
今はbensampo/laravel-enum
を使っており、とりあえず不自由ない感じなので、紹介しようと思います。
railsで初めてenumを知ったのですが、C言語の時点で既にあったのですね。
私は「限られた値しか持てないデータ型」と認識してます。
例えば注文情報テーブルに支払方法を保存する時、データベースのorders
テーブルにpayment_type
カラムを持たせるとします。card, transfer, cash
など支払方法名だけがorders.payment_type
に入るようにしつつ、表示時は「クレジットカード」や「銀行振込」といった文字列で扱いたい…なんてケースはよくあると思いますが、こういう場合に便利です。
composerコマンドでさくっとインストール完了です。
$ composer require bensampo/laravel-enum Using version ^1.19 for bensampo/laravel-enum ./composer.json has been created Loading composer repositories with package information ... ... ... Writing lock file Generating autoload files
マイグレーションはenumクラスの後に作った方が(多分)効率が良いので、まずはenumクラスを生成します。
$ php artisan make:enum PaymentType Enum created successfully.
なおデフォルトではapp/Enums
ディレクトリが生成され、その中にPHPファイルが自動生成されます。ディレクトリを切り分けたい場合、例えばapp/Enums/Order
に作りたいのであればOrder/PaymentType
となります。
このパラメータ、app/Enums
を起点とした相対パスとして扱われるので、例えば$ php artisan make:enum ../Models/Enums/TestEnum
と記述すれば、app/Models/EnumsTestEnum.php
が生成されます。
ディレクトリ構造はプロジェクトによって様々だと思いますので、お好みで。
日本語化
続いて公式の方法に従って
resources/lang/ja/enums.php
を作成し、以下のように記述します。フレームワークのlocale設定に従ったenums.php
を読みに行きますので、config/app.php
のlocale
を日本(ja)に設定しておきましょう。'ja', // 中略 ];[ PaymentType::CREDIT_CARD => 'クレジットカード', PaymentType::TRANSFER => '銀行振込', PaymentType::CASH => '現金', ], ];マイグレーション
生成
orders
テーブルにpayment_type
というenum型のカラムを追加します。$ php artisan make:migration add_column_payment_type --table=orders Created Migration: 2019_05_**_******_add_column_payment_type編集
$table->enum()
はLaravel 5.8から使用できます。直接指定できるのはいいですね。
また「マイグレーションはクラス作成の後で」と前述したのは、enum型カラムの定義にenumクラスの値を適用できるためです。もちろん配列をべた書きしてもいいのですが、この方がすっきり分かりやすいかなと。enum('payment_type', PaymentType::getValues()) ->default(PaymentType::CREDIT_CARD) ->after('id'); }); } }プロパティのキャスト
Eloquent
ではモデルクラスのプロパティに定義することで、インスタンス生成時に別の型に自動キャストしてくれる機能があります。protected $dates = [];
に入れておけば自動でCarbon
にキャストしてくれるとか、そういうやつです。laravel-enum
でも 同様の機能があり、以下のコードで自動キャストが可能です。PaymentType::class, ]; }こうしておくことで何が嬉しいのかと言うと、以下のようにbladeでの記述が減ります。
// 自動キャストしない場合 {{PaymentType::getDescription($order->payment_type)}} // 自動キャストする場合 {{$order->payment_type->description}}当初は自動キャストせず前者のように書いてたのですが
と思い至り、調べた結果見つけた方法でした。
地味ですが少しでもコード量は減らしたいので!
enum、いいですね。昔はこういった実装をする際、DBにマスターテーブル作ってconst.phpとか作って定数を列挙して…みたいなことをしてましたが、大抵const.phpが肥大化して管理が億劫になってました。定義がコード化されて気分がいいです。