Django组件——分页器(paginator)

一、视图层

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
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
from django.shortcuts import render
 
# Create your views here.
 
from .models import Book
from django.core.paginator import Paginator,EmptyPage
 
 
def index(request):
    """
    # 批量导入数据
    book_list=[]
    for i in range(100):
        book = Book.objects.create(title="book_%s" % i, price=i*i)
        book_list.append(book)
    Book.objects.bulk_create(book_list)
    """
 
    # 分页器的使用:
    book_list = Book.objects.all()
 
    paginator = Paginator(book_list, 8)   # 第二个参数设置每页显示的条数
 
    print(paginator.count)        # 数据总条数  100
    print(paginator.num_pages)    # 总页数  13
    print(paginator.page_range)   # 页码的列表 range(1, 14)
 
 
    page1 = paginator.page(1# 第一页的page对象
    # 显示某一页具体数据的两种方式:
    # 方式一:
    print("object_list", page1.object_list)  # object_list <QuerySet [<Book: Book object (1)>, <Book: Book object (2)>,...
    # 方式二:
    for i in page1:
        print(i)
    """输出
    Book object (1)
    Book object (2)
    Book object (3)
    Book object (4)
    Book object (5)
    Book object (6)
    Book object (7)
    Book object (8)
    """
 
    page2 = paginator.page(2)
    print(page2.has_next())         # 是否有下一页
    print(page2.next_page_number())   # 下一页的页码
    print(page2.has_previous())     # 是否有上一页
    print(page2.previous_page_number())  # 上一页的页码
 
 
    current_page_num = int(request.GET.get("page", 1))
 
    if paginator.num_pages > 11:
        if current_page_num-5 < 1:
            # 当前页码-5小于1时,默认显示前11个页码
            page_range = range(1, 12)
        elif current_page_num + 5 > paginator.num_pages:
            # 当前页码+5大于总页数时,默认显示后11个
            page_range = range(paginator.num_pages-10, paginator.num_pages+1)
        else:
            # 显示当前页码和左五右五共11个页码
            page_range = range(current_page_num-5, current_page_num+6)
    else:
        # 不超过11个页码的情况,显示所有页码
        page_range = paginator.page_range
 
    # 超出页码范围或小于1的页码,会抛出emptypage错误,做如下处理:
    try:
        current_page_num = int(request.GET.get("page", 1))  # /?page=1  get取不到默认给它默认值:1
        current_page = paginator.page(current_page_num)
 
        print("object_list", current_page.object_list)
        for i in current_page:
            print(i)
    except EmptyPage as e:
        current_page = paginator.page(1# 固定在出现emptypage时显示首页
 
    return render(request, "index.html", locals())

  注意:

  1、批量生成数据,添加到数组中,再一次性写入数据库表中:Book.objects.bulk_create(book_list)

  2、分页器对象生成后,.count查看数据总条数,.num_pages显示总页数,.page_range页码列表。

  3、显示某一页的具体数据有两种方式:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
# 方式一:
print("object_list", page1.object_list)  # object_list <QuerySet [<Book: Book object (1)>, <Book: Book object (2)>,...
# 方式二:
for i in page1:
    print(i)
"""输出
Book object (1)
Book object (2)
Book object (3)
Book object (4)
Book object (5)
Book object (6)
Book object (7)
Book object (8)
"""

  4、.page()方法可以生成页对象,具备如下方法可用于上一页下一页网页查看。

1
2
3
4
5
page2 = paginator.page(2)
print(page2.has_next())         # 是否有下一页
print(page2.next_page_number())   # 下一页的页码
print(page2.has_previous())     # 是否有上一页
print(page2.previous_page_number())  # 上一页的页码

  5、得到当前页码,且设置默认页码为1;根据当前页码得到当前页对象。

1
2
current_page_num = int(request.GET.get("page", 1))  # /?page=1  get取不到默认给它默认值:1
current_page = paginator.page(current_page_num)

  6、针对超出页码范围的页码,会抛出emptypage错误,可以做如下处理:

1
2
3
4
5
6
7
8
9
10
# 超出页码范围或小于1的页码,会抛出emptypage错误,做如下处理:
    try:
        current_page_num = int(request.GET.get("page", 1))  # /?page=1  get取不到默认给它默认值:1
        current_page = paginator.page(current_page_num)
 
        print("object_list", current_page.object_list)
        for i in current_page:
            print(i)
    except EmptyPage as e:
        current_page = paginator.page(1# 固定在出现emptypage时显示首页

  7、页面具备翻页功能,设置同时最多只能显示11条,处理如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
current_page_num = int(request.GET.get("page", 1))
 
if paginator.num_pages > 11:
    if current_page_num-5 < 1:
        # 当前页码-5小于1时,默认显示前11个页码
        page_range = range(1, 12)
    elif current_page_num + 5 > paginator.num_pages:
        # 当前页码+5大于总页数时,默认显示后11个
        page_range = range(paginator.num_pages-10, paginator.num_pages+1)
    else:
        # 显示当前页码和左五右五共11个页码
        page_range = range(current_page_num-5, current_page_num+6)
else:
    # 不超过11个页码的情况,显示所有页码
    page_range = paginator.page_range

二、模板层

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
46
47
48
49
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <link rel="stylesheet" href="/static/bootstrap-3.3.7/css/bootstrap.css">
</head>
<body>
<ul>
    {% for book in current_page %}
        <li>{{ book.title }}:{{ book.price }}</li>
    {% endfor %}
</ul>
<nav aria-label="Page navigation">
    <ul class="pagination">
{#        <li>#}
            {# 方法一:上一页在当前页码减1 #}
{#             <a href="?page={{ current_page_num|add:-1 }}" aria-label="Previous">#}
{#                 <span aria-hidden="true">上一页</span>#}
{#            </a>#}
{#        </li>#}
{#        方法二:#}
        {% if current_page.has_previous %}
            <li><a href="?page={{ current_page_num.previous_page_number }}" aria-label="Previous"><span aria-hidden="true">上一页</span></a></li>
        {% else %}
            {# 页码到第一页后,上一页设置为禁用 #}
            <li class="disabled"><a href="?page={{ current_page_num.previous_page_number }}" aria-label="Previous"><span aria-hidden="true">上一页</span></a></li>
        {% endif %}
 
        {# 自动生成<li>标签,路径前面不变得情况下从?开始配置即可 #}
        {% for item in page_range %}
            {# 当前页面添加active显示高亮 #}
            {% if current_page_num == item %}
                <li class="active"><a href="?page={{ item }}">{{ item }}</a></li>
            {% else %}
                <li><a href="?page={{ item }}">{{ item }}</a></li>
            {% endif %}
        {% endfor %}
 
        {% if current_page.has_next %}
            <li><a href="?page={{ current_page_num.next_page_number }}" aria-label="Next"><span aria-hidden="true">下一页</span></a></li>
        {% else %}
            <li class="disabled"><a href="?page={{ current_page_num.next_page_number }}" aria-label="Next"><span aria-hidden="true">下一页</span></a></li>
        {% endif %}
    </ul>
</nav>
 
</body>
</html>

注意:

  1、读取当前页码数据生成标签展示在页面上:

1
2
3
4
5
<ul>
    {% for book in current_page %}
        <li>{{ book.title }}:{{ book.price }}</li>
    {% endfor %}
</ul>

  2、引入bootstrap

1
2
3
4
5
6
7
8
9
10
<head>
    <link rel="stylesheet" href="/static/bootstrap-3.3.7/css/bootstrap.css">
</head>
 
body部分:
<nav aria-label="Page navigation">
    <ul class="pagination">
    ...
    </ul>
</nav>

  3、生成页面页码标签,路径前面不变的情况下可以从?开始配置:

1
2
3
4
5
6
7
8
9
{# 自动生成<li>标签,路径前面不变得情况下从?开始配置即可 #}
{% for item in page_range %}
    {# 当前页面添加active显示高亮 #}
    {% if current_page_num == item %}
        <li class="active"><a href="?page={{ item }}">{{ item }}</a></li>
    {% else %}
        <li><a href="?page={{ item }}">{{ item }}</a></li>
    {% endif %}
{% endfor %}

  4、生成页面翻页,上一页、下一页:

  (1)第一种方式是在当前页码做加减法:{{ current_page_num|add:-1 }};{{ current_page_num|add:1 }}

1
2
3
4
5
6
<li>
    <!--方法一:上一页在当前页码减1 -->
    <a href="?page={{ current_page_num|add:-1 }}" aria-label="Previous">
        <span aria-hidden="true">上一页</span>
    </a>
</li>

  (2)第二种方式是使用page对象自带的方法:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
<!--上一页-->
{% if current_page.has_previous %}
    <li><a href="?page={{ current_page_num.previous_page_number }}" aria-label="Previous"><span aria-hidden="true">上一页</span></a></li>
{% else %}
    {# 页码到第一页后,上一页设置为禁用 #}
    <li class="disabled"><a href="?page={{ current_page_num.previous_page_number }}" aria-label="Previous"><span aria-hidden="true">上一页</span></a></li>
{% endif %}
 
 
<!--下一页-->
{% if current_page.has_next %}
    <li><a href="?page={{ current_page_num.next_page_number }}" aria-label="Next"><span aria-hidden="true">下一页</span></a></li>
{% else %}
    <li class="disabled"><a href="?page={{ current_page_num.next_page_number }}" aria-label="Next"><span aria-hidden="true">下一页</span></a></li>
{% endif %}

 

posted @   休耕  阅读(980)  评论(0编辑  收藏  举报
编辑推荐:
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
阅读排行:
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 地球OL攻略 —— 某应届生求职总结
· 周边上新:园子的第一款马克杯温暖上架
· Open-Sora 2.0 重磅开源!
· 提示词工程——AI应用必不可少的技术
点击右上角即可分享
微信分享提示