// +---------------------------------------------------------------------- namespace think; use ArrayAccess; use ArrayIterator; use Countable; use IteratorAggregate; use JsonSerializable; use Traversable; abstract class Paginator implements ArrayAccess, Countable, IteratorAggregate, JsonSerializable { /** @var bool 鏄惁涓虹畝娲佹ā寮� */ protected $simple = false; /** @var Collection 鏁版嵁闆� */ protected $items; /** @var integer 褰撳墠椤� */ protected $currentPage; /** @var integer 鏈€鍚庝竴椤� */ protected $lastPage; /** @var integer|null 鏁版嵁鎬绘暟 */ protected $total; /** @var integer 姣忛〉鐨勬暟閲� */ protected $listRows; /** @var bool 鏄惁鏈変笅涓€椤� */ protected $hasMore; /** @var array 涓€浜涢厤缃� */ protected $options = [ 'var_page' => 'page', 'path' => '/', 'query' => [], 'fragment' => '', ]; public function __construct($items, $listRows, $currentPage = null, $total = null, $simple = false, $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 $items * @param $listRows * @param null $currentPage * @param bool $simple * @param null $total * @param array $options * @return Paginator */ public static function make($items, $listRows, $currentPage = null, $total = null, $simple = false, $options = []) { return new static($items, $listRows, $currentPage, $total, $simple, $options); } protected function setCurrentPage($currentPage) { if (!$this->simple && $currentPage > $this->lastPage) { return $this->lastPage > 0 ? $this->lastPage : 1; } return $currentPage; } /** * 鑾峰彇椤电爜瀵瑰簲鐨勯摼鎺� * * @param $page * @return string */ protected function url($page) { if ($page <= 0) { $page = 1; } if (strpos($this->options['path'], '[PAGE]') === false) { $parameters = [$this->options['var_page'] => $page]; $path = $this->options['path']; } else { $parameters = []; $path = str_replace('[PAGE]', $page, $this->options['path']); } if (count($this->options['query']) > 0) { $parameters = array_merge($this->options['query'], $parameters); } $url = $path; if (!empty($parameters)) { $url .= '?' . urldecode(http_build_query($parameters, null, '&')); } return $url . $this->buildFragment(); } /** * 鑷姩鑾峰彇褰撳墠椤电爜 * @param string $varPage * @param int $default * @return int */ public static function getCurrentPage($varPage = 'page', $default = 1) { $page = Request::instance()->request($varPage); if (filter_var($page, FILTER_VALIDATE_INT) !== false && (int) $page >= 1) { return $page; } return $default; } /** * 鑷姩鑾峰彇褰撳墠鐨刾ath * @return string */ public static function getCurrentPath() { return Request::instance()->baseUrl(); } public function total() { if ($this->simple) { throw new \DomainException('not support total'); } return $this->total; } public function listRows() { return $this->listRows; } public function currentPage() { return $this->currentPage; } public function lastPage() { if ($this->simple) { throw new \DomainException('not support last'); } return $this->lastPage; } /** * 鏁版嵁鏄惁瓒冲鍒嗛〉 * @return boolean */ public function hasPages() { return !(1 == $this->currentPage && !$this->hasMore); } /** * 鍒涘缓涓€缁勫垎椤甸摼鎺� * * @param int $start * @param int $end * @return array */ public function getUrlRange($start, $end) { $urls = []; for ($page = $start; $page <= $end; $page++) { $urls[$page] = $this->url($page); } return $urls; } /** * 璁剧疆URL閿氱偣 * * @param string|null $fragment * @return $this */ public function fragment($fragment) { $this->options['fragment'] = $fragment; return $this; } /** * 娣诲姞URL鍙傛暟 * * @param array|string $key * @param string|null $value * @return $this */ public function appends($key, $value = null) { if (!is_array($key)) { $queries = [$key => $value]; } else { $queries = $key; } foreach ($queries as $k => $v) { if ($k !== $this->options['var_page']) { $this->options['query'][$k] = $v; } } return $this; } /** * 鏋勯€犻敋鐐瑰瓧绗︿覆 * * @return string */ protected function buildFragment() { return $this->options['fragment'] ? '#' . $this->options['fragment'] : ''; } /** * 娓叉煋鍒嗛〉html * @return mixed */ abstract public function render(); public function items() { return $this->items->all(); } public function getCollection() { return $this->items; } public function isEmpty() { return $this->items->isEmpty(); } /** * Retrieve an external iterator * @return Traversable An instance of an object implementing Iterator or * Traversable */ public function getIterator() { return new ArrayIterator($this->items->all()); } /** * Whether a offset exists * @param mixed $offset * @return bool */ public function offsetExists($offset) { return $this->items->offsetExists($offset); } /** * Offset to retrieve * @param mixed $offset * @return mixed */ public function offsetGet($offset) { return $this->items->offsetGet($offset); } /** * Offset to set * @param mixed $offset * @param mixed $value */ public function offsetSet($offset, $value) { $this->items->offsetSet($offset, $value); } /** * Offset to unset * @param mixed $offset * @return void * @since 5.0.0 */ public function offsetUnset($offset) { $this->items->offsetUnset($offset); } /** * Count elements of an object */ public function count() { return $this->items->count(); } public function __toString() { return (string) $this->render(); } public function toArray() { try { $total = $this->total(); } catch (\DomainException $e) { $total = null; } return [ 'total' => $total, 'per_page' => $this->listRows(), 'current_page' => $this->currentPage(), 'data' => $this->items->toArray(), ]; } /** * Specify data which should be serialized to JSON */ public function jsonSerialize() { return $this->toArray(); } public function __call($name, $arguments) { return call_user_func_array([$this->getCollection(), $name], $arguments); } }