Django4.0 基于類(lèi)的視圖-使用基于類(lèi)的視圖處理表單

2022-03-16 17:57 更新

表單處理通常有3個(gè)途徑:

  • 初始 GET (空白或預(yù)填充表單)
  • 帶有合法數(shù)據(jù)的POST(通常是帶有錯(cuò)誤的重新顯示的表單)
  • 帶有合法數(shù)據(jù)的POST(處理數(shù)據(jù)和通常重定向)

自己實(shí)現(xiàn)這個(gè)會(huì)導(dǎo)致很多重復(fù)的樣板代碼。為了避免這個(gè)問(wèn)題,Django 提供了一組通用基于類(lèi)的視圖來(lái)處理表單。

基礎(chǔ)表單

提供聯(lián)系表單:

from django import forms

class ContactForm(forms.Form):
    name = forms.CharField()
    message = forms.CharField(widget=forms.Textarea)

    def send_email(self):
        # send email using the self.cleaned_data dictionary
        pass

可以使用 ?FormView ?構(gòu)建視圖:

from myapp.forms import ContactForm
from django.views.generic.edit import FormView

class ContactFormView(FormView):
    template_name = 'contact.html'
    form_class = ContactForm
    success_url = '/thanks/'

    def form_valid(self, form):
        # This method is called when valid form data has been POSTed.
        # It should return an HttpResponse.
        form.send_email()
        return super().form_valid(form)

注意:

  • ?FormView ?繼承了 ?TemplateResponseMixin ?,因此可以在這里使用 ?template_name ?。
  • 默認(rèn)實(shí)現(xiàn)了 ?form_valid()?  簡(jiǎn)單重定向至 ?success_url ?。

模型表單

通用視圖在模型一起工作時(shí)真的很贊。這些通用視圖將自動(dòng)創(chuàng)建 ?ModelForm ?,只要他們能找出要使用的模型類(lèi):

  • 如果已經(jīng)給出了 ?model ?屬性,則使用這個(gè)模型類(lèi)。
  • 如果 ?get_object()? 返回一個(gè)對(duì)象,則使用這個(gè)對(duì)象的類(lèi)。
  • 如果已經(jīng)給出了 ?queryset  ?,則使用這個(gè)查詢(xún)集的模型。

模型表單視圖提供一個(gè) ?form_valid()? 實(shí)現(xiàn),來(lái)自動(dòng)保存模型。如果你有特別的需求,你可以覆蓋它。
你甚至不需要為 ?CreateView ?或 ?UpdateView ?提供 ?success_url ?。如果可用,它們將在模型對(duì)象上使用 ?get_absolute_url()? 。
如果你想使用一個(gè)自定義的 ?ModelForm ?(比如添加額外的驗(yàn)證),用來(lái)在視圖上設(shè)置 ?form_class ?。

注解:當(dāng)指定一個(gè)自定義表單類(lèi)時(shí),必須也要指定模型,即使 ?form_class ?可能是一個(gè) ?ModelForm ?。

首先我們需要添加 ?get_absolute_url()? 到 ?Author ?類(lèi):

from django.db import models
from django.urls import reverse

class Author(models.Model):
    name = models.CharField(max_length=200)

    def get_absolute_url(self):
        return reverse('author-detail', kwargs={'pk': self.pk})

然后可以使用 ?CreateView ?并友好的執(zhí)行實(shí)際工作。注意這里我們?nèi)绾闻渲猛ㄓ没陬?lèi)的視圖。我們不用自己編寫(xiě)任何邏輯:

from django.urls import reverse_lazy
from django.views.generic.edit import CreateView, DeleteView, UpdateView
from myapp.models import Author

class AuthorCreateView(CreateView):
    model = Author
    fields = ['name']

class AuthorUpdateView(UpdateView):
    model = Author
    fields = ['name']

class AuthorDeleteView(DeleteView):
    model = Author
    success_url = reverse_lazy('author-list')

注解:我們必須在這里使用 ?reverse_lazy()? 來(lái)代替 ?reverse()? ,因?yàn)樵谖募?dǎo)入時(shí)不加載 ?urls ?。

?fields ?屬性的工作方式同 ?ModelForm ?中內(nèi)部 ?Meta ?類(lèi)的 ?fields ?屬性一樣。除非你使用其他方式定義表單類(lèi),該屬性是必需的,如果屬性不存在,視圖將引發(fā) ?ImproperlyConfigured ?異常。
如果同時(shí)指定了 ?fields ?和 ?form_class ?屬性,將會(huì)引發(fā) ?ImproperlyConfigured ?異常。
最后將這些新視圖掛鉤到?URLconf?中:

from django.urls import path
from myapp.views import AuthorCreateView, AuthorDeleteView, AuthorUpdateView

urlpatterns = [
    # ...
    path('author/add/', AuthorCreateView.as_view(), name='author-add'),
    path('author/<int:pk>/', AuthorUpdateView.as_view(), name='author-update'),
    path('author/<int:pk>/delete/', AuthorDeleteView.as_view(), name='author-delete'),
]

注解:這些視圖繼承 ?SingleObjectTemplateResponseMixin ?,它使用 ?template_name_suffix ?來(lái)構(gòu)建基于模型的 ?template_name ?。在這個(gè)例子里:

  • ?CreateView ?和 ?UpdateView ?使用 ?myapp/author_form.html? 。
  • ?DeleteView ?使用 ?myapp/author_confirm_delete.html? 。

如果你想為 ?CreateView ?和 ?UpdateView ?制作單獨(dú)的模板,你可以在視圖類(lèi)上設(shè)置 ?template_name ?或 ?template_name_suffix ?。

模型和 request.user

若要跟蹤使用 ?CreateView ?創(chuàng)建的用戶(hù),你可以使用自定義的 ?ModelForm ?來(lái)執(zhí)行此操作。首先,在模型里添加外鍵關(guān)系:

from django.contrib.auth.models import User
from django.db import models

class Author(models.Model):
    name = models.CharField(max_length=200)
    created_by = models.ForeignKey(User, on_delete=models.CASCADE)

    # ...

在這個(gè)視圖中,確保你沒(méi)有在要編輯的字段列表中包含 ?created_by ?字段 ,并且要覆蓋 ?form_valid()? 來(lái)添加用戶(hù):

from django.contrib.auth.mixins import LoginRequiredMixin
from django.views.generic.edit import CreateView
from myapp.models import Author

class AuthorCreateView(LoginRequiredMixin, CreateView):
    model = Author
    fields = ['name']

    def form_valid(self, form):
        form.instance.created_by = self.request.user
        return super().form_valid(form)

?LoginRequiredMixin ?防止那些未登錄的用戶(hù)訪(fǎng)問(wèn)表單。如果忽略,那么你將需要在 ?form_valid()? 里處理未授權(quán)的用戶(hù)。

內(nèi)容協(xié)商示例

下面是一個(gè)展示了如何實(shí)現(xiàn)基于API的工作流以及普通?POST?表單一起使用的表單:

from django.http import JsonResponse
from django.views.generic.edit import CreateView
from myapp.models import Author

class JsonableResponseMixin:
    """
    Mixin to add JSON support to a form.
    Must be used with an object-based FormView (e.g. CreateView)
    """
    def form_invalid(self, form):
        response = super().form_invalid(form)
        if self.request.accepts('text/html'):
            return response
        else:
            return JsonResponse(form.errors, status=400)

    def form_valid(self, form):
        # We make sure to call the parent's form_valid() method because
        # it might do some processing (in the case of CreateView, it will
        # call form.save() for example).
        response = super().form_valid(form)
        if self.request.accepts('text/html'):
            return response
        else:
            data = {
                'pk': self.object.pk,
            }
            return JsonResponse(data)

class AuthorCreateView(JsonableResponseMixin, CreateView):
    model = Author
    fields = ['name']


以上內(nèi)容是否對(duì)您有幫助:
在線(xiàn)筆記
App下載
App下載

掃描二維碼

下載編程獅App

公眾號(hào)
微信公眾號(hào)

編程獅公眾號(hào)