zoukankan      html  css  js  c++  java
  • Django 组件-ModelForm

    ModelForm

    组件功能就是把model和form组合起来.

    首先导入ModelForm

    from django.forms import ModelForm

    在视图函数中,定义一个类,比如就叫StudentList,这个类要继承ModelForm,在这个类中再写一个原类Meta(规定写法,并注意首字母是大写的)

    在这个原类中,有以下属性(部分):

    class StudentList(ModelForm):
        class Meta:
            model = models.Student   #对应的Model中的类
            fields = "__all__"             #字段,如果是__all__,就是表示列出所有的字段
            exclude = None              #排除的字段
            labels = None                 #显示信息
            help_texts = None          #帮助提示信息
            widgets = None              #自定义插件
            error_messages = None    #自定义错误信息]
    # error_messages用法
            error_messages = {
                  'name':{'required':'"用户名不能为空",},
                  'age':{'required':"年龄不能为空",},
            }
    
    #widgets用法,比如把输入用户名的input框给为Textarea
    #首先得导入模块
                from django.forms import widgets as wid  #因为重名,所以起个别名
                widgets = {
                       "name":wid.Textarea(attrs={"class":"c1"})  #还可以自定义属性
    }
    #labels,自定义在前端显示的名字
            labels = {
                     "name":"用户名"  
    }

    如果想增加一个字段,就在class Meta 之前写这个字段

    然后在url对应的视图函数中实例化这个类,把这个对象传给前端

    def student(request):
        
        if request.method == 'GET':
            student_list = StudentList()
            return render(request,'student.html',{'student_list':student_list})

    然后前端只需要 {{ student_list.as_p }} 一下,所有的字段就都出来了,可以用as_p显示全部,也可以通过for循环这student_list,拿到的是一个个input框,现在我们就不用as_p,手动把这些input框搞出来,as_p拿到的页面太丑。

    首先 for循环这个student_list,拿到student对象,直接在前端打印这个student,是个input框

    student.label  ,拿到数据库中每个字段的verbose_name ,如果没有设置这个属性,拿到的默认就是字段名

    还可以通过student.errors.0 拿到错误信息,还有student.field,是拿到每个字段,如果这个字段是多对多字段,还能stuent.field.queryset,拿到所有关联的字段,还能stuent.field.queryset.model,拿到所有关联的字段的类

    有了这些,我们就可以通过bootstrap,自己拼出来想要的样式了

     比如:

    <body>
        <div class="container" >
            <h1>student</h1>
            <form method="POST" novalidate>
                {% csrf_token %}
    {#            {{ student_list.as_p }}#}
                {% for student in student_list %}
                    <div class="form-group col-md-6">
                     {# 拿到数据字段的verbose_name,没有就默认显示字段名 #}
                        <label class="col-md-3 control-label">{{ student.label }}</label>
                       <div class="col-md-9" style="position: relative;">{{ student }}</div>
                    </div>
                {% endfor %}
    
                <div class="col-md-2 col-md-offset-10">
                    <input type="submit" value="提交" class="btn-primary">
                </div>
    
            </form>
        </div>
    </body>
    
    复制代码
    
    现在还缺一个input框的form-control样式,可以考虑在后台的widget里面添加
    
    比如这样:
    
    from django.forms import widgets as wid  #因为重名,所以起个别名
            widgets = {
                "name":wid.TextInput(attrs={'class':'form-control'}),
                "age":wid.NumberInput(attrs={'class':'form-control'}),
                "email":wid.EmailInput(attrs={'class':'form-control'})
            }

    当然也可以在js中,找到所有的input框,加上这个样式,也行

    保存数据的时候,不用挨个取数据了,只需要save一下

    def student(request):
    
        if request.method == 'GET':
            student_list = StudentList()
            return render(request,'student.html',{'student_list':student_list})
        else:
            student_list = StudentList(request.POST)
            if student_list.is_valid():
                student_list.save()
            return redirect(request,'student_list.html',{'student_list':student_list})

    编辑数据:

    如果不用ModelForm,编辑的时候得显示之前的数据吧,还得挨个取一遍值,如果ModelForm,只需要加一个instance=obj(obj是要修改的数据库的一条数据的对象)就可以得到同样的效果

    保存的时候要注意,一定要注意有这个对象(instance=obj),否则不知道更新哪一个数据

    代码示例:

    from django.shortcuts import render,HttpResponse,redirect
    from django.forms import ModelForm
    # Create your views here.
    from app01 import models
    
    def test(request):
    
        # model_form = models.Student
        model_form = models.Student.objects.all()
        return render(request,'test.html',{'model_form':model_form})
    
    
    class StudentList(ModelForm):
        class Meta:
            model = models.Student  #对应的Model中的类
            fields = "__all__"      #字段,如果是__all__,就是表示列出所有的字段
            exclude = None          #排除的字段
            labels = None           #提示信息
            help_texts = None       #帮助提示信息
            widgets = None          #自定义插件
            error_messages = None   #自定义错误信息
    #error_messages用法:
            error_messages = {
                'name':{'required':"用户名不能为空",},
                'age':{'required':"年龄不能为空",},
            }
    
    #widgets用法,比如把输入用户名的input框给为Textarea
    #首先得导入模块
            from django.forms import widgets as wid  #因为重名,所以起个别名
            widgets = {
                "name":wid.Textarea
            }
    #labels,自定义在前端显示的名字
            labels= {
                "name":"用户名"
            }
    def student(request):
    
        if request.method == 'GET':
            student_list = StudentList()
            return render(request,'student.html',{'student_list':student_list})
        else:
            student_list = StudentList(request.POST)
            if student_list.is_valid():
                student_list.save()
            return render(request,'student.html',{'student_list':student_list})
    
    def student_edit(request,pk):
        obj = models.Student.objects.filter(pk=pk).first()
        if not obj:
            return redirect('test')
        if request.method == "GET":
            student_list = StudentList(instance=obj)
            return render(request,'student_edit.html',{'student_list':student_list})
    
        else:
            student_list = StudentList(request.POST,instance=obj)
            if student_list.is_valid():
                student_list.save()
            return render(request,'student_edit.html',{'student_list':student_list})
    View Code

    对于验证规则,很多浏览器都比较智能,会自动帮我们做一些验证,可以在form表单上加  novalidate 属性就可以不让浏览器为我们做验证

    ModelForm还支持所有form的功能,比如钩子,所以我们就可以通过钩子来自定义验证规则

    写法和forms的写法一样:

    class AuthorForm(forms.ModelForm):
        class Meta:
            model = Author
            fields = ('name', 'title')
    
        def clean_name(self):
            if ...
                 return self.clean_data['name']
           else:
                 raise ValidationError(‘sdgsadga’)
            ...
  • 相关阅读:
    .net String.Format数字格式化输出
    sql prompt5安装好了,也破解完成了,然后到SQL里面还是没有提示是为什么?
    DataSet筛选数据然后添加到新的DataSet中引发的一系列血案
    .net 时间戳互相转换(精确到毫秒)
    记录一次MVC 3.0错误 HTTP 404您正在查找的资源(或者它的一个依赖项)可能已被移除,或其名称已更改,或暂时不可用。请检查以下 URL 并确保其拼写正确。
    IIS7部署项目时提示:"错误消息 401.2。: 未经授权: 服务器配置导致登录失败。"的解决办法
    IIS发布网站出现“未能加载文件或程序集“System.Data.SQLite”或它的某一个依赖项。”的解决方法
    微信支付(APP)集成时碰到的问题(.net提示“无权限”、iOS跳转到微信支付页面中间只有一个“确定”按钮)
    由SpringMVC中RequetContextListener说起
    关于cas-client单点登录客户端拦截请求和忽略/排除不需要拦截的请求URL的问题(不需要修改任何代码,只需要一个配置)
  • 原文地址:https://www.cnblogs.com/zbw582922417/p/9981403.html
Copyright © 2011-2022 走看看