Easy access to entity states values and to their localizations.
When you have an entity with some states you usually need to access to those states' values and to their localizations. Let's consider the following cases:
Let's assume we have a User entity with a role state. And the role state can only accept 'member', 'moderator' and 'administrator' values.
-
When you need to validate a state's acceptable values. So when we try to create a new user we need to validate the role state value. But how to do it if we don't have a main source for that values? And usually we may do like below:
/** * Get the validation rules that apply to the request. * * @return array */ public function rules() { return [ // ... other validation rules 'role' => [ 'required', Rule::in(['member', 'moderator', 'administrator']), // 🙁 values are hardcoded ], ]; }
-
When you need to display localization for one of the state's values. For example when you need to give an information about a user's role:
<li>{{ $user->name }}</li> <li>{{ $user->email }}</li> <li>{{ trans("user.states.role.{$user->role}") }}</li>
🙁 Localization file path, file name and locale key are hardcoded.
-
When you need to display those values with locales. For example in a select box:
@foreach (['member', 'moderator', 'administrator'] as $value) <option value="{{ $value }}">@lang("user.states.role.{$value}")</option> @endforeach
🙁 Hardcoded, hardcoded, ...
And yes when you hardcoded those values in different places of your project then it will be harder to change (add, remove, rename) those values. Because you don't use a main source for those values.
Unfortunately Laravel doesn't provide a solution for this problem out of the box. So I've created this package. With this package you will not have problems in the above section! Let's solve those problems in the appropriate order.
Firstly create a States
folder in the app
directory, then create a UserState
class in it with extending a StateAbstract
abstract class. And define a ROLE
constant in the class with those values from the above section. You should have something below:
<?php // app/States/UserState.php
namespace App\States\UserState;
use Zvermafia\Larastate\Abstracts\StateAbstract;
class UserState extends StateAbstract
{
/** @var array */
public const ROLE = [
'member' => 'member',
'moderator' => 'moderator',
'administrator' => 'administrator',
];
}
-
When you need to validate a state's acceptable values:
/** * Get the validation rules that apply to the request. * * @param \App\States\UserState $user_states * @return array */ public function rules(UserState $user_states) { return [ // ... other validation rules 'role' => [ 'required', Rule::in($user_states->getRoleValues()), // 🙂 values are not hardcoded! ], ]; }
-
When you need to display localization for one of the state's values. For example when you need to give an information about a user's role:
<li>{{ $user->name }}</li> <li>{{ $user->email }}</li> <li>{{ $user_states->getRoleLocale($user->role) }}</li>
🙂 Localization file path, file name and locale key are taken from the single source!
-
When you need to display those values with locales. For example in a select box:
@foreach ($user_states->getRoleValuesWithLocales() as $value => $locale) <option value="{{ $value }}">{{ $locale }}</option> @endforeach
🙂 Much better!
To install the package just pull it via composer and you're ready to setup.
$ composer require zvermafia/larastate
The package have some convention about structuring a state classes directory, naming those classes and naming constants. But all this can be changed to suit your needs by extending a StateAbstract
abstract class. So if you need it then take a look at the source code.
- By default you should put all your state classes to
app/States
directory. And yes usually there is not that directory so you need to create it by yourself. - And your state class names should end with
State
postfix, for exampleUserState
,PageState
,OrderState
and so on... - Constant names should be in upper case and each word separated by underscore.
Let's assume we want to setup a state class for a post entity, so our steps are follows:
- Create a
States
folder in theapp
directory; - Create a
PostState.php
file and put it into theapp/States
directory; - Extend a
StateAbstract
abstract class and define constants, you should have something below:<?php namespace App\States; use Zvermafia\Larastate\Abstracts\StateAbstract; class PostState extends StateAbstract { /** @var array */ public const STATUS = [ 'draft' => 0, 'published' => 1, ]; /** @var array */ public const TYPE = [ 'info' => 0, 'blog' => 1, 'news' => 2, ]; }
- Create localization file in a
resources/lang/en/entities
directory with the entity name, so in our case it will bepost.php
. It's a regular Laravel's localization file but by default all state localizations should be grouped in astate
key of the array. So you should have something like below:<?php // resources/lang/en/entities/post.php return [ // here's other localization like attribute/property... 'state' => [ 'status' => [ App\States\PostState::STATUS['draft'] => 'Draft', App\States\PostState::STATUS['published'] => 'Published', ], 'type' => [ App\States\PostState::TYPE['info'] => 'Static page', App\States\PostState::TYPE['blog'] => 'Blog post', App\States\PostState::TYPE['news'] => 'News post', ], ], ];
For usage example we will use our PostState
which we created it in the above section.
There are three cases and all works through PHP's magic __call()
method:
- Getting values for the state
- Getting values with locales for the state
- Getting a locale for one of the state's values
All methods start with prefix get
and ends with one of the case names (Values
, ValuesWithLocales
or Locale
).
And between those prefix and postfixes you should write your state name in StudlyCase
format.
For a type
state those will be:
getTypeValues()
getTypeValuesWithLocales()
getTypeLocale(1)
$post_states = new App\States\PostState();
$post_states->getTypeValues(); // [0, 1, 2]
$post_states->getTypeValuesWithLocales(); // [0 => 'Static page', 1 => 'Blog post', 2 => 'News post']
$post_states->getTypeValues(); // 'Blog post'
- BenSampo/laravel-enum
- artkonekt/enum-eloquent
- nasyrov/laravel-enums
- mad-web/laravel-enum
- spatie/laravel-enum
- cerbero90/laravel-enum
So why I've created this package when there is already such kind of packages? Because they are a bit complicated for my simple task.
Please see CHANGELOG for more information on what has changed recently.
$ composer test
Please see CONTRIBUTING and CODE_OF_CONDUCT for details.
If you discover any security related issues, please email mohirjon@gmail.com instead of using the issue tracker.
The MIT License (MIT). Please see License File for more information.