1. 程式人生 > 實用技巧 >django框架之分頁器及forms元件等相關內容-72

django框架之分頁器及forms元件等相關內容-72

1 分頁器元件介紹

1 專案資料量大了以後,比如涉及到分頁,一頁一頁的載入顯示
2 django中分頁器元件,把分頁常用的東西,封裝到一個類中
3 例項化得到一個物件,物件裡有屬性和方法

2 分頁器的簡單使用

#######1 Paginator物件的屬性和方法
book_list=models.Book.objects.all()
# 例項化得到物件
# 第一個引數:要分頁的資料,book_list
# 第二個引數:沒有條數
paginator=Paginator(book_list,10)
# Paginator物件的屬性和方法
print(paginator.per_page) # 每頁顯示的條數
print(paginator.count) # 總條數,總共要分頁多少條資料
print(paginator.num_pages) # 總頁碼數
print(paginator.page_range) # 頁碼的生成器 [1,2,3,4,5,6,7,8,9,10]



######3 Page物件的屬性和方法
# Page類 的物件
page=paginator.page(2) # 第一頁的物件
# 每一頁的物件,屬性和方法
print(page.has_next()) # 有沒有下一頁
print(page.next_page_number()) # 下一頁頁碼
print(page.has_previous()) # 是否有上一頁
print(page.previous_page_number()) # 上一頁頁面 (當前頁如果是第一頁,沒有上一頁)
print(page.object_list) # 當前頁的所有資料
print(page.number) # 當前頁的頁碼數


##### 4 表模型中預設以id排序
class Meta:
ordering=('id', ) # 預設以id排序

檢視

def index(request):
# 需要的第三個引數
page_num_int=int(request.GET.get('page',1))
book_list = models.Book.objects.all()
paginator = Paginator(book_list, 10)

# 需要的第一個引數:頁碼的生成器 [1,2,3,4,5,6,7,8,9,10]
page_range = paginator.page_range
# 需要的第二個引數,去到某一頁的page物件
page = paginator.page(page_num_int)
return render(request, 'index.html', {'page_range':page_range,'page':page,'page_num_int':page_num_int})

模板

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<link rel="stylesheet" href="/static/bootstrap/css/bootstrap.min.css">
<title>Title</title>
</head>
<body>
<div class="container-fluid">
<div class="row">
<div class="col-md-6 col-md-offset-3">
<table class="table">
<thead>
<tr>
<th>id</th>
<th>名字</th>
<th>價格</th>
</tr>
</thead>
<tbody>


{% for book in page.object_list %}
<tr>
<td>{{ book.id }}</td>
<td>{{ book.name }}</td>
<td>{{ book.price }}</td>
</tr>
{% endfor %}
</tbody>

</table>

<div class="text-center">
<nav aria-label="Page navigation">
<ul class="pagination">

{% if page.has_previous %}
<li>
<a href="/?page={{ page.previous_page_number }}" aria-label="Previous">
<span aria-hidden="true">&laquo;</span>
</a>
</li>
{% else %}
<li class="disabled">
<a href="" aria-label="Previous">
<span aria-hidden="true">&laquo;</span>
</a>
</li>
{% endif %}



{% for page_num in page_range %}
{% if page_num_int == page_num %}
<li class="active"><a href="/?page={{ page_num }}">{{ page_num }}</a></li>
{% else %}
<li><a href="/?page={{ page_num }}">{{ page_num }}</a></li>
{% endif %}

{% endfor %}


{% if page.has_next %}
<li>
<a href="/?page={{ page.next_page_number }}" aria-label="Next">
<span aria-hidden="true">&raquo;</span>
</a>
</li>
{% else %}
<li class="disabled">
<a href="" aria-label="Next">
<span aria-hidden="true">&raquo;</span>
</a>
</li>
{% endif %}

</ul>
</nav>

</div>
</div>


</div>


</div>
</body>
</html>

3 分頁器的進階使用

# 最多顯示前5 後5 和當前,總共11個頁碼,如果少於11,全部顯示出來

#邏輯分析
顯示左5,右5,總共11個頁,
1 如果總頁碼大於11
1.1 if 當前頁碼減5小於1,要生成1到12的列表(顧頭不顧尾,共11個頁碼)
page_range=range(1,12)
1.2 elif 當前頁碼+5大於總頁碼,生成當前頁碼減10,到當前頁碼加1的列表(顧頭不顧尾,共11個頁碼)
page_range=range(paginator.num_pages-10,paginator.num_pages+1)
1.3 else 生成當前頁碼-5,到當前頁碼+6的列表
page_range=range(current_page_num-5,current_page_num+6)
2 其它情況,生成的列表就是pageinator的page_range
page_range=paginator.page_range

檢視

def index(request):
# 需要的第三個引數
page_num_int = int(request.GET.get('page', 1))
book_list = models.Book.objects.all()
paginator = Paginator(book_list, 1)

# 需要的第一個引數:頁碼的生成器 [1,2,3,4,5,6,7,8,9,10]
# page_range = paginator.page_range
if paginator.num_pages > 11:

# 當前條件符合了以後,有三種情況
if page_num_int - 5 < 1:
page_range = range(1, 11)
elif page_num_int + 5 > paginator.num_pages:
page_range = range(paginator.num_pages - 10, paginator.num_pages + 1)
else:
page_range = range(page_num_int - 5, page_num_int + 5)
else:
page_range = paginator.page_range
# 需要的第二個引數,去到某一頁的page物件
page = paginator.page(page_num_int)
return render(request, 'index.html', {'page_range': page_range, 'page': page, 'page_num_int': page_num_int})

模板

<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<link rel="stylesheet" href="/static/bootstrap/css/bootstrap.min.css">
<title>Title</title>
</head>
<body>
<div class="container-fluid">
<div class="row">
<div class="col-md-6 col-md-offset-3">
<table class="table">
<thead>
<tr>
<th>id</th>
<th>名字</th>
<th>價格</th>
</tr>
</thead>
<tbody>


{% for book in page.object_list %}
<tr>
<td>{{ book.id }}</td>
<td>{{ book.name }}</td>
<td>{{ book.price }}</td>
</tr>
{% endfor %}
</tbody>

</table>

<div class="text-center">
<nav aria-label="Page navigation">
<ul class="pagination">

{% if page.has_previous %}
<li>
<a href="/?page={{ page.previous_page_number }}" aria-label="Previous">
<span aria-hidden="true">&laquo;</span>
</a>
</li>
{% else %}
<li class="disabled">
<a href="" aria-label="Previous">
<span aria-hidden="true">&laquo;</span>
</a>
</li>
{% endif %}



{% for page_num in page_range %}
{% if page_num_int == page_num %}
<li class="active"><a href="/?page={{ page_num }}">{{ page_num }}</a></li>
{% else %}
<li><a href="/?page={{ page_num }}">{{ page_num }}</a></li>
{% endif %}

{% endfor %}


{% if page.has_next %}
<li>
<a href="/?page={{ page.next_page_number }}" aria-label="Next">
<span aria-hidden="true">&raquo;</span>
</a>
</li>
{% else %}
<li class="disabled">
<a href="" aria-label="Next">
<span aria-hidden="true">&raquo;</span>
</a>
</li>
{% endif %}

</ul>
</nav>

</div>
</div>


</div>


</div>
</body>
</html>

4 forms元件介紹

1 註冊功能,登入功能,前端需要校驗(欄位長度,郵箱是否合法。。。)
2 前端校驗可以沒有,後端校驗是必須的,使用傳統方式 if判斷寫的很多
3 藉助於forms元件,可以快速實現欄位的校驗
from django.forms import Form

5 forms校驗欄位功能

### 1 寫一個類,類裡寫要校驗的欄位
class MyForm(forms.Form):
# 校驗這個欄位,最大長度是32,最小長度是3
name = forms.CharField(required=False, max_length=32, min_length=3,label='使用者名稱')
email = forms.EmailField(label='郵箱')
age=forms.IntegerField(max_value=200,min_value=0,label='年齡')


### 2 檢視函式中使用
def register(request):
# 資料可以是從前端傳過來的,也可以是自己後臺的資料

# 我現在有以下資料
data={'name':'lqz','email':'[email protected]','age':900}
# data={'email':'[email protected]','age':100}
# data={'age':100}
# 校驗資料是否合法
# 例項化得到form物件,把要校驗的資料傳入
form=myforms.MyForm(data)
# 校驗資料:form.is_valid() 返回布林型別

if form.is_valid():
print('校驗通過')
# 校驗通過的資料
print(form.cleaned_data) # 不一定是上面傳入的資料
else:
print(form.cleaned_data)
print('校驗失敗')
# 哪個欄位失敗了?失敗的原因是什麼
print(form.errors)
print(type(form.errors))
from django.forms.utils import ErrorDict
#### 重寫了__str__
print(form.errors.as_json())
print(form.errors.as_data())

# form.errors.as_ul() # 是為了渲染模板


return HttpResponse('ok')

6 forms渲染模板功能

## 檢視函式
def register(request):
if request.method=='GET':
form=myforms.MyForm()
return render(request,'register.html',{'form':form})
elif request.method=='POST':
# 資料校驗
form=myforms.MyForm(request.POST)
if form.is_valid():
print('校驗通過,存資料庫')
else:
print(form.errors.as_data())
print('校驗失敗,返回錯誤')
return HttpResponse('ok')


## 模板
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
<hr>
<h1>手動建立模板</h1>
<form action="" method="post">
<p>使用者名稱:<input type="text" name="name"></p>
<p>郵箱:<input type="text" name="email"></p>
<p>年齡:<input type="text" name="age"></p>
<p><input type="submit" value="提交"></p>
</form>

<hr>
<h1>半自動渲染模板1</h1>
<form action="" method="post">
<p>使用者名稱:{{ form.name }}</p>
<p>郵箱:{{ form.email }}</p>
<p>年齡:{{ form.age }}</p>
<p><input type="submit" value="提交"></p>
</form>

<h1>半自動渲染模板2(用的最多)</h1>
<form action="" method="post">
<p>{{ form.name.label }}--{{ form.name }}</p>
<p>{{ form.email.label }}---{{ form.email }}</p>
<p>{{ form.age.label }}---{{ form.age }}</p>
<p><input type="submit" value="提交"></p>
</form>

<h1>半自動渲染模板3(用的最多)</h1>
<form action="" method="post">
{% for foo in form %}
<p>{{ foo.label }} :{{ foo }}</p>
{% endfor %}

<p><input type="submit" value="提交"></p>
</form>

<h1>全自動(瞭解)</h1>
<form action="" method="post">
{# {{ form.as_ul }}#}
{{ form.as_p }}
{# <table>#}
{# {{ form.as_table }}#}
{# </table>#}

<p><input type="submit" value="提交"></p>
</form>
</body>
</html>