generic-editing.txt 9.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272
  1. Form handling with class-based views
  2. ====================================
  3. Form processing generally has 3 paths:
  4. * Initial GET (blank or prepopulated form)
  5. * POST with invalid data (typically redisplay form with errors)
  6. * POST with valid data (process the data and typically redirect)
  7. Implementing this yourself often results in a lot of repeated boilerplate code
  8. (see :ref:`Using a form in a view<using-a-form-in-a-view>`). To help avoid
  9. this, Django provides a collection of generic class-based views for form
  10. processing.
  11. Basic Forms
  12. -----------
  13. Given a simple contact form:
  14. .. snippet::
  15. :filename: forms.py
  16. from django import forms
  17. class ContactForm(forms.Form):
  18. name = forms.CharField()
  19. message = forms.CharField(widget=forms.Textarea)
  20. def send_email(self):
  21. # send email using the self.cleaned_data dictionary
  22. pass
  23. The view can be constructed using a ``FormView``:
  24. .. snippet::
  25. :filename: views.py
  26. from myapp.forms import ContactForm
  27. from django.views.generic.edit import FormView
  28. class ContactView(FormView):
  29. template_name = 'contact.html'
  30. form_class = ContactForm
  31. success_url = '/thanks/'
  32. def form_valid(self, form):
  33. # This method is called when valid form data has been POSTed.
  34. # It should return an HttpResponse.
  35. form.send_email()
  36. return super(ContactView, self).form_valid(form)
  37. Notes:
  38. * FormView inherits
  39. :class:`~django.views.generic.base.TemplateResponseMixin` so
  40. :attr:`~django.views.generic.base.TemplateResponseMixin.template_name`
  41. can be used here.
  42. * The default implementation for
  43. :meth:`~django.views.generic.edit.FormMixin.form_valid` simply
  44. redirects to the :attr:`~django.views.generic.edit.FormMixin.success_url`.
  45. Model Forms
  46. -----------
  47. Generic views really shine when working with models. These generic
  48. views will automatically create a :class:`~django.forms.ModelForm`, so long as
  49. they can work out which model class to use:
  50. * If the :attr:`~django.views.generic.edit.ModelFormMixin.model` attribute is
  51. given, that model class will be used.
  52. * If :meth:`~django.views.generic.detail.SingleObjectMixin.get_object()`
  53. returns an object, the class of that object will be used.
  54. * If a :attr:`~django.views.generic.detail.SingleObjectMixin.queryset` is
  55. given, the model for that queryset will be used.
  56. Model form views provide a
  57. :meth:`~django.views.generic.edit.ModelFormMixin.form_valid()` implementation
  58. that saves the model automatically. You can override this if you have any
  59. special requirements; see below for examples.
  60. You don't even need to provide a ``success_url`` for
  61. :class:`~django.views.generic.edit.CreateView` or
  62. :class:`~django.views.generic.edit.UpdateView` - they will use
  63. :meth:`~django.db.models.Model.get_absolute_url()` on the model object if available.
  64. If you want to use a custom :class:`~django.forms.ModelForm` (for instance to
  65. add extra validation) simply set
  66. :attr:`~django.views.generic.edit.FormMixin.form_class` on your view.
  67. .. note::
  68. When specifying a custom form class, you must still specify the model,
  69. even though the :attr:`~django.views.generic.edit.FormMixin.form_class` may
  70. be a :class:`~django.forms.ModelForm`.
  71. First we need to add :meth:`~django.db.models.Model.get_absolute_url()` to our
  72. ``Author`` class:
  73. .. snippet::
  74. :filename: models.py
  75. from django.core.urlresolvers import reverse
  76. from django.db import models
  77. class Author(models.Model):
  78. name = models.CharField(max_length=200)
  79. def get_absolute_url(self):
  80. return reverse('author-detail', kwargs={'pk': self.pk})
  81. Then we can use :class:`CreateView` and friends to do the actual
  82. work. Notice how we're just configuring the generic class-based views
  83. here; we don't have to write any logic ourselves:
  84. .. snippet::
  85. :filename: views.py
  86. from django.views.generic.edit import CreateView, UpdateView, DeleteView
  87. from django.core.urlresolvers import reverse_lazy
  88. from myapp.models import Author
  89. class AuthorCreate(CreateView):
  90. model = Author
  91. fields = ['name']
  92. class AuthorUpdate(UpdateView):
  93. model = Author
  94. fields = ['name']
  95. class AuthorDelete(DeleteView):
  96. model = Author
  97. success_url = reverse_lazy('author-list')
  98. .. note::
  99. We have to use :func:`~django.core.urlresolvers.reverse_lazy` here, not
  100. just ``reverse`` as the urls are not loaded when the file is imported.
  101. The ``fields`` attribute works the same way as the ``fields`` attribute on the
  102. inner ``Meta`` class on :class:`~django.forms.ModelForm`. Unless you define the
  103. form class in another way, the attribute is required and the view will raise
  104. an :exc:`~django.core.exceptions.ImproperlyConfigured` exception if it's not.
  105. If you specify both the :attr:`~django.views.generic.edit.ModelFormMixin.fields`
  106. and :attr:`~django.views.generic.edit.FormMixin.form_class` attributes, an
  107. :exc:`~django.core.exceptions.ImproperlyConfigured` exception will be raised.
  108. .. versionchanged:: 1.8
  109. Omitting the ``fields`` attribute was previously allowed and resulted in a
  110. form with all of the model's fields.
  111. .. versionchanged:: 1.8
  112. Previously if both ``fields`` and ``form_class`` were specified,
  113. ``fields`` was silently ignored.
  114. Finally, we hook these new views into the URLconf:
  115. .. snippet::
  116. :filename: urls.py
  117. from django.conf.urls import url
  118. from myapp.views import AuthorCreate, AuthorUpdate, AuthorDelete
  119. urlpatterns = [
  120. # ...
  121. url(r'author/add/$', AuthorCreate.as_view(), name='author_add'),
  122. url(r'author/(?P<pk>[0-9]+)/$', AuthorUpdate.as_view(), name='author_update'),
  123. url(r'author/(?P<pk>[0-9]+)/delete/$', AuthorDelete.as_view(), name='author_delete'),
  124. ]
  125. .. note::
  126. These views inherit
  127. :class:`~django.views.generic.detail.SingleObjectTemplateResponseMixin`
  128. which uses
  129. :attr:`~django.views.generic.detail.SingleObjectTemplateResponseMixin.template_name_suffix`
  130. to construct the
  131. :attr:`~django.views.generic.base.TemplateResponseMixin.template_name`
  132. based on the model.
  133. In this example:
  134. * :class:`CreateView` and :class:`UpdateView` use ``myapp/author_form.html``
  135. * :class:`DeleteView` uses ``myapp/author_confirm_delete.html``
  136. If you wish to have separate templates for :class:`CreateView` and
  137. :class:`UpdateView`, you can set either
  138. :attr:`~django.views.generic.base.TemplateResponseMixin.template_name` or
  139. :attr:`~django.views.generic.detail.SingleObjectTemplateResponseMixin.template_name_suffix`
  140. on your view class.
  141. Models and request.user
  142. -----------------------
  143. To track the user that created an object using a :class:`CreateView`,
  144. you can use a custom :class:`~django.forms.ModelForm` to do this. First, add
  145. the foreign key relation to the model:
  146. .. snippet::
  147. :filename: models.py
  148. from django.contrib.auth.models import User
  149. from django.db import models
  150. class Author(models.Model):
  151. name = models.CharField(max_length=200)
  152. created_by = models.ForeignKey(User, on_delete=models.CASCADE)
  153. # ...
  154. In the view, ensure that you don't include ``created_by`` in the list of fields
  155. to edit, and override
  156. :meth:`~django.views.generic.edit.ModelFormMixin.form_valid()` to add the user:
  157. .. snippet::
  158. :filename: views.py
  159. from django.views.generic.edit import CreateView
  160. from myapp.models import Author
  161. class AuthorCreate(CreateView):
  162. model = Author
  163. fields = ['name']
  164. def form_valid(self, form):
  165. form.instance.created_by = self.request.user
  166. return super(AuthorCreate, self).form_valid(form)
  167. Note that you'll need to :ref:`decorate this
  168. view<decorating-class-based-views>` using
  169. :func:`~django.contrib.auth.decorators.login_required`, or
  170. alternatively handle unauthorized users in the
  171. :meth:`~django.views.generic.edit.ModelFormMixin.form_valid()`.
  172. AJAX example
  173. ------------
  174. Here is a simple example showing how you might go about implementing a form that
  175. works for AJAX requests as well as 'normal' form POSTs::
  176. from django.http import JsonResponse
  177. from django.views.generic.edit import CreateView
  178. from myapp.models import Author
  179. class AjaxableResponseMixin(object):
  180. """
  181. Mixin to add AJAX support to a form.
  182. Must be used with an object-based FormView (e.g. CreateView)
  183. """
  184. def form_invalid(self, form):
  185. response = super(AjaxableResponseMixin, self).form_invalid(form)
  186. if self.request.is_ajax():
  187. return JsonResponse(form.errors, status=400)
  188. else:
  189. return response
  190. def form_valid(self, form):
  191. # We make sure to call the parent's form_valid() method because
  192. # it might do some processing (in the case of CreateView, it will
  193. # call form.save() for example).
  194. response = super(AjaxableResponseMixin, self).form_valid(form)
  195. if self.request.is_ajax():
  196. data = {
  197. 'pk': self.object.pk,
  198. }
  199. return JsonResponse(data)
  200. else:
  201. return response
  202. class AuthorCreate(AjaxableResponseMixin, CreateView):
  203. model = Author
  204. fields = ['name']