ModelForm

weixin_34249678發表於2017-12-09

和form表單不同的ModelForm,form表單不需要重新規劃欄位,而是直接從 ORM中取出資料

modelform建立

在models中

class UserType(models.Model):
    title = models.CharField(max_length=32)
    roles = models.ManyToManyField('Role')

views中

from django.forms import Form,ModelForm
from django.forms import fields
from django.forms import widgets as wd
class UserTypeModeForm(ModelForm):
    title = fields.CharField(max_length=6,required=True,widget=wd.Textarea())

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

        error_messages = {
            'title':{'required':'名稱不能為空','invalid':'格式錯誤'}
        }
        widgets = {
            'title':wd.TextInput(attrs={'class':'c1'})
        }

    def clean(self):

        return self.cleaned_data

class Meta中寫入關聯的model

新增資料

def user_type_add(request):
    if request.method == "GET":
        form = UserTypeModeForm()
        return render(request,'user_type_add.html',{'form':form})
    else:
        form = UserTypeModeForm(request.POST)
        if form.is_valid():
            form.instance.recv_date = ctime
            form.save()
            return redirect('/usertype/')
        else:
            return render(request, 'user_type_add.html', {'form': form})

form.save()驗證成功之後利用一句話就實現了儲存到資料庫操作
** form.instance.recv_date = ctime**可以通過form.instance.模型欄位新增到clean_data裡面

修改資料

def user_type_edit(request,nid):
    obj = models.UserType.objects.filter(id=nid).first()
    if not obj:
        return HttpResponse('...')

    if request.method == 'GET':
        # 顯示預設值
        form = UserTypeModeForm(instance=obj)
        return render(request,'user_type_edit.html',{'form':form})
    else:

        print(QueryDict(request.body))
        form = UserTypeModeForm(instance=obj,data=request.POST)
        if form.is_valid():
            form.save()
            return redirect('/usertype/')
        else:
            return render(request, 'user_type_edit.html', {'form': form})

form = UserTypeModeForm(instance=obj,data=request.POST)在修改中需要傳入在instance對應的模型物件,再在data傳入新資料

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的ClassMeta

class Meta:
# 這個是input前面的提示
labels = {
    'username':'使用者名稱',
    'password':'密碼',  
}
# 這個是input後面的提示
help_texts = {
    'username': '* 必須輸入使用者名稱',
    ...
}
# 修改元件的型別和增加類名等
from django.forms import widgets as Fwidgets
widgets = {
    'username':Fwidgets.Textarea(attrs={'class':'c1'})
}

# 修改錯誤資訊提示
error_messages = {
    # 定義所有的錯誤資訊
    '__all__':{
          ...   
    }
    ...
    'email':{
        'required':'郵箱不能為空',
        'invalid':'郵箱格式錯誤',
    }
}

# 自定義匹配格式
field_class:{
    ...
    'email':fields.URLField,
}
# 本地化,時區,
localized_fields = ('ctime',)