paginator.py 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158
  1. from math import ceil
  2. class InvalidPage(Exception):
  3. pass
  4. class PageNotAnInteger(InvalidPage):
  5. pass
  6. class EmptyPage(InvalidPage):
  7. pass
  8. class Paginator(object):
  9. def __init__(self, object_list, per_page, orphans=0, allow_empty_first_page=True):
  10. self.object_list = object_list
  11. self.per_page = int(per_page)
  12. self.orphans = int(orphans)
  13. self.allow_empty_first_page = allow_empty_first_page
  14. self._num_pages = self._count = None
  15. def validate_number(self, number):
  16. "Validates the given 1-based page number."
  17. try:
  18. number = int(number)
  19. except (TypeError, ValueError):
  20. raise PageNotAnInteger('That page number is not an integer')
  21. if number < 1:
  22. raise EmptyPage('That page number is less than 1')
  23. if number > self.num_pages:
  24. if number == 1 and self.allow_empty_first_page:
  25. pass
  26. else:
  27. raise EmptyPage('That page contains no results')
  28. return number
  29. def page(self, number):
  30. "Returns a Page object for the given 1-based page number."
  31. number = self.validate_number(number)
  32. bottom = (number - 1) * self.per_page
  33. top = bottom + self.per_page
  34. if top + self.orphans >= self.count:
  35. top = self.count
  36. return Page(self.object_list[bottom:top], number, self)
  37. def _get_count(self):
  38. "Returns the total number of objects, across all pages."
  39. if self._count is None:
  40. try:
  41. self._count = self.object_list.count()
  42. except (AttributeError, TypeError):
  43. # AttributeError if object_list has no count() method.
  44. # TypeError if object_list.count() requires arguments
  45. # (i.e. is of type list).
  46. self._count = len(self.object_list)
  47. return self._count
  48. count = property(_get_count)
  49. def _get_num_pages(self):
  50. "Returns the total number of pages."
  51. if self._num_pages is None:
  52. if self.count == 0 and not self.allow_empty_first_page:
  53. self._num_pages = 0
  54. else:
  55. hits = max(1, self.count - self.orphans)
  56. self._num_pages = int(ceil(hits / float(self.per_page)))
  57. return self._num_pages
  58. num_pages = property(_get_num_pages)
  59. def _get_page_range(self):
  60. """
  61. Returns a 1-based range of pages for iterating through within
  62. a template for loop.
  63. """
  64. return range(1, self.num_pages + 1)
  65. page_range = property(_get_page_range)
  66. QuerySetPaginator = Paginator # For backwards-compatibility.
  67. class Page(object):
  68. def __init__(self, object_list, number, paginator):
  69. self.object_list = object_list
  70. self.number = number
  71. self.paginator = paginator
  72. def __repr__(self):
  73. return '<Page %s of %s>' % (self.number, self.paginator.num_pages)
  74. def __len__(self):
  75. return len(self.object_list)
  76. def __getitem__(self, index):
  77. # The object_list is converted to a list so that if it was a QuerySet
  78. # it won't be a database hit per __getitem__.
  79. return list(self.object_list)[index]
  80. # The following four methods are only necessary for Python <2.6
  81. # compatibility (this class could just extend 2.6's collections.Sequence).
  82. def __iter__(self):
  83. i = 0
  84. try:
  85. while True:
  86. v = self[i]
  87. yield v
  88. i += 1
  89. except IndexError:
  90. return
  91. def __contains__(self, value):
  92. for v in self:
  93. if v == value:
  94. return True
  95. return False
  96. def index(self, value):
  97. for i, v in enumerate(self):
  98. if v == value:
  99. return i
  100. raise ValueError
  101. def count(self, value):
  102. return sum([1 for v in self if v == value])
  103. # End of compatibility methods.
  104. def has_next(self):
  105. return self.number < self.paginator.num_pages
  106. def has_previous(self):
  107. return self.number > 1
  108. def has_other_pages(self):
  109. return self.has_previous() or self.has_next()
  110. def next_page_number(self):
  111. return self.number + 1
  112. def previous_page_number(self):
  113. return self.number - 1
  114. def start_index(self):
  115. """
  116. Returns the 1-based index of the first object on this page,
  117. relative to total objects in the paginator.
  118. """
  119. # Special case, return zero if no items.
  120. if self.paginator.count == 0:
  121. return 0
  122. return (self.paginator.per_page * (self.number - 1)) + 1
  123. def end_index(self):
  124. """
  125. Returns the 1-based index of the last object on this page,
  126. relative to total objects found (hits).
  127. """
  128. # Special case for the last page because there can be orphans.
  129. if self.number == self.paginator.num_pages:
  130. return self.paginator.count
  131. return self.number * self.paginator.per_page