Django学习笔记之——Forms

前面学习的都只是如何显示数据,但一直没有关于如何响应用户提交的数据。
表单,是在web中,用户与服务器交互的重要途径。

import django.forms
form django import forms
之间的区别是什么?

form在Django中扮演的角色有:
*显示form
*验证用户提交的数据

form的定义
~~~~~~~~~~

form的定义与model的定义非常相似:
____________________________________________
from django import forms
class BookForm(forms.Form):
    isbn= forms.CharField(max_length=200)
    title = forms.CharField(max_length=200)
--------------------------------------------
定义了一个BookForm表单,表单里有isbn与title两项。

BookForm可以生成html表单文本,如下:
____________________________
book_form = BookForm()
html = book_form.as_table()
print(html)
----------------------------
执行打印出的结果是:
<tr><th><label for="id_isbn">Isbn:</label></th><td><input id="id_isbn" maxlength="50" name="isbn" type="text" /></td></tr>
<tr><th><label for="id_title">Title:</label></th><td><input id="id_title" maxlength="200" name="title" type="text" /></td></tr>

可以看出,打印出来的文本没有包含<form>与<input type="submit">,这些需要我们
在模板中指定。Form只负责isbn与title的输入,而不关心form的动作与提交方式。
所以,我们最好用模板的方法先定义好一个模板,如book-form.html:
_______________________________
<form action="." method="get">
    {{book_form}}
    <br>
    <input type="submit">
</form>
-------------------------------
在views.py中用render_to_response()生成HttpResponse:
____________________________________________________
def book_view(request):
    book_form = BookForm()
    response = render_to_response('book-form.html', 
            {'book_form':book_form.as_table()})
----------------------------------------------------
返回的页面源码:
<form action="." method="get">
    <tr><th><label for="id_isbn">Isbn:</label></th><td><input id="id_isbn" maxlength="50" name="isbn" type="text" /></td></tr>
<tr><th><label for="id_title">Title:</label></th><td><input id="id_title" maxlength="200" name="title" type="text" /></td></tr>
    <br>
    <input type="submit">
</form>

Form除了as_table()方法,还有as_ul()与as_p()方法,默认是as_table()方法。

Form是在 django/forms/forms.py 定义的。
打开forms.py文件,可以了解到,Form继承于BaseForm。
BaseForm的定义大致如下:
___________________________________________________________________________
class BaseForm(object):
    def __init__(self, data=None, files=None, auto_id='id_%s', prefix=None,
            initial=None, label_suffix=None):
        ...
    def as_table(self):
        ...
    def as_ul(self):
        ...
    def as_p(self):
        ...
    def __str__(self):
        return self.as_table()

class Form(BaseForm):
    ...
---------------------------------------------------------------------------
auto_id为<input id="id_xxx">,控件的自动id格式。为''或None表示不需要id。
label_suffix为lable名称的后缀,默认为":",可以改。比如:label_suffix="="
prefix为name的前缀,如果设置了prefix="aaa",那么<input name="aaa-xxx">
initial为初始值字典。

上面BookForm中只用到了一种域:CharField。除此之外还有很多种。
Fields定义在django/forms/fields.py文件里。

打开fields.py文件,可以看到除CharField外还有更多的Field供选择:
Field
|--CharField
|  |--RegexField
|  |--EmailField
|  |--URLField
|  |--IPAddressField
|  |--GenericIPAddressField
|  `--SlugField
|--IntegerField
|  |--FloatField
|  `--DecimalField
|--BaseTomporalField
|  |--DateField
|  |--TimeField
|  `--DateTimeField
|--FileField        #file选择文件
|  `--ImageField
|--BooleanField     #checkbox
|  `--NullBooleanField  #select:Unknow,Yes,No
|--ChoiceField      #select
|  |--TypedChoiceField
|  |--FilePathField
|  `--MultipleChoiceField
|     `--TypedMultipleChoiceField
|--ComboField
`--MultiValueField
   `--SplitDateTimeField

每种Field有个默认的Widget。
__________________________________
class Field(object):
    widget = TextInput
    ...
class EmailField(CharField):
    widget = EmailInput
    ...
class FileField(Field):
    widget = ClearableFileInput
    ...
----------------------------------
Widget告诉Field生成哪种web控件。

我们也可以为Field指定Widget。
比如登陆表单:
_____________________________________________________________
class LoginForm(forms.Form):
    #email = forms.EmailField()
    email = forms.CharField(widget=widgets.EmailInput())
    password = forms.CharField(widget=widgets.PasswordInput())
-------------------------------------------------------------
password这个域不能用明文显示,所以更改了widget。

所有的Widget都定义在django/forms/widgets.py中。
有如下控件:
'Media', 'MediaDefiningClass', 'Widget', 'TextInput',
'EmailInput', 'URLInput', 'NumberInput', 'PasswordInput',
'HiddenInput', 'MultipleHiddenInput', 'ClearableFileInput',
'FileInput', 'DateInput', 'DateTimeInput', 'TimeInput', 'Textarea',
'CheckboxInput', 'SplitDateTimeWidget',
'Select', 'NullBooleanSelect', 'SelectMultiple', 'RadioSelect',
'CheckboxSelectMultiple', 'MultiWidget',

基于模型的表单
~~~~~~~~~~~~~~
根据模型的定义来自动定义表单。如下:
______________________________________
from django import forms
from models import Book
class BookModelForm(froms.ModelForm):
    class Meta:
        model = Book
--------------------------------------
而Book的定义在models.py里:
____________________________________________
class Book(models.Model):
    isbn = models.CharField(max_length=50)
    title = models.CharField(max_length=200)
    author = models.ForeignKey('Author')

    def __unicode__(self):
        return self.title
--------------------------------------------
这样以来,BookModelForm也有了与Book对应的isbn,title,author。

保存ModelForm
~~~~~~~~~~~~~
ModelForm与一般的Form的重要区别是,ModelForm具有save()功能。能将表单里的数据
加入到数据库,并返回一个Model对象。
为了演示方便,我就不采用模板了。同样是引用上面BookModelForm与Book:
_________________________________________________________________
def add_book_view(request):
    book_form = forms.BookModelForm(request.GET)
    try:
        book_model = book_form.save()
        content = '<p>Add ' + book_model.title + ' success!</p>'
    except:
        content =  '<form action="." method="get">'
        content += book_form.as_p() 
        content += '<input type="submit"></form>'
    return HttpResponse(content)
-----------------------------------------------------------------
并将add_book_view视图的url指定为r'^add-book/'。

第一次访问/add-book/时,由于GET中没有参数,所以在book_form.save()就会抛出异
常。在except中返回个表单给用户。用户填好后提交。这次再处理时,GET里就有数据
了,所以book_form.save()正常,最后输出success消息。

有时,我们在save()时仅仅是想验证一下用户的输入,并不打算提交到数据为。
这里,只要save(commit=False)即可。
如此,在save(commit=False)时返回了book模型的对象。我们可以继而对其进一步修改
,再保存到数据库。
_____________________________________________________
        #book_model = book_form.save()
        book_model = book_form.save(commit=False)
        book_model.title = 'Balabala'
        book_model.save()
-----------------------------------------------------
第一行只是验证一下用户的输出是不是符合要求。然后对模型对象进行修改,最后才保
存到数据库去。就这样,在中间改了title,然后再保存到了数据库。

ModelForm显示个别域
~~~~~~~~~~~~~~~~~~~
ModelForm默认情况下,与Model是一致的。但是很多时候,并不是模型中所有的域都要
让用户填的。我们可以选择性地选择或排除个别域。
这里就要用到Meta的exclude或fields。exclude表示排除什么域,而fields表示需要显
示哪些域。二者不能同时使用。
_________________________________________________
class BookModelForm(forms.ModelForm):
    class Meta:
        model = Book 
        exclude = ('author') #表示不显示author域
-------------------------------------------------
_________________________________________________
class BookModelForm(forms.ModelForm):
    class Meta:
        model = Book 
        fields = ('title', 'isbn') 
        #表示只显示title与isbn域
-------------------------------------------------

重写ModelForm中的域
~~~~~~~~~~~~~~~~~~~
_________________________________________
class BookModelForm(forms.ModelForm):
    isbn = forms.CharField(max_length=13)
    class Meta:
        model = Book 
-----------------------------------------
将Book中指定的CharField(max_length=50),改成了13。

新增ModelForm中的域
~~~~~~~~~~~~~~~~~~~
______________________________________
class BookModelForm(forms.ModelForm):
    review = forms.CharField()
    class Meta:
        model = Book 
--------------------------------------
这样以来,BookModelForm不仅只有Book中的域,还有review。

表单的验证
~~~~~~~~~~
Form的is_valid()文件可以用来验证数据是否合法。
如果数据合法,Form对象则会有cleaned_data属性。如果不合法,则是errors。
__________________________________________________________________
def add_author_view(request):
    content = None
    if request.GET: # 检查GET是否有表单数据
        author_form = forms.AuthorModelForm(request.GET)
        if author_form.is_valid():  # 检查数据是否合法
            print(author_form.cleaned_data)
            try:
                author_form.save()
                content = '<p>Success</p>'
            except:
                content = '<p>Something wrong while saving.</p>'
        else:
            content = str(author_form.errors) # 用errors返回错误信息
    else:
        content =  '<form action=".">' 
        content += forms.AuthorModelForm().as_p() 
        content += '<input type="submit"></form>'
    return HttpResponse(content)
------------------------------------------------------------------
上面用到了author_form.is_valid()进行数据合法性验证。
如果成功,则会有author_form.cleaned_data,如果失败会有author_form.errors。
这两则只会存在一个,不会同时存在。

时间: 2024-10-30 18:14:22

Django学习笔记之——Forms的相关文章

Django学习笔记

1. 学习历程 从学了Python以来,我都一直想着用它来做点什么.从开始用python来写简单的工具脚本,然后再是网络信息抓取程序. 听说可以用Python来做网站,所以我了解了一下web.py与Django.第一次接触Django,由于没有网络方面的基础,所以没弄两下就放弃了. 后来,我研究了一下Python自来的SimpleHTTPServer,然后读懂了CGIHTTPServer,才对Web的实质有了了解. 在这个基础上,我再次重拾Django.这次,终于学会了. 原来它是那么的美妙!

django学习笔记(4)

Part 4: Forms and generic views ====> Write a simple form$ edit polls\templates\polls\detail.html <h1>{{ question.question_text }}</h1> {% if error_message %}<p><strong>{{ error_message }}</strong></p>{% endif %} <

django学习笔记(1)

Django 特点 强大的数据库功能     用python的类继承,几行代码就可以拥有一个丰富,动态的数据库操作接口(API),如果需要你也能执行SQL语句 自带的强大的后台功能     几行简单的代码就让你的网站拥有一个强大的后台,轻松管理你的内容! 优雅的网址     用正则匹配网址,传递到对应函数,随意定义,如你所想! 模板系统     强大,易扩展的模板系统,设计简易,代码,样式分开设计,更容易管理. 缓存系统     与memcached或其它的缓存系统联用,更出色的表现,更快的加载

Django学习笔记之——Models

Django里的模型是对数据库对表的一次封装,是应用业务与数据之间的桥梁. 1. 模型的Fields 在上一节,我们在mysite/blog/models.py中创建了BlogPost这个Model. from django.db import models class BlogPost(models.Model): title = models.CharField(max_length=150) body = models.TextField() timestamp = models.Date

Django学习笔记之——Start

1. 安装 下载源码文件: Django-1.6.10.tar.gz 解压后生成目录:Django-1.6.10 进入目录 $ sudo python setup.py install 就完成安装了. 2. 创建工程 执行: django-admin.py startproject <工程名> 完成后会在当前目录下生成<工程名>,比如mysite,的目录.在该目录下会有: . |-- manage.py `-- mysite |-- __init__.py |-- settings

Django学习笔记之——Urls

UrlConf模块感觉就像是"路由",它分析请求的路径,并请求派送给匹配的view去处理 打开mysite.urls.py文件,如下:_____________________________________________________from django.conf.urls import patterns, include, url from views import home import blog.urls urlpatterns = patterns('',    url

Django学习笔记之——Templates

模板~~~~顾名思意,就是一个通用的文档格式,等待填充动态的数据,形成完整的文档.在Django里,模板通常用于制定HTML文档,但是它还可以用于任何文本格式. 如下为模板的示例:templates/home.html____________________________<html><body><h1>Hello!</h1><p>My name is {{name}}.</p><p>Age: {{age}}.</p

Django学习笔记之——Views

在views中使用HttpResponse~~~~~~~~~~~~~~~~~~~~~~~~~ 每个view函数最终返回的都是一个HttpResponseBase.HttpResponse是HttpResponse-Base中最常用的一种.用法如下:_____________________________________________________________________from django.http import HttpResponse ## 最简单的view,直接返回正文de

django学习笔记(3)

Part 3: Views and templates ====> Write your first view$ edit polls\views.py from django.http import HttpResponse def index(request):    return HttpResponse("Hello, world. You're at the polls index.") $ edit polls\urls.py from django.conf.url