presseportale/app/Models/Concerns/HasUniqueSlug.php
Kevin Adametz 5b8bdf4182
Some checks are pending
linter / quality (push) Waiting to run
tests / ci (push) Waiting to run
12-05-2026 Frontend dev
2026-05-12 18:32:33 +02:00

109 lines
2.8 KiB
PHP

<?php
namespace App\Models\Concerns;
use Illuminate\Database\Eloquent\Builder;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Support\Str;
/**
* Generates a unique slug for an Eloquent model based on a source attribute.
*
* Models opt-in by overriding two methods:
* - {@see slugScopeAttributes()} returns the attribute names whose current
* model values (or provided overrides) should restrict slug uniqueness.
* Defaults to an empty array (global uniqueness on the slug column).
* - {@see slugFallback()} returns the fallback when the source attribute is
* blank (defaults to "item").
*
* The model is expected to expose `slug_column` (defaults to `slug`) and
* the source attribute name to slugify (defaults to `title` or `name`).
*
* Typical usage:
*
* $pr->slug = $pr->generateUniqueSlug($pr->title, [
* 'portal' => $pr->portal,
* 'language' => $pr->language,
* ]);
*
* @mixin Model
*/
trait HasUniqueSlug
{
/**
* Build a unique slug from `$source`. Optionally pass overrides for the
* scoping attributes (matched against {@see slugScopeAttributes()}).
*
* @param array<string, mixed> $scope
*/
public function generateUniqueSlug(string $source, array $scope = []): string
{
$base = Str::slug($source) ?: $this->slugFallback();
$slug = $base;
$suffix = 2;
while ($this->slugExists($slug, $scope)) {
$slug = $base.'-'.$suffix++;
}
return $slug;
}
/**
* @param array<string, mixed> $scope
*/
protected function slugExists(string $slug, array $scope): bool
{
/** @var Model $self */
$self = $this;
$query = $self::query()
->withoutGlobalScopes()
->where($this->slugColumn(), $slug);
if ($self->exists) {
$query->where($self->getKeyName(), '!=', $self->getKey());
}
foreach ($this->slugScopeAttributes() as $attribute) {
$value = array_key_exists($attribute, $scope)
? $scope[$attribute]
: $self->getAttribute($attribute);
$value !== null
? $query->where($attribute, $value)
: $query->whereNull($attribute);
}
$this->applySlugConstraints($query);
return $query->exists();
}
/**
* @return list<string>
*/
protected function slugScopeAttributes(): array
{
return [];
}
protected function slugColumn(): string
{
return 'slug';
}
protected function slugFallback(): string
{
return 'item';
}
/**
* Hook for models to add additional WHERE constraints (e.g. soft-deleted
* records). Default: no additional constraints.
*/
protected function applySlugConstraints(Builder $query): void
{
// no-op
}
}