第六章、forms组件

一、注册功能手写

要求

  1. 1.注册功能
  2. 用户输入的用户名中 不能包含西游记
  3. 如果包含了 就提示用户 输入的内容不符合社会主义核心价值观
  4. 用户输入的密码 不能小于三位
  5. 如果密码少于三位 提示用户 密码太短了

说明(******)


  1. 校验数据通常是前后端都有校验
  2. 但是前端校验可有可无 哪怕再牛逼
  3. 后端也必须要有校验 反正一句话 前端可有不校验 后端必须校验!!!

分析

  1. 1.搭建前端页面 >>> 渲染页面
  2. 2.获取前端用户提交的数据校验 >>> 校验数据
  3. 3.对数据的校验的结果 展示到前端页面给用户查看 >>> 展示错误信息

前端代码

  1. #reg.html部分代码
  2. <form action="" method="post">
  3. <p>username:
  4. <input type="text" name="username">
  5. <span style="color: red">{{ back_dic.username }}</span>
  6. </p>
  7. <p>password:<input type="text" name="password">
  8. <span style="color: red">{{ back_dic.password }}</span>
  9. </p>
  10. <input type="submit">
  11. </form>

后端代码

  1. def reg(request):
  2. back_dic = {"username":'','password':''}
  3. if request.method == 'POST':
  4. username = request.POST.get('username')
  5. password = request.POST.get('password')
  6. if '西游记' in username:
  7. # 提示信息
  8. back_dic['username'] = '不符合社会主义核心价值观'
  9. if len(password) < 3:
  10. # 提示信息
  11. back_dic['password'] = '密码不能少于三位'
  12. return render(request,'reg.html',locals())

二、forms组件完整写法

写法和写模型表类极其相似,但是 forms 组件的字段有约束,模型表类的字段没有约束

  1. from django import forms
  2. from django.forms import widgets # 当你forms点不出来的时候 你可以手动导入一下
  3. from django.core.validators import RegexValidator
  4. class MyRegForm(forms.Form):#写一个继承了 forms.Form 的类
  5. username = forms.CharField(min_length=3,max_length=8,label='用户名', # 用户名最长八位最短三位
  6. error_messages={
  7. 'min_length':'用户名最短三位',
  8. 'max_length':'用户名最长八位',
  9. 'required':'用户名不能为空'
  10. },initial='我是初始值',required=False,#允许不填
  11. widget= widgets.TextInput(attrs={'class':'form-control others'})
  12. )
  13. password = forms.CharField(min_length=3,max_length=8,label='密码',error_messages={
  14. 'min_length':'密码最短三位',
  15. 'max_length':'密码最长八位',
  16. 'required':'密码不能为空'
  17. },widget=widgets.PasswordInput())
  18. confirm_password = forms.CharField(min_length=3, max_length=8, label='确认密码', error_messages={
  19. 'min_length': '确认密码最短三位',
  20. 'max_length': '确认密码最长八位',
  21. 'required': '确认密码不能为空'
  22. })
  23. email = forms.EmailField(label='邮箱',error_messages={
  24. 'required':'邮箱不能为空',
  25. 'invalid':'邮箱格式不正确'
  26. },required=False,widget=widgets.EmailInput(attrs={'class':'form-control'}))

基本使用

  1. 将需要校验的数据,以字典(一般是直接 request.POST 当参数(是一个大字典))的方式传递给自定义的类,实例化产生对象

    1. form_obj = views.LoginForm({'username':'jason','password':'123','email':'123'})
  2. 如何查看数据是否全部合法

    1. form_obj.is_valid() ,只有所有的数据都符合要求 才会是True
  3. 如何查看错误原因

    1. form_obj.errors
    2. 结果:
    3. 'password': ['Ensure this value has at least 5 characters (it has 3).'],
    4. 'email': ['Enter a valid email address.']
  4. 如何查看校验通过的数据

    1. form_obj.cleaned_data
    2. 结果:
    3. {'username': 'jason'}

三、forms组件前端渲染标签组件

说明:forms组件只会帮你渲染获取用户输入(输入,选择,下拉框...)的标签 提交按钮需要你自己手动写

前提步骤:

生成一个空对象,并将对象传递给前端页面 form_obj = MyForm(),然后选一种方式渲染

  1. #后端
  2. def formmm(request):
  3. # 1.生成一个空的对象
  4. form_obj = MyRegForm()
  5. if request.method == 'POST':
  6. # 2.获取用户提交的数据
  7. # print(request.POST) # request.POST 其实也可以看成是一个字典
  8. # 3.借助于form组件帮助我们校验
  9. form_obj = MyRegForm(request.POST) # 由于request.POST其实就是一个大字典 所以直接当做参数传入即可
  10. # 4.判断用户输入的数据是否符合校验规则
  11. if form_obj.is_valid():
  12. return HttpResponse('你上传的数据没有问题')
  13. return render(request,'formmm.html',locals())

渲染页面(推荐第三种)

  1. <p>第一种渲染页面的方式(封装程度太高 一般只用于本地测试 通常不适用)</p>
  2. {{ form_obj.as_p }}
  3. {{ form_obj.as_ul }}
  4. {{ form_obj.as_table }}
  5. <p>第二种渲染页面的方式(可扩展性较高 一个一个字段地写,书写麻烦)</p>
  6. <p>{{ form_obj.username.label }}{{ form_obj.username }}</p>
  7. <p>{{ form_obj.password.label }}{{ form_obj.password }}</p>
  8. <p>{{ form_obj.email.label }}{{ form_obj.email }}</p>
  9. <p>第三种渲染页面的方式(推荐,无论有多少个字段都只需要这几个)</p>
  10. {% for form_field in form_obj %}
  11. <p>{{ form_field.label }}{{ form_field }}</p>
  12. {% endfor %}

展示错误信息

  1. 默认是 ul li ,可能有多个报错,我们索引取 0,取消它的 ul li 标签嵌套
  1. <p>第三种渲染前端页面的方式:代码量和扩展性都很高(推荐使用)</p>
  2. <form action="" method="post">
  3. {% for foo in form_obj %}
  4. <p>
  5. {{ foo.label }}:{{ foo }}
  6. <span style="color: red">{{ foo.errors.0 }}</span>
  7. </p>
  8. {% endfor %}
  9. <input type="submit">
  10. </form>

取消展示错误信息

  1. 只要加一参数novalidate
  2. <form action="" method="post" novalidate>

注意事项

  • forms 组件在帮你渲染页面的时候 只会渲染获取用户输入的标签 提交按钮需要你手动添加
  • input 框的 label 注释 不指定的情况下 默认用的是类中字段,并且首字母大写

三、forms组件其他知识点

在python console测试

类似于django的测试环境

校验数据

form对象实例化参数是一个大字典

  1. from app01 import views
  2. # 1.给自定义的类传一个字典
  3. obj = views.MyRegForm({'username':'jason','password':'12','email':'123'})
  4. # 2.判断数据是否全部合法
  5. obj.is_valid() # 只有数据全部符合要求才会是True
  6. Out[4]: False
  7. # 3.查看符合校验规则的数据
  8. obj.cleaned_data
  9. Out[5]: {'username': 'jason'}
  10. # 4.查看不符合条件的数据以及不符合的原因是什么
  11. obj.errors
  12. Out[6]:
  13. {
  14. 'password': ['Ensure this value has at least 3 characters (it has 2).'],
  15. 'email': ['Enter a valid email address.']
  16. }
  17. # 5.校验数据的时候 默认情况下类里面所有的字段都必须传值
  18. obj = views.MyRegForm({'username':'jason','password':'123'})
  19. obj.is_valid()
  20. Out[12]: False
  21. obj.errors
  22. Out[13]: {'email': ['This field is required.']}
  23. # 6.默认情况下可以多传 但是绝对不能少传
  24. obj = views.MyRegForm({'username':'jason','password':'1233','email':'123@qq.com','xxx':'ooo'})
  25. obj.is_valid()
  26. Out[15]: True

form组件数据校验的规则

  • 从上往下依次取值校验
  • 校验通过的放到 cleaned_data
  • 校验失败的放到 errors

注意:

  1. form 中所有的字段默认都是必须传值的(默认:required=True
  2. 只要有一个字段不通过,form_obj.is_valid() 的结果就是 False,所有字段都校验通过了才会返回 True
  3. 校验数据的时候可以多传(多传的数据不会做任何的校验 >> > 不会影响 form 校验规则(所以 可以传 request.POST 直接当做参数,多余的不管))

四、其他几个常见字段类型

  1. # 单选的radio框
  2. gender = forms.ChoiceField(
  3. choices=((1, "男"), (2, "女"), (3, "保密")),
  4. label="性别",
  5. initial=3,
  6. widget=forms.widgets.RadioSelect()
  7. )
  8. # 单选select
  9. hobby = forms.ChoiceField(
  10. choices=((1, "篮球"), (2, "足球"), (3, "双色球"),),
  11. label="爱好",
  12. initial=3,
  13. widget=forms.widgets.Select()
  14. )
  15. # 多选的select框
  16. hobby1 = forms.MultipleChoiceField(
  17. choices=((1, "篮球"), (2, "足球"), (3, "双色球"),),
  18. label="爱好",
  19. initial=[1, 3],
  20. widget=forms.widgets.SelectMultiple()
  21. )
  22. # 单选的checkbox
  23. keep = forms.ChoiceField(
  24. label="是否记住密码",
  25. initial="checked",
  26. widget=forms.widgets.CheckboxInput()
  27. )
  28. # 多选的checkbox
  29. hobby2 = forms.MultipleChoiceField(
  30. choices=((1, "篮球"), (2, "足球"), (3, "双色球"),),
  31. label="爱好",
  32. initial=[1, 3],
  33. widget=forms.widgets.CheckboxSelectMultiple()
  34. )
  35. phone = forms.CharField(
  36. validators=[RegexValidator(r'^[0-9]+$', '请输入数字'), RegexValidator(r'^159[0-9]+$', '数字必须以159开头')],
  37. )

五、forms组件所有内置字段类型

  1. Field
  2. required=True, 是否允许为空
  3. widget=None, HTML插件
  4. label=None, 用于生成Label标签或显示内容
  5. initial=None, 初始值
  6. help_text='', 帮助信息(在标签旁边显示)
  7. error_messages=None, 错误信息 {'required': '不能为空', 'invalid': '格式错误'}
  8. validators=[], 自定义验证规则
  9. localize=False, 是否支持本地化
  10. disabled=False, 是否可以编辑
  11. label_suffix=None Label内容后缀
  12. CharField(Field)
  13. max_length=None, 最大长度
  14. min_length=None, 最小长度
  15. strip=True 是否移除用户输入空白
  16. IntegerField(Field)
  17. max_value=None, 最大值
  18. min_value=None, 最小值
  19. FloatField(IntegerField)
  20. ...
  21. DecimalField(IntegerField)
  22. max_value=None, 最大值
  23. min_value=None, 最小值
  24. max_digits=None, 总长度
  25. decimal_places=None, 小数位长度
  26. BaseTemporalField(Field)
  27. input_formats=None 时间格式化
  28. DateField(BaseTemporalField) 格式:2015-09-01
  29. TimeField(BaseTemporalField) 格式:11:12
  30. DateTimeField(BaseTemporalField)格式:2015-09-01 11:12
  31. DurationField(Field) 时间间隔:%d %H:%M:%S.%f
  32. ...
  33. RegexField(CharField)
  34. regex, 自定制正则表达式
  35. max_length=None, 最大长度
  36. min_length=None, 最小长度
  37. error_message=None, 忽略,错误信息使用 error_messages={'invalid': '...'}
  38. EmailField(CharField)
  39. ...
  40. FileField(Field)
  41. allow_empty_file=False 是否允许空文件
  42. ImageField(FileField)
  43. ...
  44. 注:需要PIL模块,pip3 install Pillow
  45. 以上两个字典使用时,需要注意两点:
  46. - form表单中 enctype="multipart/form-data"
  47. - view函数中 obj = MyForm(request.POST, request.FILES)
  48. URLField(Field)
  49. ...
  50. BooleanField(Field)
  51. ...
  52. NullBooleanField(BooleanField)
  53. ...
  54. ChoiceField(Field)
  55. ...
  56. choices=(), 选项,如:choices = ((0,'上海'),(1,'北京'),)
  57. required=True, 是否必填
  58. widget=None, 插件,默认select插件
  59. label=None, Label内容
  60. initial=None, 初始值
  61. help_text='', 帮助提示
  62. ModelChoiceField(ChoiceField)
  63. ... django.forms.models.ModelChoiceField
  64. queryset, # 查询数据库中的数据
  65. empty_label="---------", # 默认空显示内容
  66. to_field_name=None, # HTML中value的值对应的字段
  67. limit_choices_to=None # ModelForm中对queryset二次筛选
  68. ModelMultipleChoiceField(ModelChoiceField)
  69. ... django.forms.models.ModelMultipleChoiceField
  70. TypedChoiceField(ChoiceField)
  71. coerce = lambda val: val 对选中的值进行一次转换
  72. empty_value= '' 空值的默认值
  73. MultipleChoiceField(ChoiceField)
  74. ...
  75. TypedMultipleChoiceField(MultipleChoiceField)
  76. coerce = lambda val: val 对选中的每一个值进行一次转换
  77. empty_value= '' 空值的默认值
  78. ComboField(Field)
  79. fields=() 使用多个验证,如下:即验证最大长度20,又验证邮箱格式
  80. fields.ComboField(fields=[fields.CharField(max_length=20), fields.EmailField(),])
  81. MultiValueField(Field)
  82. PS: 抽象类,子类中可以实现聚合多个字典去匹配一个值,要配合MultiWidget使用
  83. SplitDateTimeField(MultiValueField)
  84. input_date_formats=None, 格式列表:['%Y--%m--%d', '%m%d/%Y', '%m/%d/%y']
  85. input_time_formats=None 格式列表:['%H:%M:%S', '%H:%M:%S.%f', '%H:%M']
  86. FilePathField(ChoiceField) 文件选项,目录下文件显示在页面中
  87. path, 文件夹路径
  88. match=None, 正则匹配
  89. recursive=False, 递归下面的文件夹
  90. allow_files=True, 允许文件
  91. allow_folders=False, 允许文件夹
  92. required=True,
  93. widget=None,
  94. label=None,
  95. initial=None,
  96. help_text=''
  97. GenericIPAddressField
  98. protocol='both', both,ipv4,ipv6支持的IP格式
  99. unpack_ipv4=False 解析ipv4地址,如果是::ffff:192.0.2.1时候,可解析为192.0.2.1 PSprotocol必须为both才能启用
  100. SlugField(CharField) 数字,字母,下划线,减号(连字符)
  101. ...
  102. UUIDField(CharField) uuid类型

六、forms组件的字段常见参数

  1. # 基本都渲染成 HTML 标签的属性(H5 对这些属性有支持)
  2. max_length 最大长度
  3. min_length 最小长度
  4. required 是否必填
  5. label 注释信息
  6. initial 初始值(对应的是 value
  7. error_messages 报错信息
  8. widget 控制标签的属性和样式(要手动导入,如果 forms.widget 点不出来 Widget,可以先导入 from django.forms import widget(有时候点不出来就需要自己写))
  9. validators 正则(要手动导入,相当于钩子函数)
  10. 。。。

用正则来约束

  1. #每个字段 还支持正则校验
  2. from django import forms
  3. from django.forms import Form
  4. from django.core.validators import RegexValidator
  5. class MyForm(Form):
  6. user = forms.CharField(
  7. validators=[RegexValidator(r'^[0-9]+$', '请输入数字'), RegexValidator(r'^159[0-9]+$', '数字必须以159开头')],
  8. )

七、forms组件钩子函数

含义:对通过基本校验(包含正则校验)的 forms 字段再进行额外的自定义校验(定制化业务需求)

  1. 基本步骤

    • 在 cleaned_data 里面获取字段 self.cleaned_data.get('字段名')
    • 对获取的字段进行自定义校验 if not password == confirm_password:
    • 不符合的 self.add_error('字段名', '报错提示信息')
    • return 一个返回值 return self.cleaned_data/username...
  2. 局部钩子

    1. # 当你需要对某一个字段数据进行额外的一些列校验 你可以考虑使用钩子函数
    2. # 针对单个字段的 使用局部钩子
    3. #函数名:clean_formsField
    4. def clean_username(self):
    5. username = self.cleaned_data.get('username')
    6. if '西游记' in username:
    7. # 给username字段下面提示错误信息
    8. self.add_error('username','用户名不符合社会主义核心价值观')
    9. return username
    10. #针对某一个字段作额外校验(一定要返回该字段的值)
  3. 全局钩子

    1. # 针对多个字段的校验 使用全局钩子 eg:校验两次密码是否一致
    2. def clean(self):
    3. password = self.cleaned_data.get('password')
    4. confirm_password = self.cleaned_data.get('confirm_password')
    5. if not password == confirm_password:
    6. self.add_error('confirm_password','两次密码不一致')
    7. return self.cleaned_data

八、通过后端修改前端的样式

  1. 如何改变input框的type属性值
  2. widget= widgets.TextInput()
  3. widget=widgets.PasswordInput()
  1. 如何让forms组件渲染出来的input框有form-control类属性
  2. widget= widgets.TextInput(attrs={'class':'form-control others'}) # 如果有多个类属性 空格隔开
  3. widget=widgets.PasswordInput(attrs={'class':'form-control others'})

说明:详情写法见第二个大纲

第六章、forms组件的更多相关文章

  1. Django组件——forms组件

    一.校验字段功能 通过注册用户这个实例来学习校验字段功能. 1.模型:models.py from django.db import models # Create your models here. ...

  2. KnockoutJS 3.X API 第六章 组件(5) 高级应用组件加载器

    无论何时使用组件绑定或自定义元素注入组件,Knockout都将使用一个或多个组件装载器获取该组件的模板和视图模型. 组件加载器的任务是异步提供任何给定组件名称的模板/视图模型对. 本节目录 默认组件加 ...

  3. ASP.NET2.0自定义控件组件开发 第六章 深入讲解控件的属性

    原文:ASP.NET2.0自定义控件组件开发 第六章 深入讲解控件的属性 深入讲解控件的属性持久化(一) 系列文章链接: ASP.NET自定义控件组件开发 第一章 待续 ASP.NET自定义控件组件开 ...

  4. 第六章、ajax方法以及序列化组件

    目录 第六章.ajax方法 一.choice参数介绍 二.MTV与MVC模型 三.ajax方法 四.案例 五.Ajax传json格式的数据 六. AJAX传文件 代码如下 ajax传文件需要注意的事项 ...

  5. Flask 教程 第十六章:全文搜索

    本文翻译自The Flask Mega-Tutorial Part XVI: Full-Text Search 这是Flask Mega-Tutorial系列的第十六部分,我将在其中为Microblo ...

  6. 第十六章 综合实例——《跟我学Shiro》

    简单的实体关系图 简单数据字典 用户(sys_user) 名称 类型 长度 描述 id bigint 编号 主键 username varchar 100 用户名 password varchar 1 ...

  7. 第六章 Qt布局管理器Layout

    第六章 Qt布局管理器Layout 大家有没有发现一个现象,我们放置一个组件,给组件最原始的定位是给出这个控件的坐标和宽高值,这样Qt就知道这个组件的位置.当用户改变窗口的大小,组件还静静地呆在原来的 ...

  8. 【windows核心编程】 第六章 线程基础

    Windows核心编程 第六章 线程基础 欢迎转载 转载请注明出处:http://www.cnblogs.com/cuish/p/3145214.html 1. 线程的组成 ①    一个是线程的内核 ...

  9. (转)iOS Wow体验 - 第六章 - 交互模型与创新的产品概念(2)

    本文是<iOS Wow Factor:Apps and UX Design Techniques for iPhone and iPad>第六章译文精选的第二部分,其余章节将陆续放出.上一 ...

随机推荐

  1. React Native使用code-push实现热更新

    这里就不记录了,下面的传送门介绍的通俗易懂,很详细,一步一步很容易实现成功. http://www.jianshu.com/p/f8689ccf0007

  2. golang web框架设计1:框架规划

    GO WEB 编程13节,如何设计一个web框架 学习谢大的web框架设计 总体介绍 实现一个简易的web框架,我们采用mvc模式来进行开发. model:模型,代表数据结构.通常来说,模型类时包含查 ...

  3. XSSer:自动化XSS漏洞检测及利用工具

    转载自FreeBuf.COM XSS是一种非常常见的漏洞类型,它的影响非常的广泛并且很容易的就能被检测到. 攻击者可以在未经验证的情况下,将不受信任的JavaScript片段插入到你的应用程序中,然后 ...

  4. Ubuntu 修改登录界面,使用root用户登录

    修改: sudo vi /etc/pam.d/gdm-autologin注释行 "auth requied pam_succeed_if.so user != root quiet succ ...

  5. SIFT学习笔记之二 特征提取

    特征提取函数: int _sift_features( IplImage* img, struct feature** feat, int intvls, double sigma, double c ...

  6. QEMU编译安装

    QEMU是一个支持跨平台虚拟化的虚拟机,有user mode和system mode两种配置方式.其中qemu在system mode配置下模拟出整个计算机,可以在qemu之上运行一个操作系统.QEM ...

  7. poj2947(高斯消元法解同余方程组)

    题目链接:https://vjudge.net/problem/POJ-2065 题意:题目看着较复杂,实际上就是给了n个同余方程,解n个未知数. 思路:套高斯消元法的模板即可. AC代码: #inc ...

  8. RestTemplate实践(及遇到的问题)

    在微服务都是以HTTP接口的形式暴露自身服务的,因此在调用远程服务时就必须使用HTTP客户端.我们可以使用JDK原生的URLConnection.Apache的Http Client.Netty的异步 ...

  9. Java开发环境不再需要配置classpath!

    前言: 之前发布了关于java开发环境配置的文章,经过与网友的交流,我了解到在jdk1.5以后,java开发环境配置的时候,确实不需要对classpath进行配置,但市面上的书籍,以及一些博客.还是老 ...

  10. 【转帖】深挖NUMA

    深挖NUMA http://www.litrin.net/2017/10/31/深挖numa/ 首先列出本站之前相关的几篇帖子: Linux的NUMA机制 NUMA对性能的影响 cgroup的cpus ...