Django教程(三)- Django表单Form

作者: 一只写程序的猿 | 来源:发表于2017-07-12 22:53 被阅读1471次

    目录:

    1.Form 基本使用

    django中的Form组件有以下几个功能:

    1. 生成HTML标签
    1. 验证用户数据(显示错误信息)
    2. HTML Form提交保留上次提交数据
    3. 初始化页面显示内容

    2.Form中字段及插件

    创建Form类时,主要涉及到 【字段】 和 【插件】,字段用于对用户请求数据的验证,插件用于自动生成HTML;

    1.Django内置字段如下:

    • Field:
    required=True,               是否允许为空
    widget=None,                 HTML插件
    label=None,                  用于生成Label标签或显示内容
    initial=None,                初始值
    help_text='',                帮助信息(在标签旁边显示)
    error_messages=None,         错误信息 {'required': '不能为空', 'invalid': '格式错误'}
    show_hidden_initial=False,   是否在当前插件后面再加一个隐藏的且具有默认值的插件(可用于检验两次输入是否一直)
    validators=[],               自定义验证规则
    localize=False,              是否支持本地化(根据不同语言地区访问用户显示不同语言)
    disabled=False,              是否可以编辑
    label_suffix=None            Label内容后缀
    
    • CharField(Field)
    max_length=None,             最大长度
    min_length=None,             最小长度
    strip=True                   是否移除用户输入空白
    
    • IntegerField(Field), FloatField(IntegerField)
    max_value=None,              最大值
    min_value=None,              最小值
    
    • DecimalField(IntegerField) 小数,举例,涉及金钱计算保留小数点后两位
    max_value=None,              最大值
    min_value=None,              最小值
    max_digits=None,             总长度
    decimal_places=None,         小数位长度
    
    
    • BaseTemporalField(Field)
    input_formats=None          时间格式化
    
    DateField(BaseTemporalField)    格式:2015-09-01
    TimeField(BaseTemporalField)    格式:11:12
    DateTimeField(BaseTemporalField)格式:2015-09-01 11:12
    DurationField(Field)            时间间隔:%d %H:%M:%S.%f
    
    • RegexField(CharField)
    regex,                      自定制正则表达式
    max_length=None,            最大长度
    min_length=None,            最小长度
    error_message=None,         忽略,错误信息使用 error_messages={'invalid': '...'}
    
    

    EmailField(CharField) ...

    • FileField(Field)
    allow_empty_file=False     是否允许空文件
    
    • ImageField(FileField)
    ...
    注:需要PIL模块,pip install Pillow
    以上两个字典使用时,需要注意两点:
    - form表单中 enctype="multipart/form-data"
    - view函数中 obj = MyForm(request.POST, request.FILES)
    

    URLField(Field)...
    BooleanField(Field)...
    NullBooleanField(BooleanField)...

    • ChoiceField(Field)
    choices=(),                选项,如:choices = ((0,'上海'),(1,'北京'),)
    required=True,             是否必填
    widget=None,               插件,默认select插件
    label=None,                Label内容
    initial=None,              初始值
    help_text='',              帮助提示
    
    • TypedChoiceField(ChoiceField)
    coerce = lambda val: val   对选中的值进行一次转换,通过lambda函数实现
    empty_value= ''            空值的默认值
    
    • MultipleChoiceField(ChoiceField)多选框...

    • TypedMultipleChoiceField(MultipleChoiceField)

    coerce = lambda val: val   对选中的每一个值进行一次转换
    empty_value= ''            空值的默认值
    
    • ComboField(Field)
    fields=()                  使用多个验证,如下:即验证最大长度20,又验证邮箱格式
    fields.ComboField(fields=[fields.CharField(max_length=20), fields.EmailField(),])
    
    • MultiValueField(Field): 抽象类,子类中可以实现聚合多个字典去匹配一个值,要配合MultiWidget使用,提供接口,需要自己实现

    • SplitDateTimeField(MultiValueField)

    input_date_formats=None,   格式列表:['%Y--%m--%d', '%m%d/%Y', '%m/%d/%y']
    input_time_formats=None    格式列表:['%H:%M:%S', '%H:%M:%S.%f', '%H:%M']
    
    
    • FilePathField(ChoiceField) 文件选项,目录下文件显示在页面中
    path,                      文件夹路径
    match=None,                正则匹配
    recursive=False,           递归下面的文件夹
    allow_files=True,          允许文件
    allow_folders=False,       允许文件夹
    required=True,
    widget=None,
    label=None,
    initial=None,
    help_text=''
    
    • GenericIPAddressField
    protocol='both',           both,ipv4,ipv6支持的IP格式
    unpack_ipv4=False          解析ipv4地址,如果是::ffff:192.0.2.1时候,可解析为192.0.2.1, PS:protocol必须为both才能启用
    
    • SlugField(CharField) :数字,字母,下划线,减号(连字符)

    • UUIDField(CharField) :uuid类型

    import uuid
    
    # make a UUID based on the host ID and current time
    >>> uuid.uuid1()    # doctest: +SKIP
    UUID('a8098c1a-f86e-11da-bd1a-00112444be1e')
    
    # make a UUID using an MD5 hash of a namespace UUID and a name
    >>> uuid.uuid3(uuid.NAMESPACE_DNS, 'python.org')
    UUID('6fa459ea-ee8a-3ca4-894e-db77e160355e')
    
    # make a random UUID
    >>> uuid.uuid4()    # doctest: +SKIP
    UUID('16fd2706-8baf-433b-82eb-8c7fada847da')
    
    # make a UUID using a SHA-1 hash of a namespace UUID and a name
    >>> uuid.uuid5(uuid.NAMESPACE_DNS, 'python.org')
    UUID('886313e1-3b8a-5372-9b90-0c9aee199e5d')
    
    # make a UUID from a string of hex digits (braces and hyphens ignored)
    >>> x = uuid.UUID('{00010203-0405-0607-0809-0a0b0c0d0e0f}')
    
    # convert a UUID to a string of hex digits in standard form
    >>> str(x)
    '00010203-0405-0607-0809-0a0b0c0d0e0f'
    
    # get the raw 16 bytes of the UUID
    >>> x.bytes
    b'\x00\x01\x02\x03\x04\x05\x06\x07\x08\t\n\x0b\x0c\r\x0e\x0f'
    
    # make a UUID from a 16-byte string
    >>> uuid.UUID(bytes=x.bytes)
    UUID('00010203-0405-0607-0809-0a0b0c0d0e0f')
    
    • Django内置插件:
    TextInput(Input)  #input type="text"
    NumberInput(TextInput)  # 数字输入框
    EmailInput(TextInput)  # 邮箱输入框
    URLInput(TextInput)  # url输入框
    PasswordInput(TextInput)  # 密码输入框
    HiddenInput(TextInput)  # 隐藏输入框
    Textarea(Widget)  # textarea文本区
    DateInput(DateTimeBaseInput)  # 日期输入框
    DateTimeInput(DateTimeBaseInput)  # 日期时间输入框
    TimeInput(DateTimeBaseInput)  # 时间输入框
    CheckboxInput  # 多选框
    Select  # 下拉框
    NullBooleanSelect  # 非空布尔值下拉框
    SelectMultiple  # 多选下拉框
    RadioSelect  # 单选框
    CheckboxSelectMultiple  # 多选checkbox ???
    FileInput  # 文件上传
    ClearableFileInput
    MultipleHiddenInput  # 多隐藏输入框
    SplitDateTimeWidget  # 时间分割框(两个input框)
    SplitHiddenDateTimeWidget
    SelectDateWidget
    
    • 常用的选择插件
    # 单radio,值为字符串
    # user = fields.CharField(
    #     initial=2,
    #     widget=widgets.RadioSelect(choices=((1,'上海'),(2,'北京'),))
    # )
    
    # 单radio,值为字符串
    # user = fields.ChoiceField(
    #     choices=((1, '上海'), (2, '北京'),),
    #     initial=2,
    #     widget=widgets.RadioSelect
    # )
    
    # 单select,值为字符串
    # user = fields.CharField(
    #     initial=2,
    #     widget=widgets.Select(choices=((1,'上海'),(2,'北京'),))
    # )
    
    # 单select,值为字符串
    # user = fields.ChoiceField(
    #     choices=((1, '上海'), (2, '北京'),),
    #     initial=2,
    #     widget=widgets.Select
    # )
    
    # 多选select,值为列表
    # user = fields.MultipleChoiceField(
    #     choices=((1,'上海'),(2,'北京'),),
    #     initial=[1,],
    #     widget=widgets.SelectMultiple
    # )
    
    
    
    # 单checkbox
    # user = fields.CharField(
    #     widget=widgets.CheckboxInput()
    # )
    
    
    # 多选checkbox,值为列表
    # user = fields.MultipleChoiceField(
    #     initial=[2, ],
    #     choices=((1, '上海'), (2, '北京'),),
    #     widget=widgets.CheckboxSelectMultiple
    # )
    

    Django模版加减乘除:

    Django模版加法:
    {{ value|add:10}}
    value=5,则返回15 Django模版减法:
    {{value|add:-10}}
    value=5,则返回-5,这个比较好理解,减法就是加一个负数 Django模版乘法:
    {%  widthratio 5 1 100 %}
    上面的代码表示:5/1 *100,返回500,widthratio需要三个参数,它会使用 参数1/参数2*参数3,所以要进行乘法的话,就将参数2=1即可 Django模版除法
    view sourceprint?
    {%  widthratio 5 100 1 %}
    上面的代码表示:5/100*1,返回0.05,只需要将第三个参数设置为1即可 
    

    3.通过Django表单Form来完成需求

    1.根据用户填写表单的不同跳往不同的页面

    1.先创建app项目名:djangoform


    树形图

    2.app下创建文件夹djangoform,并建立表单form1.py

    # -*- coding:utf8 -*-
    from django.forms import Form
    from django.forms import widgets  # 插件
    from django.forms import fields # 字段
    
    class webpage(Form):
        page = fields.CharField()
    

    3.app下创建templates文件夹,并创建不同的html网页

    • index.html
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>首页</title>
    </head>
    <body>
    
        <form action="" method="post">
            {% csrf_token %}
    
            请选择要进入的页面:{{ web.page }}
            <input type="submit" value="提交">
    
        </form>
    
    </body>
    </html>
    
    • page1.html
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>page1</title>
    </head>
    <body>
    
    
            Page1:一颦一笑一伤悲,一生痴迷一世醉.
    
    
    </body>
    </html>
    
    • page2.html
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>page2</title>
    </head>
    <body>
    
            Page2:一嗟一叹一轮回,一寸相思一寸灰.
    
    
    </body>
    </html>
    

    其他几个网页类似
    4.建立视图views.py

    # -*- coding: utf-8 -*-
    from __future__ import unicode_literals
    
    from django.shortcuts import render,redirect
    from django.http import HttpResponse
    from djangoform import form1
    
    # Create your views here.
    
    
    def indexPage(request):
        if request.method == "GET":
            webPage=form1.webpage()
            return render(request,'index.html',{'web':webPage})
    
        elif request.method == "POST":
            webPage = form1.webpage(request.POST,request.FILES)
            if webPage.is_valid():
                values = webPage.clean()
                print(values)
                if values['page'] == '1':
                    return render(request, 'page1.html', {'web': webPage})
                elif values['page']== '2':
                    return render(request, 'page2.html', {'web': webPage})
                elif values['page']== '3':
                    return render(request, 'page3.html', {'web': webPage})
            else:
                errors = webPage.errors
                print(errors)
            return render(request, 'index.html', {'web': webPage})
        else:
            return redirect('http://www.baidu.com')
    
    
    
    
    def index(request):
        if request.method == "GET":
            obj = forms.MyForm()  # 没有值,在页面上渲染form中的标签
            return render(request, 'index.html', {'form': obj})
    
        elif request.method == "POST":
            obj = forms.MyForm(request.POST, request.FILES)  # 将post提交过来的数据作为参数传递给自定义的Form类
            if obj.is_valid():  # obj.is_valid()返回一个bool值,如果检查通过返回True,否则返回False
                values = obj.clean()  # 拿到处理后的所有数据,键值对的形式
                print(values)
            else:
                errors = obj.errors  # 拿到未通过的错误信息,里面封装的都是对象
                print(errors)
            return render(request, 'index.html', {'form': obj})
        else:
            return redirect('http://www.baidu.com')
    

    5.定义视图函数相关的·urls.py·

    from django.conf.urls import include, url
    from django.contrib import admin
    from . import views
    
    urlpatterns = [
        url(r'^page/',views.indexPage,),
    ]
    

    6.把我们新定义的app加到settings.py中的INSTALL_APPS中和urls中,详情见Django教程(一)- Django视图与网址

    效果展示:

    首页index
    输入数字1跳转的页面 输入数字2跳转的页面

    2.在网页上打印9*9乘法表

    • home.html
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>九九乘法表</title>
    </head>
    <body>
        {% for i in list %}
            {% for j in list %}
                {% if j <= i %}
                    {{i}}*{{j}}={% widthratio j 1 i %}
                {% endif %}
            {% endfor %}<br>
        {% endfor %}
    
    
    
    </body>
    </html>
    
    • views.py
    # -*- coding: utf-8 -*-
    from __future__ import unicode_literals
    
    from django.shortcuts import render
    
    # Create your views here.
    
    def home(request):
        list= [1,2,3,4,5,6,7,8,9,]
        return render(request,'home.html',{'list':list})
    
    • urls.py
    from django.conf.urls import url
    from . import views
    
    urlpatterns=[
        url(r'^home/$',views.home,name='home',)
    ]
    
    

    效果展示:

    九九乘法表

    3.在网页上打印1-100之间的偶数

    先了解下python中map函数

    >>> map(str, range(5))           #对range(5)各项进行str操作
    ['0', '1', '2', '3', '4']        #返回列表
    >>> def add(n):return n+n
    ... 
    >>> map(add, range(5))           #对range(5)各项进行add操作
    [0, 2, 4, 6, 8]
    >>> map(lambda x:x+x,range(5))   #lambda 函数,各项+本身
    [0, 2, 4, 6, 8]
    >>> map(lambda x:x+1,range(10))  #lambda 函数,各项+1
    [1, 2, 3, 4, 5, 6, 7, 8, 9, 10]
    >>> map(add,'zhoujy')            
    ['zz', 'hh', 'oo', 'uu', 'jj', 'yy']
    
    #想要输入多个序列,需要支持多个参数的函数,注意的是各序列的长度必须一样,否则报错:
    >>> def add(x,y):return x+y
    ... 
    >>> map(add,'zhoujy','Python')
    ['zP', 'hy', 'ot', 'uh', 'jo', 'yn']
    >>> def add(x,y,z):return x+y+z
    ... 
    >>> map(add,'zhoujy','Python','test')     #'test'的长度比其他2个小
    Traceback (most recent call last):
      File "<stdin>", line 1, in <module>
    TypeError: add() takes exactly 2 arguments (3 given)
    
    >>> map(add,'zhoujy','Python','testop')
    ['zPt', 'hye', 'ots', 'uht', 'joo', 'ynp']
    
    • views.py
    # -*- coding: utf-8 -*-
    from __future__ import unicode_literals
    
    from django.shortcuts import render
    
    # Create your views here.
    
    def even(request):
        list = map(str,range(100))  #对range(100)各项进行str操作
        return render(request,'even.html',{'list':list})
    
    • urls.py
    from django.conf.urls import url
    from . import views
    
    urlpatterns=[
        url(r'^even/$',views.even,name='even',)
    ]
    
    
    • even.html
    <!DOCTYPE html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>1-100之间的偶数</title>
    </head>
    <body>
        {% for item in list %}
        {% if forloop.counter|divisibleby:2 %}{{forloop.counter}} {% if not forloop.last %},{% endif %} {% endif %}
        {% endfor %}
    
    
    
    </body>
    </html>
    

    效果如下:

    在网页上打印1-100之间的偶数

    4.自定义验证验证规则

    • 方式1:在字段中自定义validators设计正则匹配
    from django.forms import Form
    from django.forms import widgets
    from django.forms import fields
    from django.core.validators import RegexValidator
    
    class MyForm(Form):
        user = fields.CharField(
            validators=[RegexValidator(r'^[0-9]+$', '请输入数字'), RegexValidator(r'^159[0-9]+$', '数字必须以159开头')],
        )
    
    • 方式2:自定义规则函数处理数据
    import re
    from django.forms import Form
    from django.forms import widgets
    from django.forms import fields
    from django.core.exceptions import ValidationError
    
    
    # 自定义验证规则
    def mobile_validate(value):
        mobile_re = re.compile(r'^(13[0-9]|15[012356789]|17[678]|18[0-9]|14[57])[0-9]{8}$')
        if not mobile_re.match(value):
            raise ValidationError('手机号码格式错误')
    
    
    class PublishForm(Form):
    
    
        title = fields.CharField(max_length=20,
                                min_length=5,
                                error_messages={'required': '标题不能为空',
                                                'min_length': '标题最少为5个字符',
                                                'max_length': '标题最多为20个字符'},
                                widget=widgets.TextInput(attrs={'class': "form-control",
                                                              'placeholder': '标题5-20个字符'}))
    
    
        # 使用自定义验证规则
        phone = fields.CharField(validators=[mobile_validate, ],
                                error_messages={'required': '手机不能为空'},
                                widget=widgets.TextInput(attrs={'class': "form-control",
                                                              'placeholder': u'手机号码'}))
    
        email = fields.EmailField(required=False,
                                error_messages={'required': u'邮箱不能为空','invalid': u'邮箱格式错误'},
                                widget=widgets.TextInput(attrs={'class': "form-control", 'placeholder': u'邮箱'}))
    

    相关文章

      网友评论

        本文标题:Django教程(三)- Django表单Form

        本文链接:https://www.haomeiwen.com/subject/xpkihxtx.html