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
- Serializer factory with Django Rest Framework by julio 5 months, 3 weeks ago
- Image compression before saving the new model / work with JPG, PNG by Schleidens 6 months, 1 week ago
- Help text hyperlinks by sa2812 7 months, 1 week ago
- Stuff by NixonDash 9 months, 1 week ago
- Add custom fields to the built-in Group model by jmoppel 11 months, 2 weeks 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.