12-05-2026 Frontend dev
This commit is contained in:
parent
405df0a122
commit
5b8bdf4182
779 changed files with 480564 additions and 6241 deletions
109
app/Models/Concerns/HasUniqueSlug.php
Normal file
109
app/Models/Concerns/HasUniqueSlug.php
Normal file
|
|
@ -0,0 +1,109 @@
|
|||
<?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
|
||||
}
|
||||
}
|
||||
Loading…
Add table
Add a link
Reference in a new issue