// +---------------------------------------------------------------------- declare(strict_types=1); namespace think; use ArrayAccess; use ArrayIterator; use Closure; use Countable; use DomainException; use IteratorAggregate; use JsonSerializable; use think\paginator\driver\Bootstrap; use Traversable; /** * 分页基础类. * * @mixin Collection */ abstract class Paginator implements ArrayAccess, Countable, IteratorAggregate, JsonSerializable { /** * 是否简洁模式. * * @var bool */ protected $simple = false; /** * 数据集. * * @var Collection */ protected $items; /** * 当前页. * * @var int */ protected $currentPage; /** * 最后一页. * * @var int */ protected $lastPage; /** * 数据总数. * * @var int|null */ protected $total; /** * 每页数量. * * @var int */ protected $listRows; /** * 是否有下一页. * * @var bool */ protected $hasMore; /** * 分页配置. * * @var array */ protected $options = [ 'var_page' => 'page', 'path' => '/', 'query' => [], 'fragment' => '', ]; /** * 获取当前页码 * * @var Closure */ protected static $currentPageResolver; /** * 获取当前路径. * * @var Closure */ protected static $currentPathResolver; /** * @var Closure */ protected static $maker; public function __construct($items, int $listRows, int $currentPage = 1, int $total = null, bool $simple = false, array $options = []) { $this->options = array_merge($this->options, $options); $this->options['path'] = '/' != $this->options['path'] ? rtrim($this->options['path'], '/') : $this->options['path']; $this->simple = $simple; $this->listRows = $listRows; if (!$items instanceof Collection) { $items = Collection::make($items); } if ($simple) { $this->currentPage = $this->setCurrentPage($currentPage); $this->hasMore = count($items) > ($this->listRows); $items = $items->slice(0, $this->listRows); } else { $this->total = $total; $this->lastPage = (int) ceil($total / $listRows); $this->currentPage = $this->setCurrentPage($currentPage); $this->hasMore = $this->currentPage < $this->lastPage; } $this->items = $items; } /** * @param mixed $items * @param int $listRows * @param int $currentPage * @param int $total * @param bool $simple * @param array $options * * @return Paginator */ public static function make($items, int $listRows, int $currentPage = 1, int $total = null, bool $simple = false, array $options = []) { if (isset(static::$maker)) { return call_user_func(static::$maker, $items, $listRows, $currentPage, $total, $simple, $options); } return new Bootstrap($items, $listRows, $currentPage, $total, $simple, $options); } public static function maker(Closure $resolver) { static::$maker = $resolver; } protected function setCurrentPage(int $currentPage): int { if (!$this->simple && $currentPage > $this->lastPage) { return $this->lastPage > 0 ? $this->lastPage : 1; } return $currentPage; } /** * 获取页码对应的链接. * * @param int $page * * @return string */ protected function url(int $page): string { if ($page <= 0) { $page = 1; } if (!str_contains($this->options['path'], '[PAGE]')) { $parameters = [$this->options['var_page'] => $page]; $path = $this->options['path']; } else { $parameters = []; $path = str_replace('[PAGE]', (string) $page, $this->options['path']); } if (count($this->options['query']) > 0) { $parameters = array_merge($this->options['query'], $parameters); } $url = $path; if (!empty($parameters)) { $url .= '?' . http_build_query($parameters, '', '&'); } return $url . $this->buildFragment(); } /** * 自动获取当前页码 * * @param string $varPage * @param int $default * * @return int */ public static function getCurrentPage(string $varPage = 'page', int $default = 1): int { if (isset(static::$currentPageResolver)) { return call_user_func(static::$currentPageResolver, $varPage); } return $default; } /** * 设置获取当前页码闭包. * * @param Closure $resolver */ public static function currentPageResolver(Closure $resolver) { static::$currentPageResolver = $resolver; } /** * 自动获取当前的path. * * @param string $default * * @return string */ public static function getCurrentPath($default = '/'): string { if (isset(static::$currentPathResolver)) { return call_user_func(static::$currentPathResolver); } return $default; } /** * 设置获取当前路径闭包. * * @param Closure $resolver */ public static function currentPathResolver(Closure $resolver) { static::$currentPathResolver = $resolver; } /** * 获取数据总条数. * * @return int */ public function total(): int { if ($this->simple) { throw new DomainException('not support total'); } return $this->total; } /** * 获取每页数量. * * @return int */ public function listRows(): int { return $this->listRows; } /** * 获取当前页页码 * * @return int */ public function currentPage(): int { return $this->currentPage; } /** * 获取最后一页页码 * * @return int */ public function lastPage(): int { if ($this->simple) { throw new DomainException('not support last'); } return $this->lastPage; } /** * 数据是否足够分页. * * @return bool */ public function hasPages(): bool { return !(1 == $this->currentPage && !$this->hasMore); } /** * 创建一组分页链接. * * @param int $start * @param int $end * * @return array */ public function getUrlRange(int $start, int $end): array { $urls = []; for ($page = $start; $page <= $end; $page++) { $urls[$page] = $this->url($page); } return $urls; } /** * 设置URL锚点. * * @param string|null $fragment * * @return $this */ public function fragment(string $fragment = null) { $this->options['fragment'] = $fragment; return $this; } /** * 添加URL参数. * * @param array $append * * @return $this */ public function appends(array $append) { foreach ($append as $k => $v) { if ($k !== $this->options['var_page']) { $this->options['query'][$k] = $v; } } return $this; } /** * 构造锚点字符串. * * @return string */ protected function buildFragment(): string { return $this->options['fragment'] ? '#' . $this->options['fragment'] : ''; } /** * 渲染分页html. * * @return mixed */ abstract public function render(); public function items() { return $this->items->all(); } /** * 获取数据集. * * @return Collection|\think\model\Collection */ public function getCollection() { return $this->items; } /** * 设置数据集. * * @param Collection $items * * @return $this */ public function setCollection(Collection $items) { $this->items = $items; return $this; } public function isEmpty(): bool { return $this->items->isEmpty(); } /** * 给每个元素执行个回调. * * @param callable $callback * * @return $this */ public function each(callable $callback) { foreach ($this->items as $key => $item) { $result = $callback($item, $key); if (false === $result) { break; } elseif (!is_object($item)) { $this->items[$key] = $result; } } return $this; } /** * Retrieve an external iterator. * * @return Traversable An instance of an object implementing Iterator or * Traversable */ public function getIterator(): Traversable { return new ArrayIterator($this->items->all()); } /** * Whether a offset exists. * * @param mixed $offset * * @return bool */ public function offsetExists(mixed $offset): bool { return $this->items->offsetExists($offset); } /** * Offset to retrieve. * * @param mixed $offset * * @return mixed */ public function offsetGet(mixed $offset): mixed { return $this->items->offsetGet($offset); } /** * Offset to set. * * @param mixed $offset * @param mixed $value */ public function offsetSet(mixed $offset, mixed $value): void { $this->items->offsetSet($offset, $value); } /** * Offset to unset. * * @param mixed $offset * * @return void * * @since 5.0.0 */ public function offsetUnset(mixed $offset): void { $this->items->offsetUnset($offset); } /** * 统计数据集条数. * * @return int */ public function count(): int { return $this->items->count(); } public function __toString() { return (string) $this->render(); } /** * 转换为数组. * * @return array */ public function toArray(): array { try { $total = $this->total(); } catch (DomainException $e) { $total = null; } return [ 'total' => $total, 'per_page' => $this->listRows(), 'current_page' => $this->currentPage(), 'last_page' => $this->lastPage, 'data' => $this->items->toArray(), ]; } /** * Specify data which should be serialized to JSON. */ public function jsonSerialize(): array { return $this->toArray(); } public function __call($name, $arguments) { $result = call_user_func_array([$this->items, $name], $arguments); if ($result instanceof Collection) { $this->items = $result; return $this; } return $result; } }