Django教程(三)- Django表单Form

目录:

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'邮箱'}))
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,732评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,496评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,264评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,807评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,806评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,675评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,029评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,683评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 41,704评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,666评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,773评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,413评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,016评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,204评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,083评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,503评论 2 343

推荐阅读更多精彩内容