Django 【第二十二篇】ModelForm

一、ModelForm的介绍

ModelForm
    a.  class Meta:
            model,                           # 对应Model的
            fields=None,                     # 字段
            exclude=None,                    # 排除字段
            labels=None,                     # 提示信息
            help_texts=None,                 # 帮助提示信息
            widgets=None,                    # 自定义插件
            error_messages=None,             # 自定义错误信息(整体错误信息from django.core.exceptions import NON_FIELD_ERRORS)
            field_classes=None               # 自定义字段类 (也可以自定义字段)
            localized_fields=(‘birth_date‘,) # 本地化,如:根据不同时区显示数据
            如:
                数据库中
                    2016-12-27 04:10:57
                setting中的配置
                    TIME_ZONE = ‘Asia/Shanghai‘
                    USE_TZ = True
                则显示:
                    2016-12-27 12:10:57
    b. 验证执行过程
        is_valid -> full_clean -> 钩子 -> 整体错误

    c. 字典字段验证
        def clean_字段名(self):
            # 可以抛出异常
            # from django.core.exceptions import ValidationError
            return "新值"
    d. 用于验证
        model_form_obj = XXOOModelForm()
        model_form_obj.is_valid()
        model_form_obj.errors.as_json()
        model_form_obj.clean()
        model_form_obj.cleaned_data
    e. 用于创建
        model_form_obj = XXOOModelForm(request.POST)
        #### 页面显示,并提交 #####
        # 默认保存多对多
            obj = form.save(commit=True)
        # 不做任何操作,内部定义 save_m2m(用于保存多对多)
            obj = form.save(commit=False)
            obj.save()      # 保存单表信息
            obj.save_m2m()  # 保存关联多对多信息

    f. 用于更新和初始化
        obj = model.tb.objects.get(id=1)
        model_form_obj = XXOOModelForm(request.POST,instance=obj)
        ...

        PS: 单纯初始化
            model_form_obj = XXOOModelForm(initial={...})

应用场景:
            - ModelForm - 中小型应用程序。因为ModelForm是依赖于models的
            - Form      - 大型应用程序  *

注意事项:

注意事项:
            - 1. 类
                  class Foo(ModelForm):
                    class Meta:
                        # model = models.Role
                        # fields = "__all__"
                        # fields = [‘caption‘,]
                        # exclude = [‘catpion‘]
                        model = models.UserType
                        fields = "__all__"

                        error_messages = {
                            ‘title‘:{‘required‘:‘名称不能为空‘,‘invalid‘:‘格式错误‘}
                        }
                        widgets = {
                            ‘title‘:wd.TextInput(attrs={‘class‘:‘c1‘})
                        }

            - 2. 添加
                 GET:
                    form = Foo()
                 POST:
                    form = Foo(data=request.POST)
                    form.is_valid()
                    form.cleaned_data
                    form.erros
                    form.save()
            - 3. 修改
                 GET:
                    form = Foo(instance=obj)

                 POST:
                    form = Foo(instance=obj,dat=request.POST)
                    ...
                    form.save()

二、表结构

from django.db import models

# Create your models here.
class UserInfo (models.Model):
    username = models.CharField(max_length=32)
    email = models.EmailField(max_length=32)
    ut = models.ForeignKey("UserType")

class UserType (models.Model):
    title = models.CharField(max_length=32)
    roles = models.ManyToManyField(to="Roles")
    def __str__(self):
        return self.title

class Roles(models.Model):
    caption = models.CharField(max_length=32)
    def __str__(self):
        return self.caption

三、基于Form组件的添加和编辑

添加:这只是单表的添加

from django.forms import Form, fields,widgets,ModelForm
from django.shortcuts import render,redirect,HttpResponse
from app01 import models
# Create your views here.
class RoleForm(Form):
    ‘‘‘利用Form‘‘‘
    caption = fields.CharField(required=True,error_messages={"required":True})

def role(request):
    role_obj = models.Roles.objects.all()
    print(role_obj)
    return render(request, "role.html",{"role_obj":role_obj})

# 基于Form实现的
def role_add(request):
    ‘‘‘添加角色‘‘‘
    if request.method=="GET":
        form = RoleForm()
        return render(request,"role_add.html",{"form":form})
    else:
        form = RoleForm(data=request.POST)
        if form.is_valid():
            print("zzzzz")
            caption = form.cleaned_data.get("caption")
            models.Roles.objects.create(caption=caption)
            # models.Roles.objects.create(**form.cleaned_data)
            return redirect("/role/")
        else:
            return render(request,"role_add.html",{"form":form})

编辑:单表的编辑

#基于Form实现的编辑
def role_edit(request,nid):
    obj = models.Roles.objects.filter(id=nid).first()
    print(obj.caption)
    if not obj :
        return HttpResponse("页面不存在")
    if request.method=="GET":
        form = RoleForm(initial={"caption":obj.caption})   #编辑的时候需要一个instance,让instance=一个你要编辑的那个对象
    else:
        form = RoleForm(data = request.POST)
        if form.is_valid():
            models.Roles.objects.filter(id=nid).update(**form.cleaned_data)
            return redirect("/role/")
    return render(request,"role_edit.html",{"form":form})

具体基于Form组件实现的一对多添加或者多对多添加编辑详见博客http://www.cnblogs.com/haiyan123/p/7816877.html

四、基于ModelForm的添加和编辑

添加:单表的添加

# 基于ModelForm的添加
class RoleModelForm(ModelForm):
    class Meta:   #这个类必须写,而且名字必须是这个
        model = models.Roles   #这个model也是固定的,注意不加s,
        fields = "__all__"   #代表所有的字段,但是你也可以指定单个的字段

def role_add(request):
    if request.method == "GET":
        form = RoleModelForm()
        return render(request,"role_add.html",{"form":form})
    else:
        form = RoleModelForm(data=request.POST)
        if form.is_valid():
            form.save()   #这里直接可以用save方法,就把数据创建了
            return redirect("/role/")
        else:
            return render(request,"role_add.html",{"form":form})

添加:多对多的添加,一堆多的提添加也是一样

# 多对多的添加
def usertype(request):
    user_type_list = models.UserType.objects.all()
    return render(request,"usertype.html",{‘user_type_list‘:user_type_list})

class UserTypeModelForm(ModelForm):
    title = fields.CharField(max_length=6,required=True,widget=widgets.Textarea)   #这个字段是临时添加的,
    # 也就是说modelForm也可以用Form的方式。
    # 也可以以这样的方式新增字段, 如果有就覆盖,没有就增加;像现在这种情况就是吧下面的给覆盖了,当然没有上面的这个就用下面的了

    class Meta:
        model = models.UserType
        fields = "__all__"

        error_messages = {
            "title":{"required":"用户名不能为空","invalid":"邮箱格式不一致"}
        }
        widgets = {
            "title":widgets.TextInput(attrs={"class":"c1"})
        }

def usertype_add(request):
    ‘‘‘多对多的添加‘‘‘
    if request.method=="GET":
        modelform = UserTypeModelForm()
        return render(request,"usertype_add.html",{"modelform":modelform})
    else:
        modelform = UserTypeModelForm(data=request.POST)
        if modelform.is_valid():
            modelform.save()   #也可以用save来实现,就连关系表的字段也都添加了
            return redirect("/usertype/")
        else:
            return render(request, "usertype_add.html", {"modelform": modelform})

编辑:单表的编辑

# 基于modelForm实现的编辑
def role_edit(request,nid):
    obj = models.Roles.objects.filter(id=nid).first()
    if not obj :
        return HttpResponse("页面不存在")
    if request.method=="GET":
        form = RoleModelForm(instance=obj)   #编辑的时候需要一个instance,让instance=一个你要编辑的那个对象
    else:
        form = RoleModelForm(data = request.POST,instance=obj)
        if form.is_valid:
            form.save()
            return redirect("/role/")
    return render(request,"role_edit.html",{"form":form})

编辑:多对多的编辑

# 多对多的编辑
def usertype(request):
    user_type_list = models.UserType.objects.all()
    return render(request,"usertype.html",{‘user_type_list‘:user_type_list})

class UserTypeModelForm(ModelForm):
    title = fields.CharField(max_length=6,required=True,widget=widgets.Textarea)   #这个字段是临时添加的,
    # 也就是说modelForm也可以用Form的方式。
    # 也可以以这样的方式新增字段, 如果有就覆盖,没有就增加;像现在这种情况就是吧下面的给覆盖了,当然没有上面的这个就用下面的了

    class Meta:
        model = models.UserType
        fields = "__all__"

        error_messages = {
            "title":{"required":"用户名不能为空","invalid":"邮箱格式不一致"}
        }
        widgets = {
            "title":widgets.TextInput(attrs={"class":"c1"})
        }

def usertype_edit(request,nid):
    #查出当前类型用户对应的角色
    obj = models.UserType.objects.filter(id =nid).first()
    if not obj:
        return HttpResponse("页面不存在")
    if request.method =="GET":
        form = UserTypeModelForm(instance=obj)
        return render(request,"usertype_edit.html",{"form":form})
    else:
        form = UserTypeModelForm(instance=obj,data=request.POST)
        if form.is_valid():
            form.save()
            return redirect("/usertype/")
    return render(request,"usertype_edit.html",{"form":form})

原文地址:https://www.cnblogs.com/xiaohema/p/8456609.html

时间: 2024-10-08 17:21:35

Django 【第二十二篇】ModelForm的相关文章

Python开发【第二十二篇】:Web框架之Django【进阶】

Python开发[第二十二篇]:Web框架之Django[进阶] 猛击这里:http://www.cnblogs.com/wupeiqi/articles/5246483.html 博客园 首页 新随笔 联系 订阅 管理 随笔-124  文章-127  评论-205 Python之路[第十七篇]:Django[进阶篇 ] Model 到目前为止,当我们的程序涉及到数据库相关操作时,我们一般都会这么搞: 创建数据库,设计表结构和字段 使用 MySQLdb 来连接数据库,并编写数据访问层代码 业务逻

第二十二篇:再写Windows驱动,再玩Windbg---NET

2011年到现在,就没再怎么搞过Windows驱动了. 最近, 由于项目需要, 试着改一改一个显卡驱动(KMDOD), 从实践上证明, 我在理论上对一个驱动的架构的正确与否.(USB Display = KMDOD + AVStream). 其中, KMDOD是完成显示的部分功能, 完成其中的VidPN(Video present network), 将驱动中原来的POST物理设备转变为USB物理设备. 而AVStream之所以这样提出, 完成是由于USB Video class的启发, 要不然

第二十二篇 信念

第二十二篇  信念 "信念"能带给一个人无穷的力量,这些力量可以支撑自己走过漫长的人生.一个人如果没有信念,就很难找到自己的人生方向,所以"信念"也可以理解为希望. 信念可以给到我们希望,也可以给到我们力量,所以一个人的信念会影响到自己的整个人生.当然信念也有好坏之分,好的信念能让自己积极向上.不畏艰难:坏的信念会让我们不思进取.随波逐流.这两种不同的信念会给到我们两种完全不同的人生,就看亲人们如何作出正确的选择. 一个人活在世上,可以选择走正确的人生道路,依靠好的

Egret入门学习日记 --- 第二十二篇(书中 9.7~9.8 节 内容)

第二十二篇(书中 9.7~9.8 节 内容) 开始 9.7节 内容. 重点: 1.进度条ProgressBar的声明和使用. 操作: 1.进度条ProgressBar的声明和使用. 现在真的轻车熟路了,很简单.无非就是设置一下最大值,当前值的属性. 然后,事件监听的话,也是一样的.只不过事件名字的话,我就选书中这个事件吧. 可惜不能发动图,不然你们就可以看到这个进度条,每帧+1的速度前进. 当然,如果你想换自定义皮肤,还是老规矩,去找默认的 EXML 文件. 然后,怎么换素材,就按照自己喜欢的换

Python之路【第二十二篇】:Django之Model操作

Django之Model操作 一.字段 AutoField(Field) - int自增列,必须填入参数 primary_key=True BigAutoField(AutoField) - bigint自增列,必须填入参数 primary_key=True 注:当model中如果没有自增列,则自动会创建一个列名为id的列 from django.db import models class UserInfo(models.Model): # 自动创建一个列名为id的且为自增的整数列 usern

【Python之路】第二十二篇--Django【基础篇】

1 Django流程介绍 MTV模式       著名的MVC模式:所谓MVC就是把web应用分为模型(M),控制器(C),视图(V)三层:他们之间以一种插件似的,松耦合的方式连接在一起. 模型负责业务对象与数据库的对象(ORM),视图负责与用户的交互(页面),控制器(C)接受用户的输入调用模型和视图完成用户的请求. Django的MTV模式本质上与MVC模式没有什么差别,也是各组件之间为了保持松耦合关系,只是定义上有些许不同,Django的MTV分别代表: Model(模型):负责业务对象与数

第二十二篇 Java的一些关键字、 作用域 还有运算符的使用

大家好,我来和大家一起分享一下今天我所学习到的一些知识点,今天我学会了怎么去使用Java中的一些关键字 .作业域和运算符的使用.... 首先来介绍一下 Java中的关键字有哪些,由于数量有很多 ,我就来说一下我今天所学会的关键字 与其作用: 我学到的第一个关键字是----Scanner ,他在Java中 代表这扫描器的作用,并且要引入包,他的作用就是监听键盘的输入,其中见过次数最多的是在注册或登陆界面上,那是候会直接采用你所输入的值,并取出来使用, 第二个关键字是 是-------return,

python全栈开发基础【第二十二篇】进程池和回调函数

一.数据共享 1.进程间的通信应该尽量避免共享数据的方式 2.进程间的数据是独立的,可以借助队列或管道实现通信,二者都是基于消息传递的. 虽然进程间数据独立,但可以用过Manager实现数据共享,事实上Manager的功能远不止于此. 命令就是一个程序,按回车就会执行(这个只是在windows情况下) tasklist 查看进程 tasklist | findstr pycharm #(findstr是进行过滤的),|就是管道(tasklist执行的内容就放到管道里面了, 管道后面的findst

第二十二篇:信号的接收和处理

前言 要想掌握 Linux 系统编程,自然要好好学学其信号机制. 本文介绍一个简单的信号接收处理程序,为后面继续深入学习信号机制打下基础. 什么是信号 信号是软件中断,它提供一种处理异步事件的方法. 信号产生的条件 1. 当用户按某些终端按键时.比如:Ctrl + D / Ctrl + C 等. 2. 硬件异常.比如:除数为 0,无效内存引用等. 3. 调用 kill 函数可以将信号发送给另一个进程或者进程组. 4. 当检测到某种软件条件已经发生时.比如:alarm 到时,网络传来某些带外数据时