paginator.py 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224
  1. import collections.abc
  2. import inspect
  3. import warnings
  4. from math import ceil
  5. from django.utils.functional import cached_property
  6. from django.utils.inspect import method_has_no_args
  7. from django.utils.translation import gettext_lazy as _
  8. class UnorderedObjectListWarning(RuntimeWarning):
  9. pass
  10. class InvalidPage(Exception):
  11. pass
  12. class PageNotAnInteger(InvalidPage):
  13. pass
  14. class EmptyPage(InvalidPage):
  15. pass
  16. class Paginator:
  17. # Translators: String used to replace omitted page numbers in elided page
  18. # range generated by paginators, e.g. [1, 2, '…', 5, 6, 7, '…', 9, 10].
  19. ELLIPSIS = _('…')
  20. def __init__(self, object_list, per_page, orphans=0,
  21. allow_empty_first_page=True):
  22. self.object_list = object_list
  23. self._check_object_list_is_ordered()
  24. self.per_page = int(per_page)
  25. self.orphans = int(orphans)
  26. self.allow_empty_first_page = allow_empty_first_page
  27. def __iter__(self):
  28. for page_number in self.page_range:
  29. yield self.page(page_number)
  30. def validate_number(self, number):
  31. """Validate the given 1-based page number."""
  32. try:
  33. if isinstance(number, float) and not number.is_integer():
  34. raise ValueError
  35. number = int(number)
  36. except (TypeError, ValueError):
  37. raise PageNotAnInteger(_('That page number is not an integer'))
  38. if number < 1:
  39. raise EmptyPage(_('That page number is less than 1'))
  40. if number > self.num_pages:
  41. if number == 1 and self.allow_empty_first_page:
  42. pass
  43. else:
  44. raise EmptyPage(_('That page contains no results'))
  45. return number
  46. def get_page(self, number):
  47. """
  48. Return a valid page, even if the page argument isn't a number or isn't
  49. in range.
  50. """
  51. try:
  52. number = self.validate_number(number)
  53. except PageNotAnInteger:
  54. number = 1
  55. except EmptyPage:
  56. number = self.num_pages
  57. return self.page(number)
  58. def page(self, number):
  59. """Return a Page object for the given 1-based page number."""
  60. number = self.validate_number(number)
  61. bottom = (number - 1) * self.per_page
  62. top = bottom + self.per_page
  63. if top + self.orphans >= self.count:
  64. top = self.count
  65. return self._get_page(self.object_list[bottom:top], number, self)
  66. def _get_page(self, *args, **kwargs):
  67. """
  68. Return an instance of a single page.
  69. This hook can be used by subclasses to use an alternative to the
  70. standard :cls:`Page` object.
  71. """
  72. return Page(*args, **kwargs)
  73. @cached_property
  74. def count(self):
  75. """Return the total number of objects, across all pages."""
  76. c = getattr(self.object_list, 'count', None)
  77. if callable(c) and not inspect.isbuiltin(c) and method_has_no_args(c):
  78. return c()
  79. return len(self.object_list)
  80. @cached_property
  81. def num_pages(self):
  82. """Return the total number of pages."""
  83. if self.count == 0 and not self.allow_empty_first_page:
  84. return 0
  85. hits = max(1, self.count - self.orphans)
  86. return ceil(hits / self.per_page)
  87. @property
  88. def page_range(self):
  89. """
  90. Return a 1-based range of pages for iterating through within
  91. a template for loop.
  92. """
  93. return range(1, self.num_pages + 1)
  94. def _check_object_list_is_ordered(self):
  95. """
  96. Warn if self.object_list is unordered (typically a QuerySet).
  97. """
  98. ordered = getattr(self.object_list, 'ordered', None)
  99. if ordered is not None and not ordered:
  100. obj_list_repr = (
  101. '{} {}'.format(self.object_list.model, self.object_list.__class__.__name__)
  102. if hasattr(self.object_list, 'model')
  103. else '{!r}'.format(self.object_list)
  104. )
  105. warnings.warn(
  106. 'Pagination may yield inconsistent results with an unordered '
  107. 'object_list: {}.'.format(obj_list_repr),
  108. UnorderedObjectListWarning,
  109. stacklevel=3
  110. )
  111. def get_elided_page_range(self, number=1, *, on_each_side=3, on_ends=2):
  112. """
  113. Return a 1-based range of pages with some values elided.
  114. If the page range is larger than a given size, the whole range is not
  115. provided and a compact form is returned instead, e.g. for a paginator
  116. with 50 pages, if page 43 were the current page, the output, with the
  117. default arguments, would be:
  118. 1, 2, …, 40, 41, 42, 43, 44, 45, 46, …, 49, 50.
  119. """
  120. number = self.validate_number(number)
  121. if self.num_pages <= (on_each_side + on_ends) * 2:
  122. yield from self.page_range
  123. return
  124. if number > (1 + on_each_side + on_ends) + 1:
  125. yield from range(1, on_ends + 1)
  126. yield self.ELLIPSIS
  127. yield from range(number - on_each_side, number + 1)
  128. else:
  129. yield from range(1, number + 1)
  130. if number < (self.num_pages - on_each_side - on_ends) - 1:
  131. yield from range(number + 1, number + on_each_side + 1)
  132. yield self.ELLIPSIS
  133. yield from range(self.num_pages - on_ends + 1, self.num_pages + 1)
  134. else:
  135. yield from range(number + 1, self.num_pages + 1)
  136. class Page(collections.abc.Sequence):
  137. def __init__(self, object_list, number, paginator):
  138. self.object_list = object_list
  139. self.number = number
  140. self.paginator = paginator
  141. def __repr__(self):
  142. return '<Page %s of %s>' % (self.number, self.paginator.num_pages)
  143. def __len__(self):
  144. return len(self.object_list)
  145. def __getitem__(self, index):
  146. if not isinstance(index, (int, slice)):
  147. raise TypeError(
  148. 'Page indices must be integers or slices, not %s.'
  149. % type(index).__name__
  150. )
  151. # The object_list is converted to a list so that if it was a QuerySet
  152. # it won't be a database hit per __getitem__.
  153. if not isinstance(self.object_list, list):
  154. self.object_list = list(self.object_list)
  155. return self.object_list[index]
  156. def has_next(self):
  157. return self.number < self.paginator.num_pages
  158. def has_previous(self):
  159. return self.number > 1
  160. def has_other_pages(self):
  161. return self.has_previous() or self.has_next()
  162. def next_page_number(self):
  163. return self.paginator.validate_number(self.number + 1)
  164. def previous_page_number(self):
  165. return self.paginator.validate_number(self.number - 1)
  166. def start_index(self):
  167. """
  168. Return the 1-based index of the first object on this page,
  169. relative to total objects in the paginator.
  170. """
  171. # Special case, return zero if no items.
  172. if self.paginator.count == 0:
  173. return 0
  174. return (self.paginator.per_page * (self.number - 1)) + 1
  175. def end_index(self):
  176. """
  177. Return the 1-based index of the last object on this page,
  178. relative to total objects found (hits).
  179. """
  180. # Special case for the last page because there can be orphans.
  181. if self.number == self.paginator.num_pages:
  182. return self.paginator.count
  183. return self.number * self.paginator.per_page