前面学习的都只是如何显示数据,但一直没有关于如何响应用户提交的数据。
表单,是在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。
这两则只会存在一个,不会同时存在。