Trong phần này chúng ta sẽ học cách phân trang. Django cung cấp lớp django.core.paginator
hỗ trợ phân trang rất tốt.
Ví dụ
Chúng ta tạo một app có tên là pagination.
C:\Project\mysite>python manage.py startapp pagination
Thêm app vào danh sách INSTALLED_APPS:
INSTALLED_APPS = [ #... 'pagination', #... ]
Tiếp theo chúng ta tạo model và thêm vào một số dòng dữ liệu mẫu để test.
from django.db import models # Create your models here. class Customer(models.Model): name = models.CharField(max_length=100) country = models.CharField(max_length=20)
Chúng ta định nghĩa model Customer
có 2 trường là name
và country.
Cập nhật lại CSDL:
python manage.py makemigrations python manage.py migrate
Đoạn script sau sẽ tạo một số dòng dữ liệu mẫu trong bảng pagination_customer.
import os import django os.environ.setdefault("DJANGO_SETTINGS_MODULE", "mysite.settings"); django.setup() from pagination.models import Customer Customer.objects.create(name='Alfreds Futterkiste', country='Germany') Customer.objects.create(name='Ana Trujillo Emparedados y helados', country='Mexico') Customer.objects.create(name='Antonio Moreno Taquería', country='Mexico') Customer.objects.create(name='Around the Horn', country='UK') Customer.objects.create(name='Berglunds snabbköp', country='Sweden') Customer.objects.create(name='Blauer See Delikatessen', country='Germany') Customer.objects.create(name='Blondel père et fils', country='France') Customer.objects.create(name='Bólido Comidas preparadas', country='Spain') Customer.objects.create(name='Bon app', country='France') Customer.objects.create(name='Bottom-Dollar Marketse', country='Canada') Customer.objects.create(name='Bs Beverages', country='UK') Customer.objects.create(name='Cactus Comidas para llevar', country='Argentina') Customer.objects.create(name='Centro comercial Moctezuma', country='Mexico') Customer.objects.create(name='Chop-suey Chinese', country='Switzerland') Customer.objects.create(name='Comércio Mineiro', country='Brazil') Customer.objects.create(name='Consolidated Holdings', country='UK') Customer.objects.create(name='Drachenblut Delikatessend', country='Germany') Customer.objects.create(name='Du monde entier', country='France') Customer.objects.create(name='Eastern Connection', country='UK')
Tiếp theo chúng ta viết template và hàm view.
from django.shortcuts import render # Create your views here. from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger from .models import Customer def listing(request): customer_list = Customer.objects.all() paginator = Paginator(customer_list, 5) pageNumber = request.GET.get('page') try: customers = paginator.page(pageNumber) except PageNotAnInteger: customers = paginator.page(1) except EmptyPage: customers = paginator.page(paginator.num_pages) return render(request, 'list.html', {'customers':customers})
Chúng ta sử dụng lớp Pagination
để thực hiện phân trang.
from django.core.paginator import Paginator, EmptyPage, PageNotAnInteger
Đầu tiên chúng ta import một số lớp cần thiết.
paginator = Paginator(customer_list, 5)
Hàm khởi tạo Paginator()
nhận vào 2 tham số, tham số đầu tiên là một đối tượng QuerySet,
tham số thứ 2 là số tượng item trên mỗi “trang”. Trong ví dụ trên chúng ta đưa đối tượng customer_list
vào với số lượng 5 item mỗi trang.
pageNumber = request.GET.get('page')
URL của chúng ta có thêm tham số page
là số thứ tự của trang muốn xem.
try: customers = paginator.page(pageNumber) except PageNotAnInteger: customers = paginator.page(1) except EmptyPage: customers = paginator.page(paginator.num_pages)
Nếu tham số page
không hợp lệ, chẳng hạn như page=abc
thì Paginator
sẽ giải phóng lỗi PageNotAnInterger,
trong trường hợp này chúng ta trả về trang đầu tiên với phương thức Paginator.page()
, hoặc nếu page nằm ngoài pham vi trang cho phép, chẳng hạn như chúng ta chỉ có 4 trang nhưng tham số page=1000
thì Paginator
sẽ giải phóng lỗi EmptyPage,
ở đây chúng ta xử lý bằng cách trả về trang cuối cùng bằng thuộc tính num_pages.
Template:
<table> <tr> <th>Customer name</th> <th>Country</th> </tr> {% for customer in customers %} <tr> <td>{{ customer.name }}</td> <td>{{ customer.country }}</td> </tr> {% endfor %} </table> <div class="pagination"> <span class="step-links"> {% if customers.has_previous %} <a href="?page={{ customers.previous_page_number }}">Previous</a> {% endif %} </span> <span class="current"> Page {{ customers.number }} of {{ customers.paginator.num_pages }}. </span> <span> {% if customers.has_next %} <a href="?page={{ customers.next_page_number }}">Next</a> {% endif %} </span> </div>
Cuối cùng chúng ta tạo URL và chạy server để xem kết quả.
from django.conf.urls import url from . import views urlpatterns = [ url(r'^$', views.listing), ]
from django.conf.urls import url, include from django.contrib import admin urlpatterns = [ url(r'^admin/', admin.site.urls), url(r'^pagination/', include('pagination.urls')), ]