To complete the whole picture, the last example in this sequence comes from a well-known package Spatie Media Library.
Here's the primary information from their installation instructions:
To associate media with a model, the Model must implement the following interface and Trait:
namespace App\Models; use Illuminate\Database\Eloquent\Model;use Spatie\MediaLibrary\HasMedia;use Spatie\MediaLibrary\InteractsWithMedia; class YourModel extends Model implements HasMedia{ use InteractsWithMedia;}
So, here we have all three things we discussed earlier:
And I often see this structure in Laravel packages:
laravel-medialibrary/src/HasMedia.php:
interface HasMedia{ public function media(): MorphMany; public function addMedia(string|UploadedFile $file): FileAdder; // ... more methods public function registerMediaConversions(Media $media = null): void;
In addition to that interface, the package provides the Trait to implement all those methods.
laravel-medialibrary/src/InteractsWithMedia.php:
trait InteractsWithMedia{ /** @var Conversion[] */ public array $mediaConversions = []; // ... public function media(): MorphMany { return $this->morphMany(config('media-library.media_model'), 'model'); } // ...
In other words, the package gives you the rules and its own variant of how they follow those rules.
But in your Eloquent Model, you're free to override the implementation.
The difference from the Filament canAccessPanel()
method is that Filament didn't "know" how to implement it. It's all on you as a user of the package. In Spatie's case, they "know" how that media()
should be a polymorphic relation to their Media model but also make it configurable for you.
I've found a few more packages that work with the same logic.
Laratrust package has this in the installation instructions:
use Laratrust\Contracts\LaratrustUser;use Laratrust\Traits\HasRolesAndPermissions;use Illuminate\Foundation\Auth\User as Authenticatable; class User extends Authenticatable implements LaratrustUser{ use HasRolesAndPermissions; // ...}
As you can see, there's an interface LaratrustUser
and a trait HasRolesAndPermissions
.
Another example comes from the same Spatie team but in a different package Eloquent Sortable. Here's what you need to define in your Eloquent Model:
use Spatie\EloquentSortable\Sortable;use Spatie\EloquentSortable\SortableTrait; class MyModel extends Model implements Sortable{ use SortableTrait; public $sortable = [ 'order_column_name' => 'order_column', 'sort_when_creating' => true, ]; // ...}
Again: implements Sortable
(interface) and use SortableTrait
(trait).
Such a complex OOP approach with interfaces is more widely used in packages and frameworks because they have to create some strict structure for users to follow. But it's beneficial to understand because you may never know when you work on a more complex project that would benefit from this, too.