paginator.php 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. <?php namespace Laravel;
  2. class Paginator_Factory {
  3. protected $request;
  4. protected $html;
  5. protected $lang;
  6. public function __construct(Request $request, HTML $html, Lang_Factory $lang)
  7. {
  8. $this->html = $html;
  9. $this->lang = $lang;
  10. $this->request = $request;
  11. }
  12. public function make($results, $total, $per_page)
  13. {
  14. $page = Paginator::page($total, $per_page);
  15. $last_page = ceil($total / $per_page);
  16. return new Paginator($this->request, $this->html, $this->lang, $results, $page, $total, $per_page, $last_page);
  17. }
  18. }
  19. class Paginator {
  20. /**
  21. * The results for the current page.
  22. *
  23. * @var array
  24. */
  25. public $results;
  26. /**
  27. * The total number of results.
  28. *
  29. * @var int
  30. */
  31. public $total;
  32. /**
  33. * The current page.
  34. *
  35. * @var int
  36. */
  37. public $page;
  38. /**
  39. * The number of items per page.
  40. *
  41. * @var int
  42. */
  43. public $per_page;
  44. /**
  45. * The last page available for the result set.
  46. *
  47. * @var int
  48. */
  49. public $last_page;
  50. /**
  51. * The language that should be used when generating page links.
  52. *
  53. * @var string
  54. */
  55. public $language;
  56. /**
  57. * The values that should be appended to the end of the link query strings.
  58. *
  59. * @var array
  60. */
  61. public $append = array();
  62. /**
  63. * Create a new Paginator instance.
  64. *
  65. * @param array $results
  66. * @param int $page
  67. * @param int $total
  68. * @param int $per_page
  69. * @param int $last_page
  70. * @return void
  71. */
  72. protected function __construct(Request $request, HTML $html, Lang_Factory $lang, $results, $page, $total, $per_page, $last_page)
  73. {
  74. $this->html = $html;
  75. $this->lang = $lang;
  76. $this->page = $page;
  77. $this->total = $total;
  78. $this->request = $request;
  79. $this->results = $results;
  80. $this->per_page = $per_page;
  81. $this->last_page = $last_page;
  82. }
  83. /**
  84. * Create a new Paginator instance.
  85. *
  86. * @param array $results
  87. * @param int $total
  88. * @param int $per_page
  89. * @return Paginator
  90. */
  91. public static function make($results, $total, $per_page)
  92. {
  93. return new static($results, static::page($total, $per_page), $total, $per_page, ceil($total / $per_page));
  94. }
  95. /**
  96. * Get the current page from the request query string.
  97. *
  98. * The page will be validated and adjusted if it is less than one or greater than the last page.
  99. * For example, if the current page is not an integer or less than one, one will be returned.
  100. * If the current page is greater than the last page, the last page will be returned.
  101. *
  102. * @param int $total
  103. * @param int $per_page
  104. * @return int
  105. */
  106. public static function page($total, $per_page)
  107. {
  108. $page = IoC::container()->resolve('laravel.input')->get('page', 1);
  109. if (is_numeric($page) and $page > $last_page = ceil($total / $per_page))
  110. {
  111. return ($last_page > 0) ? $last_page : 1;
  112. }
  113. return ($page < 1 or filter_var($page, FILTER_VALIDATE_INT) === false) ? 1 : $page;
  114. }
  115. /**
  116. * Create the HTML pagination links.
  117. *
  118. * @param int $adjacent
  119. * @return string
  120. */
  121. public function links($adjacent = 3)
  122. {
  123. if ($this->last_page <= 1) return '';
  124. // The hard-coded "7" is to account for all of the constant elements in a sliding range.
  125. // Namely: The the current page, the two ellipses, the two beginning pages, and the two ending pages.
  126. if ($this->last_page < 7 + ($adjacent * 2))
  127. {
  128. $numbers = $this->range(1, $this->last_page);
  129. }
  130. else
  131. {
  132. $numbers = $this->slider($adjacent);
  133. }
  134. return '<div class="pagination">'.$this->previous().$numbers.$this->next().'</div>';
  135. }
  136. /**
  137. * Build sliding list of HTML numeric page links.
  138. *
  139. * @param int $adjacent
  140. * @return string
  141. */
  142. private function slider($adjacent)
  143. {
  144. if ($this->page <= $adjacent * 2)
  145. {
  146. return $this->range(1, 2 + ($adjacent * 2)).$this->ending();
  147. }
  148. elseif ($this->page >= $this->last_page - ($adjacent * 2))
  149. {
  150. return $this->beginning().$this->range($this->last_page - 2 - ($adjacent * 2), $this->last_page);
  151. }
  152. else
  153. {
  154. return $this->beginning().$this->range($this->page - $adjacent, $this->page + $adjacent).$this->ending();
  155. }
  156. }
  157. /**
  158. * Generate the "previous" HTML link.
  159. *
  160. * @return string
  161. */
  162. public function previous()
  163. {
  164. $text = Lang::line('pagination.previous')->get($this->language);
  165. if ($this->page > 1)
  166. {
  167. return $this->link($this->page - 1, $text, 'prev_page').' ';
  168. }
  169. return HTML::span($text, array('class' => 'disabled prev_page')).' ';
  170. }
  171. /**
  172. * Generate the "next" HTML link.
  173. *
  174. * @return string
  175. */
  176. public function next()
  177. {
  178. $text = Lang::line('pagination.next')->get($this->language);
  179. if ($this->page < $this->last_page)
  180. {
  181. return $this->link($this->page + 1, $text, 'next_page');
  182. }
  183. return HTML::span($text, array('class' => 'disabled next_page'));
  184. }
  185. /**
  186. * Build the first two page links for a sliding page range.
  187. *
  188. * @return string
  189. */
  190. private function beginning()
  191. {
  192. return $this->range(1, 2).'<span class="dots">...</span>';
  193. }
  194. /**
  195. * Build the last two page links for a sliding page range.
  196. *
  197. * @return string
  198. */
  199. private function ending()
  200. {
  201. return '<span class="dots">...</span>'.$this->range($this->last_page - 1, $this->last_page);
  202. }
  203. /**
  204. * Build a range of page links.
  205. *
  206. * For the current page, an HTML span element will be generated instead of a link.
  207. *
  208. * @param int $start
  209. * @param int $end
  210. * @return string
  211. */
  212. private function range($start, $end)
  213. {
  214. $pages = '';
  215. for ($i = $start; $i <= $end; $i++)
  216. {
  217. $pages .= ($this->page == $i) ? HTML::span($i, array('class' => 'current')).' ' : $this->link($i, $i, null).' ';
  218. }
  219. return $pages;
  220. }
  221. /**
  222. * Create a HTML page link.
  223. *
  224. * @param int $page
  225. * @param string $text
  226. * @param string $attributes
  227. * @return string
  228. */
  229. private function link($page, $text, $class)
  230. {
  231. $append = '';
  232. foreach ($this->append as $key => $value)
  233. {
  234. $append .= '&'.$key.'='.$value;
  235. }
  236. return HTML::link(Request::uri().'?page='.$page.$append, $text, compact('class'), Request::is_secure());
  237. }
  238. /**
  239. * Set the language that should be used when generating page links.
  240. *
  241. * @param string $language
  242. * @return Paginator
  243. */
  244. public function lang($language)
  245. {
  246. $this->language = $language;
  247. return $this;
  248. }
  249. /**
  250. * Set the items that should be appended to the link query strings.
  251. *
  252. * @param array $values
  253. * @return Paginator
  254. */
  255. public function append($values)
  256. {
  257. $this->append = $values;
  258. return $this;
  259. }
  260. }