Django 分页 以及自定义分页

Django提供了一个新的类来帮助你管理分页数据,这个类存放在django/core/paginator.py.它可以接收列表、元组或其它可迭代的对象。

基本语法

1
2
3
4
5
6
7
8
9
10
class Paginator(object):
 
    def __init__(self, object_list, per_page, orphans=0,
                 allow_empty_first_page=True):
        self.object_list = object_list
        self.per_page = int(per_page)
        self.orphans = int(orphans)
        self.allow_empty_first_page = allow_empty_first_page
 
……

 

基本语法实例

from django.core.paginator import Paginator
objects = ['john','paul','george','ringo','lucy','meiry','checy','wind','flow','rain']
p = Paginator(objects,3) # 3条数据为一页,实例化分页对象
print(p.count) # 10 对象总共10个元素
print(p.num_pages) # 4 对象可分4页
print(p.page_range) # xrange(1, 5) 对象页的可迭代范围

page1 = p.page(1) # 取对象的第一分页对象
print(page1.object_list) # 第一分页对象的元素列表['john', 'paul', 'george']
print(page1.number) # 第一分页对象的当前页值 1

page2 = p.page(2) # 取对象的第二分页对象
print(page2.object_list) # 第二分页对象的元素列表 ['ringo', 'lucy', 'meiry']
print(page2.number) # 第二分页对象的当前页码值 2

print(page1.has_previous()) # 第一分页对象是否有前一页 False
print(page1.has_other_pages()) # 第一分页对象是否有其它页 True

print(page2.has_previous()) # 第二分页对象是否有前一页 True
print(page2.has_next()) # 第二分页对象是否有下一页 True
print(page2.next_page_number()) # 第二分页对象下一页码的值 3
print(page2.previous_page_number()) # 第二分页对象的上一页码值 1
print(page2.start_index()) # 第二分页对象的元素开始索引 4
print(page2.end_index()) # 第2分页对象的元素结束索引 6

 

 官方解释

在视图中的应用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
from django.shortcuts import render
 
def listing(request):
    contact_list = Contacts.objects.all()
    paginator = Paginator(contact_list, 25) # Show 25 contacts per page
 
    page = request.GET.get('page')
    try:
        contacts = paginator.page(page)
    except PageNotAnInteger:
        # If page is not an integer, deliver first page.
        contacts = paginator.page(1)
    except EmptyPage:
        # If page is out of range (e.g. 9999), deliver last page of results.
        contacts = paginator.page(paginator.num_pages)
 
    return render(request, 'list.html', {'contacts': contacts})

 

在template的html模板中的应用

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
{% for contact in contacts %}
    {# Each "contact" is a Contact model object. #}
    {{ contact.full_name|upper }}<br />
    ...
{% endfor %}
 
<div class="pagination">
    <span class="step-links">
        {% if contacts.has_previous %}
            <a href="?page={{ contacts.previous_page_number }}">previous</a>
        {% endif %}
 
        <span class="current">
            Page {{ contacts.number }} of {{ contacts.paginator.num_pages }}.
        </span>
 
        {% if contacts.has_next %}
            <a href="?page={{ contacts.next_page_number }}">next</a>
        {% endif %}
    </span>
</div>

分割线。。。。。

回归到easyadmin项目中,我们来定制自己的分页功能,当然数据的获取我们依旧通过对象来提取,但是难点在于href属性,我们要如何构造url是关键,因为我们的分页功能是在过滤功能的基础上,因此不能简单的使用?page=%s 格式,而且为了能得到更好的体验也迫使我们不得不重写分页功能

首先看下视图函数吧

def display_table_objs(request,app_name,table_name):
    admin_class = easy_admin.enabled_admins[app_name][table_name]
    object_list,filter_condtions = table_filter(request, admin_class)
    paginator = Paginator(object_list, admin_class.list_per_page)  # Show 25 contacts per page
    print(request.get_full_path())
    page = request.GET.get('page')
    try:
        object_list=paginator.page(page)
    except PageNotAnInteger:
        object_list=paginator.page(1)
    except EmptyPage:
        object_list=paginator.page(paginator.num_pages)
    return render(request,'easyadmin/display_table.html',{"admin_class":admin_class,"filter_condtions":filter_condtions,'object_list':object_list})

基本没有改动 接下来看下自定义标签

 

@register.simple_tag()
def easypaginator(request,object_list):
    """now_page:当前页
        show_page:需要显示多少页
        num_pages:总共有多少页
    """
    now_page=object_list.number
    show_page=6
    num_pages=object_list.paginator.num_pages
    base_url=""
    for k,v in request.GET.items():
        if k=='page':
            continue
        if v:
            base_url+="&%s=%s" %(k,v)
    if num_pages <= show_page:
        """如果总页数小于需要展示的页数 那么久把所有的页码都打出来吧"""
        start_page=1
        end_page=num_pages
    else:
        start_page = now_page - (show_page - 1) / 2
        end_page = now_page + (show_page - 1) / 2
        if start_page <= 0:
            start_page = 1
            end_page = start_page + show_page - 1
        if end_page >= num_pages:
            end_page = num_pages
            start_page = num_pages - show_page + 1
    str_list = []
    if now_page == 1:
        prevpage = '<li class="disabled"><a>上一页</a></li>'
    else:
        active_url="?page=%s" % str(now_page-1)
        prevpage = '<li><a href=%s>上一页</a></li>' %(active_url+base_url)
    if now_page == num_pages:
        lastpage = '<li class="disabled" ><a href="#">下一页</a></li>'
    else:
        active_url = "?page=%s" % str(now_page + 1)
        lastpage='<li><a class="" href=%s>下一页</a></li>' %(active_url+base_url)
    str_list.append(prevpage)
    for i in range(int(start_page), int(end_page + 1)):
        if i == now_page:
            active_url = "?page=%s" % str(i)
            tmp = '<li class="active" ><a href=%s >%s</a></li>' %(active_url+base_url,str(i))
        else:
            active_url = "?page=%s" % str(i)
            tmp = '<li class="" ><a href=%s >%s</a></li>' % (active_url + base_url, str(i))
        str_list.append(tmp)
    str_list.append(lastpage)
    omit='<li class="" ><a>....</a></li>'
    if num_pages > show_page:
        str_list.insert(int(len(str_list)/2),omit)
    str_list = mark_safe("".join(str_list))
    return str_list

 

 

 

  我们接受两个对象,一个request,一个是object_list也就是数据。那么我们可以获得哪些数据呢?总共的页数,当前页数,以及params,同时我们也可以设置需要显示多少页的内容,而不是显示100页或者更多,这样更符合人们的需求,这个值的改变不是很多我们就直接写在函数中,而不考虑写在baseadmin类中。通过params我们可以拼接url,通过当前页我们可以确定标签的样式,当然这都是基于bootsharp插件,当然如果不用bootsharp对分页功能完全不影响。

 <nav class="page pagination">
      <ul class="pagination">
      {% easypaginator request object_list %}
       </ul>
 </nav>

效果:

 

posted @ 2017-10-20 09:38  童小哥总是不开心。  阅读(530)  评论(0编辑  收藏  举报