123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227 |
- <?php
- namespace App\Traits;
- use ArrayAccess;
- use Illuminate\Database\Eloquent\Builder;
- use Illuminate\Database\Eloquent\SoftDeletingScope;
- use Illuminate\Support\Facades\Event;
- use InvalidArgumentException;
- use Spatie\EloquentSortable\EloquentModelSortedEvent;
- use Spatie\EloquentSortable\Sortable;
- /*
- * 列表排序is_pinned desc,order desc时用这个trait
- */
- trait SortableTraitPinned
- {
- public static function bootSortableTraitPinned()
- {
- static::creating(function ($model) {
- if ($model instanceof Sortable && $model->shouldSortWhenCreating()) {
- $model->setHighestOrderNumber();
- }
- });
- }
- public function setHighestOrderNumber(): void
- {
- $orderColumnName = $this->determineOrderColumnName();
- $this->$orderColumnName = $this->getHighestOrderNumber() + 1;
- }
- public function getHighestOrderNumber(): int
- {
- return (int)$this->buildSortQuery()->max($this->determineOrderColumnName());
- }
- public function getLowestOrderNumber(): int
- {
- return (int)$this->buildSortQuery()->min($this->determineOrderColumnName());
- }
- public function scopeOrdered(Builder $query, string $direction = 'asc')
- {
- return $query->orderBy($this->determineOrderColumnName(), $direction);
- }
- public static function setNewOrder(
- $ids,
- int $startOrder = 1,
- string $primaryKeyColumn = null,
- callable $modifyQuery = null
- ): void {
- if (! is_array($ids) && ! $ids instanceof ArrayAccess) {
- throw new InvalidArgumentException('You must pass an array or ArrayAccess object to setNewOrder');
- }
- $model = new static();
- $orderColumnName = $model->determineOrderColumnName();
- if (is_null($primaryKeyColumn)) {
- $primaryKeyColumn = $model->getQualifiedKeyName();
- }
- if (config('eloquent-sortable.ignore_timestamps', false)) {
- static::$ignoreTimestampsOn = array_values(array_merge(static::$ignoreTimestampsOn, [static::class]));
- }
- foreach ($ids as $id) {
- static::withoutGlobalScope(SoftDeletingScope::class)
- ->when(is_callable($modifyQuery), function ($query) use ($modifyQuery) {
- return $modifyQuery($query);
- })
- ->where($primaryKeyColumn, $id)
- ->update([$orderColumnName => $startOrder++]);
- }
- Event::dispatch(new EloquentModelSortedEvent(static::class));
- if (config('eloquent-sortable.ignore_timestamps', false)) {
- static::$ignoreTimestampsOn = array_values(array_diff(static::$ignoreTimestampsOn, [static::class]));
- }
- }
- public static function setNewOrderByCustomColumn(string $primaryKeyColumn, $ids, int $startOrder = 1)
- {
- self::setNewOrder($ids, $startOrder, $primaryKeyColumn);
- }
- public function determineOrderColumnName(): string
- {
- return $this->sortable['order_column_name'] ?? config('eloquent-sortable.order_column_name', 'order_column');
- }
- /**
- * Determine if the order column should be set when saving a new model instance.
- */
- public function shouldSortWhenCreating(): bool
- {
- return $this->sortable['sort_when_creating'] ?? config('eloquent-sortable.sort_when_creating', true);
- }
- /*
- * 由于排序在order desc时会出错,所以这里重写了 SortableTrait 中 moveOrderDown 和 moveOrderUp方法
- */
- public function moveOrderDown(): static
- {
- $orderColumnName = $this->determineOrderColumnName();
- $swapWithModel = $this->buildSortQuery()->limit(1)
- ->orderBy('is_pinned', 'desc')->orderBy('order', 'desc')
- ->where($orderColumnName, '<', $this->$orderColumnName)
- ->first();
- if (! $swapWithModel) {
- return $this;
- }
- if (($this->is_pinned == 1 && $swapWithModel->is_pinned == 0)
- || ($this->is_pinned == 0 && $swapWithModel->is_pinned == 1)) {
- throw new \Exception('No sorting allowed');
- }
- return $this->swapOrderWithModel($swapWithModel);
- }
- public function moveOrderUp(): static
- {
- $orderColumnName = $this->determineOrderColumnName();
- $swapWithModel = $this->buildSortQuery()->limit(1)
- ->orderBy('is_pinned', 'asc')->orderBy('order', 'asc')
- ->where($orderColumnName, '>', $this->$orderColumnName)
- ->first();
- if (! $swapWithModel) {
- return $this;
- }
- if (($this->is_pinned == 1 && $swapWithModel->is_pinned == 0)
- || ($this->is_pinned == 0 && $swapWithModel->is_pinned == 1)) {
- throw new \Exception('No sorting allowed');
- }
- return $this->swapOrderWithModel($swapWithModel);
- }
- public function swapOrderWithModel(Sortable $otherModel): static
- {
- $orderColumnName = $this->determineOrderColumnName();
- $oldOrderOfOtherModel = $otherModel->$orderColumnName;
- $otherModel->$orderColumnName = $this->$orderColumnName;
- $otherModel->save();
- $this->$orderColumnName = $oldOrderOfOtherModel;
- $this->save();
- return $this;
- }
- public static function swapOrder(Sortable $model, Sortable $otherModel): void
- {
- $model->swapOrderWithModel($otherModel);
- }
- public function moveToStart(): static
- {
- $firstModel = $this->buildSortQuery()->limit(1)
- ->ordered()
- ->first();
- if ($firstModel->getKey() === $this->getKey()) {
- return $this;
- }
- $orderColumnName = $this->determineOrderColumnName();
- $this->$orderColumnName = $firstModel->$orderColumnName;
- $this->save();
- $this->buildSortQuery()->where($this->getQualifiedKeyName(), '!=', $this->getKey())->increment(
- $orderColumnName
- );
- return $this;
- }
- public function moveToEnd(): static
- {
- $maxOrder = $this->getHighestOrderNumber();
- $orderColumnName = $this->determineOrderColumnName();
- if ($this->$orderColumnName === $maxOrder) {
- return $this;
- }
- $oldOrder = $this->$orderColumnName;
- $this->$orderColumnName = $maxOrder;
- $this->save();
- $this->buildSortQuery()->where($this->getQualifiedKeyName(), '!=', $this->getKey())
- ->where($orderColumnName, '>', $oldOrder)
- ->decrement($orderColumnName);
- return $this;
- }
- public function isLastInOrder(): bool
- {
- $orderColumnName = $this->determineOrderColumnName();
- return (int)$this->$orderColumnName === $this->getHighestOrderNumber();
- }
- public function isFirstInOrder(): bool
- {
- $orderColumnName = $this->determineOrderColumnName();
- return (int)$this->$orderColumnName === $this->getLowestOrderNumber();
- }
- public function buildSortQuery(): Builder
- {
- return static::query();
- }
- }
|