Tìm hiểu và sử dụng Enum trong Laravel
Bài đăng này đã không được cập nhật trong 2 năm
Giới thiệu
Chào mọi người, sau một tháng thì mình đã trở lại đây (hehe).
Nếu như các bạn đã từng làm việc với các ngôn ngữ như Java, C#, C++,... thì hẳn là các bạn không còn xa lại gì với Enum. Còn đối với các bạn chưa từng nghe qua Enum thì không sao bởi vì trong bài viết này chúng ta sẽ cùng tìm hiểu về Enum và cách sử dụng Enum trong Laravel với pagake Laravel Enum.
Enum là gì?
Enum là một kiểu dữ liệu đặc biệt,  thường sử dụng cho việc định nghĩa một tập hợp cho các hằng số có giá trị cố định. Ví dụ như:
- Các ngày trong tuần (Monday, Tuesday, ... Sunday)
- Giới tính (Man, Woman, Other)
- Các mùa trong năm (Spring, Summer, Autumn, Winter)
- .....
Lợi ích của Enum:
- Giảm các lỗi gây ra bởi chuyển đổi số hoặc nhập sai số.
- Thuận lợi cho việc quản lý, dễ dàng thay đổi các giá trị trong tương lai.
- Làm cho code tường minh hơn, dễ đọc hơn, giảm việc xuất hiện bug.  
- .....
Tuy nhiên PHP nói chung và Laravel nói riêng lại không hỗ trợ trực tiếp Enum, nhưng dù vậy chúng ta vẫn có thể sử dụng Enum  gián tiếp bằng các hằng số (const) trong các class.
Trong Laravelcó một package vô cùng hữu ích hỗ trợ việc tạo và sử dụng Enum đó là Laravel Enum. Chúng ta cùng tìm hiểu package hữu ích này nhé.
Laravel Enum
Cài đặt
Yêu cầu: Laravel >= 5.4 và PHP >= 7.1
Bạn hãy mở terminal lên và chạy lệnh sau để cài đặt package:
composer require bensampo/laravel-enum
Nếu bạn sử dụng Laravel < 5.5, thì bạn cần copy dòng sau đây và thêm vào providers trong file config/app.php:
BenSampo\Enum\EnumServiceProvider
Tạo Enum và sử dụng
Ví dụ đặt ra: Giả sử trong hệ thống của mình, User có 3 quyền (role) với các giá trị tương ứng lưu trong DB là :
- Administrator = 0
- Moderator = 1
- Member = 2
Thông thường các bạn sẽ dùng file config để quản lý các giá trị này để nhằm mục đích dễ dàng thay đổi khi cần thiết. Enum cũng có tác dụng tương tự, ngoài ra nó còn làm cho code trờ nên tường mình hơn. Ở ví dụ này mình sẽ sử dụng Enum nhé:
Mình sẽ tạo một Enum tên là UserRole để lưu các quyền của User trong hệ thống và các giá trị tương ứng của nó bằng lệnh sau:
php artisan make:enum UserRole
Một Enum UserRole sẽ được tạo ra ở trong folder App\Enums, bạn cần thêm các giá trị role của User vào file này như sau:
<?php
namespace App\Enums;
use BenSampo\Enum\Enum;
final class UserRole extends Enum
{
    const Administrator = 1;
    const Moderator = 2;
    const Member = 3;
}
Vậy là chúng ta đã có một Enum đơn giản, và sử dụng nó cũng vô cùng đơn giản như sau:
UserRole::Administrator  // return 0
Các bạn có thể sử dụng Enum này ở bất cứ chỗ nào mà bạn muốn. Lưu ý rằng bạn cần phải Use Enum UserRole ở đầu mỗi file nhé:
use App\Enums\UserRole;
Một số ví dụ về sử dụng Enum đơn giản:
- 
Dùng trong migrationđể tạo giá trị mặc định cho các cột:$table->tinyInteger('role')->unsigned()->default(UserRole::Member);
- 
Dùng trong các Modelhay cácPolicyđể kiểm tra quyền admin của hệ thống:public function isAdmin() { return $this->role === UserRole::Administrator; }
- 
Dùng trong các biểu thức, phương thức, ... ở Controller:if (Auth::user()->role === UserRole::Administrator) { User::create([ 'name' => 'Peter Paker', 'nickname' => 'Spider Man', 'role' => UserRole::Moderator, ]); }
- 
........ 
Các phương thức hỗ trợ
- 
getKeys(): trả về một mảng chứa các key của Enum UserRole::getKeys() // return ['Administrator', 'Moderator', 'Member']
- 
getValues(): trả về một mảng chứa các giá trị của Enum UserRole::getValues() // return [1, 2, 3]
- 
getKey(string|int $value): trả về key của Enum $valueUserRole::getKey(2); // Returns 'Moderator' UserRole::getKey(UserRole::Moderator); // Returns 'Moderator'
- 
getValue(string $key): trả về giá trị của Enum $keyUserRole::getValue('Administrator'); // Returns 1
- 
hasKey(string $key): kiểm tra các key trong Enum có chứa $keyhay khôngUserRole::hasKey('Member'); // Returns true
- 
hasValue(string|int $value, bool $strict = true): kiểm tra các giá trị trong Enum có chứa $valuehay không. Tham số$strictdùng để bật/tắt chế độ so sánh tuyệt đối (===), giá trị mặc định của$strictlà false.UserRole::hasValue(1); // Returns true // It's possible to disable the strict type checking: UserRole::hasValue('1'); // Returns false UserRole::hasValue('1', false); // Returns true
- 
getDescription(string|int $value) : trả về mô tả của Enum $value(giá trị trả về mặc định là key).UserRole::getDescription(3); // Returns 'Member' UserRole::getDescription(UserRole::Member); // Returns 'Member'Bạn cũng có thể overridephương thức này để custom giá trị trả về:public static function getDescription($value): string { if ($value === self::Member) { return 'This is normal user of system'; } return parent::getDescription($value); }UserRole::getDescription(UserRole::Member); // Returns 'This is normal user of system'
- 
getRandomKey(): trả về một key ngẫu nhiên trong Enum. Phương thức này cực kì hữu dụng khi dùng trong việc Seed data.UserRole::getRandomKey(); // Returns 'Administrator', 'Moderator' or 'Member'
- 
getRandomValue(): trả về một giá trị ngẫu nhiên trong Enum. Phương thức này cực kì hữu dụng khi dùng trong việc Seed data.UserRole::getRandomValue(); // Returns 1, 2 or 3
- 
toArray(): Enum được trả về dưới dạng một mảng với key và value tương ứng UserRole::toArray(); // Returns ['Administrator' => 1, 'Moderator' => 2, 'Member' => 3]
- 
toSelectArray(): trả về một mảng có dạng [value => description]. UserRole::toSelectArray(); // Returns [1 => 'Administrator', 2 => 'Moderator', 3 => 'This is normal user of system']Phương thức này cực kì hữu ích khi bạn muốn đổ các giá trị role này kèm với đa ngôn ngữ tương ứng ra một Select boxđể hiển thị. Đầu tiên bạnoverridephương thứcgetDescription()và custom các giá trị trả về với đa ngôn ngữ, sau đó bạn có thể sử dụng phương thứctoSelectArray()này để lấy và đổ ra Select box.
- 
getInstance(string|int $enumValue): trả về một thể hiện của Enum UserRole::getInstance(UserRole::Administrator);
Khởi tạo và sử dụng thể hiện của Enum
Điều này thật sự hữu ích vì bạn có thể tạo các thể hiện của Enum và sử dụng chúng như là các tham số cho các phương thức.
Có nhiều cách để khởi tạo một thể hiện của Enum:
// Standard new PHP class, passing the desired enum value as a parameter
$enumInstance = new UserRole(UserRole::Member);
// Static getInstance method, again passing the desired enum value as a parameter
$enumInstance = UserRole::getInstance(UserRole::Member);
// Statically calling the key name as a method
$enumInstance = UserRole::Member();
Với thể hiện của Enum thì bạn có thể truy cập trực tiếp đến các thuộc tính của Enum:
$enumInstance->key; // Member
$enumInstance->value; // 3
$enumInstance->description; // 'This is normal user of system'
Sử dụng thể hiện của Enum làm tham số cho các phương thức:
function canManageSystem(UserRole $userRole)
{
    if ($userRole->is(UserRole::Administrator)) {
        return true;
    }
    return false;
}
$userRole1 = UserRole::getInstance(UserRole::Administrator);
$userRole2 = UserRole::Member()
canManageSystem($userRole1);  // Returns true
canManageSystem($userRole2);  // Returns false
Trong đó phương thức is() dùng để kiểm tra thể hiện Enum có bằng với các giá trị Enum hợp lệ không.
$enumInstance = UserRole::getInstance(UserRole::Administrator);
$enumInstance->is(UserRole::Administrator);  // Returns true
$enumInstance->is(UserRole::Moderator);      // Returns false
$enumInstance->is(UserRole::InvalidKey);     // Throws InvalidEnumMemberException exception
Validation Enum
Bạn có thể validate Enum Value và Enum Key bằng cách sử dụng các Enum Rule:
- Bằng Rule:
 Đối với[ 'user_role' => 'required|enum_value:' . UserRole::class, 'user_role' => 'required|enum_key:' . UserRole::class, ]Enum Value Rulethì mặc định chế độ so sánh tuyệt đối (===) sẽ được bật (enum_value:enum_class,[strict = true]). Bạn có thể tắt nó đi như sau:'user_role' => 'required|enum_value:' . UserRole::class . ',false',
- Bằng Rule Object:
 Tương tự, bạn có thể tắt chế độ so sánh tuyệt đối của[ 'user_role' => ['required', new EnumValue(UserRole::class)], 'user_role' => ['required', new EnumKey(UserRole::class)], ]Enum Value Rule Object:
 Lưu ý rằng, nếu bạn dùng'user_role' => ['required', new EnumValue(UserRole::class, false)],Rule Objectthì bạn cần phảiUseEnumValue RulevàEnumKey Ruletương ứng ở đầu mỗi file:use BenSampo\Enum\Rules\EnumValue; use BenSampo\Enum\Rules\EnumKey;
Đa ngôn ngữ (Localization)
Ngoài việc sử dụng đa ngôn ngữ với phương thức trans() trong phương thức getDescription() ở trên, bạn còn có thể sử dụng tính năng đa ngôn ngữ của Laravel Enum.
Đầu tiên bạn cần tạo file enums.php trong các thư mục ngôn ngữ resources/lang/{language_folder} như sau:
- resources/lang/en/enums.php:- <?php use App\Enums\UserRole; return [ UserRole::class => [ UserRole::Administrator => 'Administrator', UserRole::Moderator => 'Moderator', UserRole::Member => 'Member', ], ];
- resources/lang/vn/enums.php- <?php use App\Enums\UserRole; return [ UserRole::class => [ UserRole::Administrator => 'Quản trị viên', UserRole::Moderator => 'Điều hành viên', UserRole::Member => 'Thành viên', ], ];
Sau đó bạn chỉ cầnimplements interface LocalizedEnum trong Enum UserRole là xong:
<?php
namespace App\Enums;
use BenSampo\Enum\Enum;
use BenSampo\Enum\Contracts\LocalizedEnum;
final class UserRole extends Enum implements LocalizedEnum
{
    // ....
}
Các mô tả (description) của Enum sẽ tự động thay đổi giá trị tùy theo ngôn ngữ của hệ thống đang dùng.
Kết luận
Qua bài viết trên mình đã giới thiệu cho các bạn Enum cũng như cách sử dụng Enum trong Laravel với package Laravel Enum. Hy vọng bài viết này sẽ có ích đối với các bạn 
Tham khảo
All rights reserved
 
  
 