froms组件前戏
需求:
"""
写一个注册功能
获取用户名和密码 利用form表单提交数据
在后端判断用户名和密码是否符合一定的条件
用户名中不能含有666
密码不能少于三位
如何符合条件需要你将提示信息展示到前端页面
"""
普通方式实现(不结合ajax)
views.py
from django.shortcuts import render
# Create your views here.
def ab_form(request):
back_dic = {"username":"","password":""}
if request.method == "POST":
username = request.POST.get("username")
password = request.POST.get("password")
if "666" in username:
back_dic["username"] = "*用户名不合法"
if len(password) < 3:
back_dic["password"] = "*密码长度小于3"
return render(request,"ab_form.html",locals())
"""
无论是post请求还是get请求
页面都能够获取到字典 只不过get请求来的时候 字典值都是空的
而post请求来之后 字典可能有值
"""
ab_form.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<!-- Bootstrap3 核心 CSS 文件 -->
<link href="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet">
<!-- jQuery文件。务必在bootstrap.min.js 之前引入 -->
<script src="https://cdn.staticfile.org/jquery/2.1.1/jquery.min.js"></script>
<!-- Bootstrap3 核心 JavaScript 文件 -->
<script src="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/js/bootstrap.min.js"></script>
<!-- font-awesome.min.css图标库4.7版本 -->
<link href="https://cdn.bootcss.com/font-awesome/4.7.0/css/font-awesome.min.css" rel="stylesheet">
</head>
<body>
<div class="container-fluid">
<div class="row">
<div class="col-md-8 col-md-offset-2">
<form action="" method="post">
<p>username:
<input type="text" class="form-control" name="username">
<span style="color: red">{{ back_dic.username }}</span>
</p>
<p>password:
<input type="text" class="form-control" name="password">
<span style="color: red">{{ back_dic.password }}</span>
</p>
<input type="submit" class="btn btn-info btn-block">
</form>
</div>
</div>
</div>
</body>
</html>
效果:
使用到的技术点
# 1.手动书写前端获取用户数据的html代码 渲染html代码
# 2.后端对用户数据进行校验 校验数据
# 3.对不符合要求的数据进行前端提示 展示提示信息
forms组件实现
forms组件能够完成的事情:
# 1.渲染html代码
# 2.校验数据
# 3.展示提示信息
urls.py
from django.conf.urls import url
from django.contrib import admin
from app01 import views
urlpatterns = [
url(r'^admin/', admin.site.urls),
# forms组件
url(r'^ab_form/', views.ab_form),
]
urls.py
views.py,本文forms是直接写在views.py中的,也可以在app01应用中单独建一个文件夹或py文件解耦合,然后再views.py中导入
from django.shortcuts import render,HttpResponse
# 基本使用
from django import forms # forms组件所需模块
class MyForms(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(max_length=8,min_length=3,label="用户名",
error_messages={
'min_length':'用户名最少3位',
'max_length':'用户名最大8位',
'required':"用户名不能为空"
}
)
# password字符串类型最小3位最大8位
password = forms.CharField(max_length=8,min_length=3,label="密码",
error_messages = {
'min_length': '密码最少3位',
'max_length': '密码最大8位',
'required': "密码不能为空"
}
)
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField(label="邮箱",
error_messages={
'invalid': '邮箱格式不正确',
'required': "邮箱不能为空"
}
)
# 钩子函数
# 局部钩子
def clean_username(self): # 用来校验username,走这一步前提是上面username字段条件满足了
# 获取到用户名
username = self.cleaned_data.get("username")
if '666' in username:
# 提示前端展示报错信息
self.add_error("username","用户名不合法") # 添加一个错误信息,第一个参数是给哪个字段增加错误信息,第二个参数是错误信息内容
# 将钩子函数钩取到的数据再放回去
return username
# 校验数据
def ab_form(request):
# 1 先产生一个空对象
form_obj = MyForms()
if request.method == "POST":
# 获取用户数据
"""
1.数据获取繁琐
2.校验数据需要构造成字典的格式传入才行
ps:但是request.POST可以看成就是一个字典
"""
# 3.校验数据
form_obj = MyForms(request.POST)
# 4.判断数据是否合法
if form_obj.is_valid():
# 5.如果合法 操作数据库存储数据
return HttpResponse('OK')
# 5.不合法 有错误
return render(request,"ab_form.html",locals())
ab_form.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<!-- Bootstrap3 核心 CSS 文件 -->
<link href="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet">
<!-- jQuery文件。务必在bootstrap.min.js 之前引入 -->
<script src="https://cdn.staticfile.org/jquery/2.1.1/jquery.min.js"></script>
<!-- Bootstrap3 核心 JavaScript 文件 -->
<script src="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/js/bootstrap.min.js"></script>
<!-- font-awesome.min.css图标库4.7版本 -->
<link href="https://cdn.bootcss.com/font-awesome/4.7.0/css/font-awesome.min.css" rel="stylesheet">
</head>
<body>
<div class="container-fluid">
<div class="row">
<div class="col-md-8 col-md-offset-2">
<form action="" method="post" novalidate>
{% for form in form_obj %}
<p>
{{ form.label }}:{{ form }}
<span style="color: red">{{ form.errors.0 }}</span>
</p>
{% endfor %}
<input type="submit" class="btn btn-info">
</form>
</div>
</div>
</div>
</body>
</html>
效果
补充:
为什么数据校验非要去后端 不能在前端利用js直接完成呢?
数据校验前端可有可无
但是后端必须要有!!!
因为前端的校验是弱不禁风的 你可以直接修改
或者利用爬虫程序绕过前端页面直接朝后端提交数据
购物网站
选取了货物之后 会计算一个价格发送给后端 如果后端不做价格的校验
实际是获取到用户选择的所有商品的主键值
然后在后端查询出所有商品的价格 再次计算一遍
如果跟前端一致 那么完成支付如果不一致直接拒绝
forms组件基本使用
from django.shortcuts import render
# 基本使用
from django import forms # forms组件所需模块
class MyForms(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(max_length=8,min_length=3)
# password字符串类型最小3位最大8位
password = forms.CharField(max_length=8,min_length=3)
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField()
froms组件校验数据
"""
1.测试环境的准备 可以自己拷贝代码准备
2.其实在pycharm里面已经帮你准备一个测试环境
python console
"""
在python console操作示例:
from app01 import views
# 1 将带校验的数据组织成字典的形式传入即可
form_obj = views.MyForm({'username':'jason','password':'123','email':'123'})
# 2 判断数据是否合法 注意该方法只有在所有的数据全部合法的情况下才会返回True
form_obj.is_valid()
False
# 3 查看所有校验通过的数据
form_obj.cleaned_data
{'username': 'jason', 'password': '123'}
# 4 查看所有不符合校验规则以及不符合的原因
form_obj.errors
{
'email': ['Enter a valid email address.']
}
# 5 校验数据只校验类中出现的字段 多传不影响 多传的字段直接忽略
form_obj = views.MyForm({'username':'jason','password':'123','email':'123@qq.com','hobby':'study'})
form_obj.is_valid()
True
# 6 校验数据 默认情况下 类里面所有的字段都必须传值
form_obj = views.MyForm({'username':'jason','password':'123'})
form_obj.is_valid()
False
"""
也就意味着校验数据的时候 默认情况下数据可以多传但是绝不可能少传
"""
forms渲染标签
"""
forms组件只会自动帮你渲染获取用户输入的标签(input select radio checkbox)
不能帮你渲染提交按钮
"""
三种渲染方式:
views.py
from django.shortcuts import render,HttpResponse
# 基本使用
from django import forms # forms组件所需模块
class MyForms(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(max_length=8,min_length=3)
# password字符串类型最小3位最大8位
password = forms.CharField(max_length=8,min_length=3)
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField()
# 校验数据
def ab_form(request):
# 1 先产生一个空对象
form_obj = MyForms()
# 2 直接将该空对象传递给html页面
return render(request,"ab_form.html",locals())
ab_form.html,前端利用空对象做操作
第一种渲染方式:
<form action="" method="post">
<p>第一种渲染方式:代码书写极少,封装程度太高 不便于后续的扩展 一般情况下只在本地测试使用</p>
{{ form_obj.as_p }}
{{ form_obj.as_ul }}
{{ form_obj.as_table }}
<input type="submit" class="btn btn-info btn-block">
</form>
第二种渲染方式:
<form action="" method="post">
<p>第二种渲染方式:可扩展性很强 但是需要书写的代码太多 一般情况下不用</p>
<!--form_obj.username(forms类中的字段名):会帮你渲染一个name属性值为username(那个类中字段名)的input框-->
<!--form_obj.username.label:拿到标签注释-->
<p>{{ form_obj.username.label }}:{{ form_obj.username }}</p>
<p>{{ form_obj.password.label }}:{{ form_obj.password }}</p>
<p>{{ form_obj.email.label }}:{{ form_obj.email }}</p>
<input type="submit" class="btn btn-info btn-block">
</form>
补充:
"""
label属性默认展示的是类中定义的字段首字母大写的形式
也可以自己修改 直接给forms类中字段对象加label属性即可
username = forms.CharField(min_length=3,max_length=8,label='用户名')
"""
示例:
from django.shortcuts import render,HttpResponse
# 基本使用
from django import forms # forms组件所需模块
class MyForms(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(max_length=8,min_length=3,label="用户名")
# password字符串类型最小3位最大8位
password = forms.CharField(max_length=8,min_length=3,label="密码")
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField(label="邮箱")
# 校验数据
def ab_form(request):
# 1 先产生一个空对象
form_obj = MyForms()
# 2 直接将该空对象传递给html页面
return render(request,"ab_form.html",locals())
字段添加label属性
效果:
第三种渲染方式
<form action="" method="post">
<p>第三种渲染方式(推荐使用):代码书写简单 并且扩展性也高</p>
{% for form in form_obj %} <!--此时的form等价于form_obj.username-->
<p>{{ form.label }}:{{ form }}</p>
{% endfor %}
<input type="submit" class="btn btn-info btn-block">
</form>
效果:
渲染标签总结:
"""
forms组件只会自动帮你渲染获取用户输入的标签(input select radio checkbox)
不能帮你渲染提交按钮
"""
def index(request):
# 1 先产生一个空对象
form_obj = MyForm()
# 2 直接将该空对象传递给html页面
return render(request,'index.html',locals())
# 前端利用空对象做操作
<p>第一种渲染方式:代码书写极少,封装程度太高 不便于后续的扩展 一般情况下只在本地测试使用</p>
{{ form_obj.as_p }}
{{ form_obj.as_ul }}
{{ form_obj.as_table }}
<p>第二种渲染方式:可扩展性很强 但是需要书写的代码太多 一般情况下不用</p>
<p>{{ form_obj.username.label }}:{{ form_obj.username }}</p>
<p>{{ form_obj.password.label }}:{{ form_obj.password }}</p>
<p>{{ form_obj.email.label }}:{{ form_obj.email }}</p>
<p>第三种渲染方式(推荐使用):代码书写简单 并且扩展性也高</p>
{% for form in form_obj %}
<p>{{ form.label }}:{{ form }}</p>
{% endfor %}
"""
字段的label属性默认展示的是类中定义的字段首字母大写的形式
也可以自己修改 直接给字段对象加label属性即可
username = forms.CharField(min_length=3,max_length=8,label='用户名')
"""
补充:可以通过auto_id获取forms渲染的input框的id
{% for form in form_obj %}
<div class="from-group">
<!--form.auto_id获取forms渲染的input框的id-->
<label for="{{ form.auto_id }}">{{ form.label }}</label>
{{ form }}
<span style="color: red" class="pull-right"></span>
</div>
{% endfor %}
forms展示提示信息
示例:
urls.py
from django.conf.urls import url
from django.contrib import admin
from app01 import views
urlpatterns = [
url(r'^admin/', admin.site.urls),
# forms组件
url(r'^ab_form/', views.ab_form),
]
urls.py
views.py
from django.shortcuts import render,HttpResponse
# 基本使用
from django import forms # forms组件所需模块
class MyForms(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(max_length=8,min_length=3,label="用户名")
# password字符串类型最小3位最大8位
password = forms.CharField(max_length=8,min_length=3,label="密码")
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField(label="邮箱")
# 校验数据
def ab_form(request):
# 1 先产生一个空对象
form_obj = MyForms()
if request.method == "POST":
# 获取用户数据
"""
1.数据获取繁琐
2.校验数据需要构造成字典的格式传入才行
ps:但是request.POST可以看成就是一个字典
"""
# 3.校验数据
form_obj = MyForms(request.POST) # 注意这里变量名一定要和产生的空对象的变量名一致
# 4.判断数据是否合法
if form_obj.is_valid():
# 5.如果合法 操作数据库存储数据
return HttpResponse('OK')
# 5.不合法 在前端通过forms对象errors结合errors展示错误信息
return render(request,"ab_form.html",locals())
注意:
"""
1.必备的条件 get请求和post传给html页面对象变量名必须一样
2.forms组件当你的数据不合法的情况下 会保存你上次的数据 让你基于之前的结果进行修改
更加的人性化
"""
ab_form.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<!-- Bootstrap3 核心 CSS 文件 -->
<link href="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet">
<!-- jQuery文件。务必在bootstrap.min.js 之前引入 -->
<script src="https://cdn.staticfile.org/jquery/2.1.1/jquery.min.js"></script>
<!-- Bootstrap3 核心 JavaScript 文件 -->
<script src="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/js/bootstrap.min.js"></script>
<!-- font-awesome.min.css图标库4.7版本 -->
<link href="https://cdn.bootcss.com/font-awesome/4.7.0/css/font-awesome.min.css" rel="stylesheet">
</head>
<body>
<div class="container-fluid">
<div class="row">
<div class="col-md-8 col-md-offset-2">
<form action="" method="post" novalidate>
{% for form in form_obj %}
<p>
{{ form.label }}:{{ form }}
<span style="color: red">{{ form.errors }}</span>
</p>
{% endfor %}
<input type="submit" class="btn btn-info">
</form>
</div>
</div>
</div>
</body>
</html>
浏览器输入不符合校验的数据
注意:这个提示信息是浏览器自动帮你校验的
"""
浏览器会自动帮你校验数据 但是前端的校验弱不禁风
如何让浏览器不做校验
<form action="" method="post" novalidate>
"""
关闭浏览器自动校验以及注意事项
<form action="" method="post" novalidate>
{% for form in form_obj %}
<p>
{{ form.label }}:{{ form }}
<span style="color: red">{{ form.errors }}</span>
</p>
{% endfor %}
<input type="submit" class="btn btn-info">
</form>
发现错误信息是以li标签展示的
标准姿势:form.errors.0:只拿列表第一个错误信息
<form action="" method="post" novalidate>
{% for form in form_obj %}
<p>
{{ form.label }}:{{ form }}
<span style="color: red">{{ form.errors.0 }}</span>
</p>
{% endfor %}
<input type="submit" class="btn btn-info">
</form>
错误提示信息默认为英文,可以自定义,看下面自定义错误提示信息。
自定义错误提示信息
字段的添加error_messages={“错误条件”:”错误信息”,”错误条件”:”错误信息”…}
# 针对错误的提示信息还可以自己自定制
class MyForm(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(min_length=3,max_length=8,label='用户名',
error_messages={
'min_length':'用户名最少3位',
'max_length':'用户名最大8位',
'required':"用户名不能为空"
}
)
# password字符串类型最小3位最大8位
password = forms.CharField(min_length=3,max_length=8,label='密码',
error_messages={
'min_length': '密码最少3位',
'max_length': '密码最大8位',
'required': "密码不能为空"
}
)
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField(label='邮箱',
error_messages={
'invalid':'邮箱格式不正确',
'required': "邮箱不能为空"
}
)
forms组件钩子函数(HOOK)
"""
在特定的节点自动触发完成响应操作
钩子函数在forms组件中就类似于第二道关卡,能够让我们自定义校验规则
在forms组件中有两类钩子
1.局部钩子
当你需要给单个字段增加校验规则的时候可以使用
2.全局钩子
当你需要给多个字段增加校验规则的时候可以使用
"""
实际案例
需求:
# 1.校验用户名中不能含有666 只是校验username字段 局部钩子
# 2.校验密码和确认密码是否一致 password confirm两个字段 全局钩子
urls.py
from django.conf.urls import url
from django.contrib import admin
from app01 import views
urlpatterns = [
url(r'^admin/', admin.site.urls),
# forms组件
url(r'^ab_form/', views.ab_form),
]
urls.py
views.py
from django.shortcuts import render,HttpResponse
# 基本使用
from django import forms # forms组件所需模块
class MyForms(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(max_length=8,min_length=3,label="用户名",
error_messages={
'min_length':'用户名最少3位',
'max_length':'用户名最大8位',
'required':"用户名不能为空"
}
)
# password字符串类型最小3位最大8位
password = forms.CharField(max_length=8,min_length=3,label="密码",
error_messages = {
'min_length': '密码最少3位',
'max_length': '密码最大8位',
'required': "密码不能为空"
}
)
re_password = forms.CharField(max_length=8, min_length=3, label="确认密码",
error_messages={
'min_length': '确认密码最少3位',
'max_length': '确认密码最大8位',
'required': "确认密码不能为空"
}
)
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField(label="邮箱",
error_messages={
'invalid': '邮箱格式不正确',
'required': "邮箱不能为空"
}
)
# 钩子函数
# 局部钩子
def clean_username(self): # 用来校验username,走这一步前提是上面username字段条件满足了
# 获取到用户名
username = self.cleaned_data.get("username")
if '666' in username:
# 提示前端展示报错信息
self.add_error("username","用户名不合法") # 添加一个错误信息,第一个参数是给哪个字段增加错误信息,第二个参数是错误信息内容
# 将钩子函数钩取到的数据再放回去
return username
# 全局钩子
def clean(self): # 全局钩子
password = self.cleaned_data.get("password")
re_password = self.cleaned_data.get("re_password")
if not password == re_password:
self.add_error("re_password","两次密码不一致")
# 将钩子函数钩取的数据返回
return self.cleaned_data
# 校验数据
def ab_form(request):
# 1 先产生一个空对象
form_obj = MyForms()
if request.method == "POST":
# 获取用户数据
"""
1.数据获取繁琐
2.校验数据需要构造成字典的格式传入才行
ps:但是request.POST可以看成就是一个字典
"""
# 3.校验数据
form_obj = MyForms(request.POST)
# 4.判断数据是否合法
if form_obj.is_valid():
# 5.如果合法 操作数据库存储数据
return HttpResponse('OK')
# 5.不合法 有错误
return render(request,"ab_form.html",locals())
ab_form.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<!-- Bootstrap3 核心 CSS 文件 -->
<link href="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet">
<!-- jQuery文件。务必在bootstrap.min.js 之前引入 -->
<script src="https://cdn.staticfile.org/jquery/2.1.1/jquery.min.js"></script>
<!-- Bootstrap3 核心 JavaScript 文件 -->
<script src="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/js/bootstrap.min.js"></script>
<!-- font-awesome.min.css图标库4.7版本 -->
<link href="https://cdn.bootcss.com/font-awesome/4.7.0/css/font-awesome.min.css" rel="stylesheet">
</head>
<body>
<div class="container-fluid">
<div class="row">
<div class="col-md-8 col-md-offset-2">
<form action="" method="post" novalidate>
{% for form in form_obj %}
<p>
{{ form.label }}:{{ form }}
<span style="color: red">{{ form.errors.0 }}</span>
</p>
{% endfor %}
<input type="submit" class="btn btn-info">
</form>
</div>
</div>
</div>
</body>
</html>
效果:
forms组件字段的其他参数及补充知识点
forms类中常见字段
max_length # 允许输入的最大长度
min_length # 最小长度
label # input的提示信息,name、password等
error_messages # 自定义报错的提示信息
initial # 设置默认值
required # 控制字段是否必填
widget # 控制type类型及属性
字段样式以及正则参数:
"""
1.字段没有样式
2.针对不同类型的input如何修改
text
password
date
radio
checkbox
...
"""
# 可以通过为字段添加wiget参数进行设置,属性多个值用空格隔开
widget=forms.widgets.标签类型(attrs={'class':'form-control c1 c2'})
# 多个属性值的话 直接空格隔开即可
# 字段里面还有支持正则校验的参数,注意要导入正则匹配的组件
from django.core.validators import RegexValidator # 导入正则匹配组件
validators=[
RegexValidator(r'^[0-9]+$', '请输入数字'),
RegexValidator(r'^159[0-9]+$', '数字必须以159开头')
]
案例:提示报错信息时,文本框也会变红
views.py
from django.shortcuts import render,HttpResponse
from app01 import models
# 基本使用
from django import forms # forms组件所需模块
from django.core.validators import RegexValidator # 导入正则匹配的组件
class MyForms(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(max_length=8,min_length=3,label="用户名",
error_messages={
'min_length':'用户名最少3位',
'max_length':'用户名最大8位',
'required':"用户名不能为空"
},
widget=forms.widgets.TextInput(attrs={'class':'form-control'})
)
# password字符串类型最小3位最大8位
password = forms.CharField(max_length=8,min_length=3,label="密码",
error_messages = {
'min_length': '密码最少3位',
'max_length': '密码最大8位',
'required': "密码不能为空"
},
widget=forms.widgets.PasswordInput(attrs={'class':'form-control'})
)
re_password = forms.CharField(max_length=8, min_length=3, label="确认密码",
error_messages={
'min_length': '确认密码最少3位',
'max_length': '确认密码最大8位',
'required': "确认密码不能为空"
},
widget=forms.widgets.PasswordInput(attrs={'class':'form-control'})
)
phone =forms.CharField(max_length=11, min_length=11, label="手机号",
error_messages={
'min_length': '手机号最少11位',
'max_length': '手机号最大11位',
'required': "手机号不能为空"
},
widget=forms.widgets.TextInput(attrs={'class':'form-control'}),
validators=[
RegexValidator(r'^[0-9]+$', '请输入数字'),
RegexValidator(r'^159[0-9]+$', '数字必须以159开头')
]
)
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField(label="邮箱",
error_messages={
'invalid': '邮箱格式不正确',
'required': "邮箱不能为空"
},
widget=forms.widgets.EmailInput(attrs={'class':'form-control'})
)
# 钩子函数
# 局部钩子
def clean_username(self): # 用来校验username,走这一步前提是上面username字段条件满足了
# 获取到用户名
username = self.cleaned_data.get("username")
# 判断用户是否存在
user_obj = models.User.objects.filter(username=username).first()
if user_obj:
self.add_error("username","用户已存在")
# if '666' in username:
# 提示前端展示报错信息
# 方法一:add_error()
# self.add_error("username","用户名不合法") # 添加一个错误信息,第一个参数是给哪个字段增加错误信息,第二个参数是错误信息内容
# 方法二:主动抛出异常raise 需要导入ValidationError, from django.core.exceptions import ValidationError
# raise ValidationError("用户名不合法")
# 将钩子函数钩取到的数据再放回去
return username
# 全局钩子
def clean(self): # 全局钩子
password = self.cleaned_data.get("password")
re_password = self.cleaned_data.get("re_password")
if not password == re_password:
self.add_error("re_password","两次密码不一致")
# 将钩子函数钩取的数据返回
return self.cleaned_data
# 校验数据
def ab_form(request):
# 1 先产生一个空对象
form_obj = MyForms()
if request.method == "POST":
# 获取用户数据
"""
1.数据获取繁琐
2.校验数据需要构造成字典的格式传入才行
ps:但是request.POST可以看成就是一个字典
"""
# 3.校验数据
form_obj = MyForms(request.POST)
# 4.判断数据是否合法
if form_obj.is_valid():
# 5.如果合法 操作数据库存储数据
return HttpResponse('OK')
# 5.不合法 有错误
return render(request,"ab_form.html",locals())
views.py
ab_form.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<!-- Bootstrap3 核心 CSS 文件 -->
<link href="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/css/bootstrap.min.css" rel="stylesheet">
<!-- jQuery文件。务必在bootstrap.min.js 之前引入 -->
<script src="https://cdn.staticfile.org/jquery/2.1.1/jquery.min.js"></script>
<!-- Bootstrap3 核心 JavaScript 文件 -->
<script src="https://cdn.staticfile.org/twitter-bootstrap/3.3.7/js/bootstrap.min.js"></script>
<!-- font-awesome.min.css图标库4.7版本 -->
<link href="https://cdn.bootcss.com/font-awesome/4.7.0/css/font-awesome.min.css" rel="stylesheet">
</head>
<body>
<div class="container-fluid">
<div class="row">
<div class="col-md-8 col-md-offset-2">
<form action="" method="post" novalidate>
{% for form in form_obj %}
<p>
{{ form.label }}:{{ form }}
<span style="color: red" class="input_span">{{ form.errors.0 }}</span>
</p>
{% endfor %}
<p>
性别:<input type="radio" name="gender" value="1" checked> 男
<input type="radio" name="gender" value="2"> 女
<input type="radio" name="gender" value="3"> 保密
</p>
<input type="submit" class="btn btn-info btn-block" id="btn">
</form>
</div>
</div>
</div>
<script>
$("p span").each(function(){
if($(this).text()){
$(this).prev().css("border","solid red")
}
});
</script>
</body>
</html>
ab_form.html
效果
其他类型渲染
# radio 单选按钮
gender = forms.ChoiceField(
choices=((1, "男"), (2, "女"), (3, "保密")),
label="性别",
initial=3,
widget=forms.widgets.RadioSelect()
)
# select
hobby = forms.ChoiceField(
choices=((1, "篮球"), (2, "足球"), (3, "双色球"),),
label="爱好",
initial=3,
widget=forms.widgets.Select()
)
# 多选
hobby1 = forms.MultipleChoiceField(
choices=((1, "篮球"), (2, "足球"), (3, "双色球"),),
label="爱好",
initial=[1, 3],
widget=forms.widgets.SelectMultiple()
)
# 单选checkbox
keep = forms.ChoiceField(
label="是否记住密码",
initial="checked",
widget=forms.widgets.CheckboxInput()
)
# 多选checkbox
hobby2 = forms.MultipleChoiceField(
choices=((1, "篮球"), (2, "足球"), (3, "双色球"),),
label="爱好",
initial=[1, 3],
widget=forms.widgets.CheckboxSelectMultiple()
)
forms组件源码
"""
切入点:
form_obj.is_valid()
"""
def is_valid(self):
"""
Returns True if the form has no errors. Otherwise, False. If errors are
being ignored, returns False.
"""
return self.is_bound and not self.errors
# 如果is_valid要返回True的话 那么self.is_bound要为True self.errors要为Flase
self.is_bound = data is not None or files is not None # 只要你传值了肯定为True
@property
def errors(self):
"Returns an ErrorDict for the data provided for the form"
if self._errors is None:
self.full_clean()
return self._errors
# forms组件所有的功能基本都出自于该方法
def full_clean(self):
self._clean_fields() # 校验字段 + 局部钩子
self._clean_form() # 全局钩子
self._post_clean()
源码大体执行流程
# forms组件源码执行流程
form.is_valid()----》内部起了一个for循环,先去校验每个字段配置的规则,校验完成,走该字段的局部钩子函数,一个一个执行完(交验完)---》会走全局钩子(clean())--->self就会有clean_data和errors
1 流程:
1 form.is_valid()
2 self.errors
3 self.full_clean()
4 self._clean_fields() 局部字段的校验(自己的和局部钩子)
if hasattr(self, 'clean_%s' % name):
func=getattr(self, 'clean_%s' % name)
value = func()
self.cleaned_data[name] = value
self._clean_form() 全局的钩子
self._post_clean()
看源码知道使用钩子函数要提示前端展示报错信息还有一种通过raise主动抛出一次的方法(针对局部钩子):不过不推荐
示例:(看局部钩子函数部分)
views.py
from django.shortcuts import render,HttpResponse
from django.core.exceptions import ValidationError
# 基本使用
from django import forms # forms组件所需模块
from django.core.validators import RegexValidator # 导入正则匹配的组件
class MyForms(forms.Form):
# username字符串类型最小3位最大8位
username = forms.CharField(max_length=8,min_length=3,label="用户名",
error_messages={
'min_length':'用户名最少3位',
'max_length':'用户名最大8位',
'required':"用户名不能为空"
},
widget=forms.widgets.TextInput(attrs={'class':'form-control'})
)
# password字符串类型最小3位最大8位
password = forms.CharField(max_length=8,min_length=3,label="密码",
error_messages = {
'min_length': '密码最少3位',
'max_length': '密码最大8位',
'required': "密码不能为空"
},
widget=forms.widgets.PasswordInput(attrs={'class':'form-control'})
)
re_password = forms.CharField(max_length=8, min_length=3, label="确认密码",
error_messages={
'min_length': '确认密码最少3位',
'max_length': '确认密码最大8位',
'required': "确认密码不能为空"
},
widget=forms.widgets.PasswordInput(attrs={'class':'form-control'})
)
phone =forms.CharField(max_length=11, min_length=11, label="手机号",
error_messages={
'min_length': '手机号最少11位',
'max_length': '手机号最大11位',
'required': "手机号不能为空"
},
widget=forms.widgets.TextInput(attrs={'class':'form-control'}),
validators=[
RegexValidator(r'^[0-9]+$', '请输入数字'),
RegexValidator(r'^159[0-9]+$', '数字必须以159开头')
]
)
# email字段必须符合邮箱格式 xxx@xx.com
email = forms.EmailField(label="邮箱",
error_messages={
'invalid': '邮箱格式不正确',
'required': "邮箱不能为空"
},
widget=forms.widgets.EmailInput(attrs={'class':'form-control'})
)
# 钩子函数
# 局部钩子
def clean_username(self): # 用来校验username,走这一步前提是上面username字段条件满足了
# 获取到用户名
username = self.cleaned_data.get("username")
if '666' in username:
# 提示前端展示报错信息
# 方法一:add_error(),推荐使用
# self.add_error("username","用户名不合法") # 添加一个错误信息,第一个参数是给哪个字段增加错误信息,第二个参数是错误信息内容
# 方法二:(不推荐使用)主动抛出异常raise 需要导入ValidationError, from django.core.exceptions import ValidationError
raise ValidationError("用户名不合法")
# 将钩子函数钩取到的数据再放回去
return username
# 全局钩子
def clean(self): # 全局钩子
password = self.cleaned_data.get("password")
re_password = self.cleaned_data.get("re_password")
if not password == re_password:
self.add_error("re_password","两次密码不一致")
# 将钩子函数钩取的数据返回
return self.cleaned_data
# 校验数据
def ab_form(request):
# 1 先产生一个空对象
form_obj = MyForms()
if request.method == "POST":
# 获取用户数据
"""
1.数据获取繁琐
2.校验数据需要构造成字典的格式传入才行
ps:但是request.POST可以看成就是一个字典
"""
print(request.POST)
# 3.校验数据
form_obj = MyForms(request.POST)
# 4.判断数据是否合法
if form_obj.is_valid():
# 5.如果合法 操作数据库存储数据
return HttpResponse('OK')
# 5.不合法 有错误
return render(request,"ab_form.html",locals())