四【用django2.0来开发】后台会员管理(二) ModelForm表单的使用方法以及数据验证

上一节我们讲完了ModelAdmin的使用, 但是在操作中也发现, 新增编辑会员时, 我们无法验证数据是否正确, 比如

  1. 用户名、手机号、邮箱都应该是唯一的
  2. 用户名和密码长度的验证
  3. 编辑用户信息时不填写密码则不更新密码, 填写了则更新密码
  4. 自定义验证不通过的文案
    ...

这些要求, 我们就必须得使用自定义的表单来完成了

项目地址:https://gitee.com/ccnv07/django_example

Form介绍

通过表单, 我们可以实现以下的功能

  1. 自定义字段的样式
  2. 类似的表单可以通过类继承减少代码量
  3. 完成自定义的表单数据验证

定义表单字段

关于表单的代码我们一般放在每个模块的forms.py中


image.png

表单字段类型说明

CharField
单行文本输入字段, 对应模型的CharField字段
表单中的样式就是input type=text
max_length: 最大长度
min_length: 最小长度
strip: 是否过滤左右的空格
empty_value: 为空时的值, 默认是空字符串

EmailField
邮箱输入文本字段, 对应模型的EmailField字段
标案中是input type=text
但是会自动增加一个邮箱格式的校验

ChoiceField
下拉单选字段, 这个在模型中是没有的
对应表单的select标签
choices参数也是二维元组的格式

BooleanField
选择字段, 对应表单中的checkbox

DateField
日期选择字段
input_formats: 定义时间格式, 默认是:

['%Y-%m-%d',      # '2006-10-25'
 '%m/%d/%Y',      # '10/25/2006'
 '%m/%d/%y']      # '10/25/06'

DateTimeField
日期时间字段, 同DateField

TimeField
时间字段, 同DateField

DecimalField
十进制数字字段
max_value: 最大值
min_value: 最小值
max_digits: 前置0被去除后的最大位数
decimal_places: 允许的小数位的长度

FileField
文件上传字段

IntegerField
整数字段
max_value: 最大值
min_value: 最小值

剩下还有很多表单字段类型, 在之后的教程中再继续介绍, 目前这些, 就算是比较常见的字段了

通过介绍以后, 大家发现, 我没说最常见的PasswordField, 其实django的表单中就没有这个字段类型, 那么, 如何实现密码字段呢?

password = forms.CharField(widget=forms.PasswordInput(),max_length=12,min_length=6, strip=True,
        help_text='编辑时为空则不更改密码')

通过制定widget参数为forms.PasswordInput(), 就可以实现密码字段了

所以其实并不是Form类来规定表单字段的类型, 而是widget来实现的表单字段的类型, 对于每种表单字段的类型, django都有对应的模板, 通过字段的参数, 生成对应的html代码

表单字段的通用参数

label: 表单字段的label标签的名称
widget: 指定此字段采用的字段样式类型
help_text: 字段的帮助文本

表单的Meta 元类

只有ModelForm才有元类, Form是不需要元类的
以下是一个元类的例子

from django.forms import ModelForm
class Meta:
        model = Account
        # 使用自定义的Form, 就必须指定fields or exclude属性, 否则报错
        fields = ('account', 'password', 'email', 'phone', 'status')
        error_messages = {
            NON_FIELD_ERRORS: {
                'unique_together': "%(model_name)s's %(field_labels)s are not unique.",
            }
        }

model: 指定ModelForm绑定的模型
fields: 指定后台新增编辑时要显示的字段
error_messages: 指定通用的错误信息文案

其他还有一些复杂的操作, 会在之后的教程中详细讲解。现在我们的一个表单就完成了

将表单绑定到ModelAdmin中

但是如果想让表单在后台中生效, 就需要把表单绑定到ModelAdmin上

from django.contrib import admin
from .forms import AccountForm

@admin.register(Account)
class AccountAdmin(admin.ModelAdmin):
    form = AccountForm

在ModelAdmin中指定form参数, 就可以把表单绑定上去了。 点击新增/编辑页面, 也就可以看到表单生效了

image.png

表单数据的验证

接下来, 我们就需要完成表单提交数据以后的验证的方法了
其实在定义表单字段时, 我们就已经完成了一部分的验证了
比如

 account = forms.CharField(
        required=True, error_messages={
            'required': '请输入用户名',
        }, label='用户名')

就定义了account字段必须填写, 如果出错则返回“请输入用户名”的提示

但是这个并不能完成我们所有的验证, 所以我们也可以根据字段进行自定义的验证
比如, 我要实现account用户名字段是唯一的

from django import forms
from django.core.exceptions import ValidationError
from .models import Account

class AccountForm(forms.ModelForm):
    ...省略代码
    def clean_account(self):
        _info = Account.objects.filter(account=self.cleaned_data['account'],
                is_deleted=0).values('id')

        if _info:
            raise ValidationError('用户已存在')
        return self.cleaned_data['account']

当我们执行form.is_valid()方法进行验证时, django的form类会依次执行clean_字段名的自定义验证方法, 如果有抛出异常(raise ValidationError('用户已存在')), 则中断并返回错误, 否则读取到clean_字段名的方法的返回值, 并且写入到cleaned_data这个字典中

根据同样的方法, 我们也可以写出来对email和phone字段的验证

在ModelAdmin操作Form

以上已经基本能实现我们的功能了, 但是在后台的操作中, 新增和编辑用的是同一个表单, 如果我们需要针对新增和编辑的不同场景, 进行一些不同的操作, 就比较麻烦了。
所以我们需要通过在ModelAdmin中, 对Form进行一定的操作, 似的Form可以完成更多的判断

# account/admin.py
class AccountAdmin(admin.ModelAdmin):
    def get_form(self, request, obj=None, **kwargs):
        form = super(AccountAdmin, self).get_form(request, obj=obj, **kwargs)

        # obj 保存的是models.Account的信息
        # 根据是否有pk, 来赋予form不同的场景, 根据不同的场景可以进行不同的验证
        if (obj is not None):
            form.id = obj.pk
            form.scene = 'update'
        else:
            form.scene = 'insert'
        return form

get_form方法的参数request保存的是HttpRequest操作对象
而obj是当前操作的数据模型对象(Model), 而且在新增的操作时, obj是None, 只有在编辑时, 才存在obj

form = super(AccountAdmin, self).get_form(request, obj=obj, **kwargs) 会返回当前操作的form对象
如果obj不是None, 则当前操作是编辑, 我们就可以给form增加一个自定义的属性scene(场景) = 'update', 否则就是新增

并且我们在Form表单中, 可以针对不同的场景, 进行不同的验证判断

# account/forms.py
class AccountForm(forms.ModelForm):
    def clean_account(self):
        # 自动验证account字段
        if self.scene == 'insert':
            _info = Account.objects.filter(
                account=self.cleaned_data['account'],
                is_deleted=0).values('id')

        elif self.scene == 'update':
            _info = Account.objects.filter(~Q(id=self.id) & Q(
                account=self.cleaned_data['account']) & Q(
                    is_deleted=0)).values('id')

        if _info:
            raise ValidationError('用户已存在')
        return self.cleaned_data['account']

这个验证的意思是, 如果当前场景(self.scene)是insert, 就只按照account查询, 如果是update, 则增加id不为当前操作id的过滤条件


image.png

Django的密码加密方法

django.contrib.auth.hashers 有两个关于密码加密的操作方法
make_password和check_password

make_password 是将指定的明文密码加密
check_password 是校验给出的明文密码是否正确

image.png

新增与编辑操作下密码框的不同操作

当新增用户时, 密码框必填, 当编辑时, 密码框非必填。如果填了, 则修改密码, 如果没填, 则不更改密码

第一步, 我们在校验输入的密码时, 同时也需要实现对密码的加密(毕竟数据库被人破解了, 后台还是很严重的)

from django import forms
class AccountForm(forms.ModelForm):
    def clean_password(self):
        # 自动验证密码字段
        if self.scene == 'insert':
            if not self.cleaned_data['password']:
                raise ValidationError('请输入密码')
        elif self.scene == 'update':
            if not self.cleaned_data['password']:
                return None
            else:
                return self.cleaned_data['password']
        return make_password(self.cleaned_data['password'])

这个也很好理解, 如果新增用户时, 未输入密码, 则返回错误
更新时没有输入密码, 则返回None
如果输入, 就返回make_password加密后的密码字符串

根据之前教程的scene场景参数的指定, 就可以跟容易的实现这个功能, 但是在编辑时, 如果密码为空不填的话, 密码居然也会被设置为空。

这个是因为, 如果不填写密码, 的model对象会把password=None一直带着, 转换为sql执行时, 就变成password=''了

所以, 如果password没有输入值, 我们就要在执行保存之前, 干掉model携带的password, 这样才正确。

重写ModelAdmin的保存方法

class AccountAdmin(admin.ModelAdmin):
    def save_model(self, request, obj, form, change):
        if form.cleaned_data['password'] is None:
            del obj.password
        super().save_model(request, obj, form, change)

form.cleaned_data中就是表单提交后验证过的数据, 如果password是None, 就del掉, 然后调用父类的save_model方法, 继续执行保存操作。

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

推荐阅读更多精彩内容