epessine / vessel
Global state for Laravel Livewire components
Requires
- php: ^8.2|^8.3
- livewire/livewire: ^3.0
Requires (Dev)
- laravel/pint: ^1.15
- orchestra/testbench: ^9.0
- pestphp/pest: ^2.34
- pestphp/pest-plugin-laravel: ^2.4
- pestphp/pest-plugin-livewire: ^2.1
- phpstan/phpstan: ^1.10.34
- rector/rector: ^1.1
README
Vessel - global state for Livewire
Vessel provides a global state for Livewire, allowing multiple components to share state without a multitude of events to handle interaction.
Just create a Vessel class, set it's properties and initialization and use it on your components.
Installation
composer install epessine/vessel
No configuration needed.
Usage
Vessel works by creating 'vessel classes', where you define the properties that you global state will have:
use Vessel\BaseVessel; class FilterVessel extends BaseVessel { public string $selectedUserType = 'admin'; }
You can also initialize your properties using the init()
method:
use Vessel\BaseVessel; class FilterVessel extends BaseVessel { public string $selectedUserType; public function init(): void { $this->selectedUserType = str('admin')->toString(); } }
Then use the Vessel on your components with the Vessel\Attributes\Vessel
attribute. Take a look at the components below:
use Livewire\Component; use Vessel\Attributes\Vessel; /** * @property-read FilterVessel globalFilters */ class Filter extends Component { #[Vessel] public function globalFilters(): string { return FilterVessel::class; // vessel class declared above } public function selectUserFilter(string $type): void { // here we change the vessel property value $this->globalFilters->selectedUserType = $type; } // ... }
use Livewire\Component; use Vessel\Attributes\Vessel; /** * @property-read FilterVessel globalFilters */ class UserList extends Component { #[Vessel] public function globalFilters(): string { return FilterVessel::class; // vessel class declared above } #[Computed] public function users(): array { // here we get the updated vessel property value User::query() ->where('type', $this->globalFilters->selectedUserType) ->get() ->all(); } // ... }
After calling Filter::selectUserFilter()
, the selectedUserType
will be updated on all components that use the same Vessel, and will reflect the change on the UserList::users
computed property query, for example.
All of that without a single event dispatch/listener!
Updated Hook
You can use the updated
lifecycle hook on Vessel properties the same way as Livewire properties. The updated method name should include the Vessel method name as well. Let's rewrite the example above:
use Livewire\Component; use Vessel\Attributes\Vessel; /** * @property-read FilterVessel globalFilters */ class UserList extends Component { public array $users; #[Vessel] public function globalFilters(): string { return FilterVessel::class; // vessel class declared above } public function updatedGlobalFiltersSelectedUserType(string $value): void { // here we get the updated vessel property value $this->users = User::query()->where('type', $value)->get()->all(); } // ... }
This way, whenever the $selectedUserType
property on the FilterVessel
class is updated on any component that uses it, the updatedGlobalFiltersSelectedUserType
method will be called.
Troubleshooting
There are some base rules when using Vessel:
- Vessel properties cannot be accessed/mutated on the front-end via
$wire
. You can create methods that can be called on the front-end to interact with the Vessel indirectly, though. - Vessel properties are 'immutable' so any changes inside the property will not reflect on other components. Always reassign the property after making changes.
- Vessel uses the application cache to operate and maintain state, so storing huge amounts of data on a Vessel can slow down your whole application.