Digg-like page numbering using inclusion tag.
Usage in template:
{% load pagination %}
{% pagination yourpage %}
Inclusion template pagination.html
:
{% load i18n %}
<div class="pagination">
<span class="step-links">
{% if page.has_previous %}
<a href="?page={{ page.previous_page_number }}" class="previous">{% trans "previous" %}</a>
{% endif %}
{% for pnum in begin %}
{% ifequal page.number pnum %}
<span class="current">{{ pnum }}</span>
{% else %}
<a href="?page={{ pnum }}">{{ pnum }}</a>
{% endifequal %}
{% endfor %}
{% if middle %}
<span class="continue">...</span>
{% for pnum in middle %}
{% ifequal page.number pnum %}
<span class="current">{{ pnum }}</span>
{% else %}
<a href="?page={{ pnum }}">{{ pnum }}</a>
{% endifequal %}
{% endfor %}
{% endif %}
{% if end %}
<span class="continue">...</span>
{% for pnum in end %}
{% ifequal page.number pnum %}
<span class="current">{{ pnum }}</span>
{% else %}
<a href="?page={{ pnum }}">{{ pnum }}</a>
{% endifequal %}
{% endfor %}
{% endif %}
{% if page.has_next %}
<a href="?page={{ page.next_page_number }}" class="next">{% trans "next" %}</a>
{% endif %}
</span>
</div>
Produces:
previous 1 2 ... 4 5 6 7 8 9 10 11 12 ... 17 18 next
Or:
1 2 3 4 5 6 7 8 ... 17 18 next
Or:
previous 1 2 ... 10 11 12 13 14 15 16 17 18 next
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 | from django import template
register = template.Library()
@register.inclusion_tag('pagination.html')
def pagination(page, begin_pages=2, end_pages=2, before_current_pages=4, after_current_pages=4):
# Digg-like pages
before = max(page.number - before_current_pages - 1, 0)
after = page.number + after_current_pages
begin = page.paginator.page_range[:begin_pages]
middle = page.paginator.page_range[before:after]
end = page.paginator.page_range[-end_pages:]
last_page_number = end[-1]
def collides(firstlist, secondlist):
""" Returns true if lists collides (have same entries)
>>> collides([1,2,3,4],[3,4,5,6,7])
True
>>> collides([1,2,3,4],[5,6,7])
False
"""
return any(item in secondlist for item in firstlist)
# If middle and end has same entries, then end is what we want
if collides(middle, end):
end = range(max(last_page_number - before_current_pages - after_current_pages, 1), last_page_number+1)
middle = []
# If begin and middle ranges has same entries, then begin is what we want
if collides(begin, middle):
begin = range(1, min(before_current_pages + after_current_pages, last_page_number)+1)
middle = []
# If begin and end has same entries then begin is what we want
if collides(begin, end):
begin = range(1, last_page_number+1)
end = []
return {'page' : page,
'begin' : begin,
'middle' : middle,
'end' : end}
|
More like this
- Template tag - list punctuation for a list of items by shapiromatron 11 months, 2 weeks ago
- JSONRequestMiddleware adds a .json() method to your HttpRequests by cdcarter 11 months, 3 weeks ago
- Serializer factory with Django Rest Framework by julio 1 year, 6 months ago
- Image compression before saving the new model / work with JPG, PNG by Schleidens 1 year, 7 months ago
- Help text hyperlinks by sa2812 1 year, 8 months ago
Comments
IMHO, this code have some bad logic, I replaced line 28 with:
and line 33 with:
Without those two changes, one can't access (in some cases) all pages with using only generated links
#
This is very much in use, and I would like to know when it fails if it fails...
But note that it is intended to drop entries if there are too many. When user selects entry from closer it will show the next entries, so they are not really gone.
#
Please login first before commenting.