2012-06-11 10:34:00 +02:00
|
|
|
=================
|
|
|
|
Class-based views
|
|
|
|
=================
|
|
|
|
|
|
|
|
A view is a callable which takes a request and returns a
|
|
|
|
response. This can be more than just a function, and Django provides
|
|
|
|
an example of some classes which can be used as views. These allow you
|
|
|
|
to structure your views and reuse code by harnessing inheritance and
|
2019-06-17 16:54:55 +02:00
|
|
|
mixins. There are also some generic views for tasks which we'll get to later,
|
|
|
|
but you may want to design your own structure of reusable views which suits
|
|
|
|
your use case. For full details, see the :doc:`class-based views reference
|
|
|
|
documentation</ref/class-based-views/index>`.
|
2012-06-11 10:34:00 +02:00
|
|
|
|
|
|
|
.. toctree::
|
|
|
|
:maxdepth: 1
|
|
|
|
|
2013-02-27 07:12:37 -05:00
|
|
|
intro
|
2012-06-11 10:34:00 +02:00
|
|
|
generic-display
|
|
|
|
generic-editing
|
|
|
|
mixins
|
|
|
|
|
|
|
|
Basic examples
|
|
|
|
==============
|
|
|
|
|
|
|
|
Django provides base view classes which will suit a wide range of applications.
|
|
|
|
All views inherit from the :class:`~django.views.generic.base.View` class, which
|
2019-12-18 11:36:27 +01:00
|
|
|
handles linking the view into the URLs, HTTP method dispatching and other
|
2019-06-17 16:54:55 +02:00
|
|
|
common features. :class:`~django.views.generic.base.RedirectView` provides a
|
|
|
|
HTTP redirect, and :class:`~django.views.generic.base.TemplateView` extends the
|
|
|
|
base class to make it also render a template.
|
2012-06-11 10:34:00 +02:00
|
|
|
|
|
|
|
|
2019-06-17 16:54:55 +02:00
|
|
|
Usage in your URLconf
|
|
|
|
=====================
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2019-06-17 16:54:55 +02:00
|
|
|
The most direct way to use generic views is to create them directly in your
|
|
|
|
URLconf. If you're only changing a few attributes on a class-based view, you
|
|
|
|
can pass them into the :meth:`~django.views.generic.base.View.as_view` method
|
|
|
|
call itself::
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2016-10-20 19:29:04 +02:00
|
|
|
from django.urls import path
|
2012-06-11 10:34:00 +02:00
|
|
|
from django.views.generic import TemplateView
|
|
|
|
|
2014-04-01 20:46:34 -04:00
|
|
|
urlpatterns = [
|
2023-02-28 20:53:28 +01:00
|
|
|
path("about/", TemplateView.as_view(template_name="about.html")),
|
2014-04-01 20:46:34 -04:00
|
|
|
]
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2012-11-21 19:06:17 -05:00
|
|
|
Any arguments passed to :meth:`~django.views.generic.base.View.as_view` will
|
|
|
|
override attributes set on the class. In this example, we set ``template_name``
|
|
|
|
on the ``TemplateView``. A similar overriding pattern can be used for the
|
|
|
|
``url`` attribute on :class:`~django.views.generic.base.RedirectView`.
|
2012-06-11 10:34:00 +02:00
|
|
|
|
|
|
|
|
2012-08-25 08:11:56 -04:00
|
|
|
Subclassing generic views
|
|
|
|
=========================
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2012-08-25 08:11:56 -04:00
|
|
|
The second, more powerful way to use generic views is to inherit from an
|
|
|
|
existing view and override attributes (such as the ``template_name``) or
|
|
|
|
methods (such as ``get_context_data``) in your subclass to provide new values
|
|
|
|
or methods. Consider, for example, a view that just displays one template,
|
|
|
|
``about.html``. Django has a generic view to do this -
|
2019-06-17 16:54:55 +02:00
|
|
|
:class:`~django.views.generic.base.TemplateView` - so we can subclass it, and
|
|
|
|
override the template name::
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2012-08-25 08:11:56 -04:00
|
|
|
# some_app/views.py
|
|
|
|
from django.views.generic import TemplateView
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2023-02-28 20:53:28 +01:00
|
|
|
|
2012-08-25 08:11:56 -04:00
|
|
|
class AboutView(TemplateView):
|
|
|
|
template_name = "about.html"
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2019-06-17 16:54:55 +02:00
|
|
|
Then we need to add this new view into our URLconf.
|
|
|
|
:class:`~django.views.generic.base.TemplateView` is a class, not a function, so
|
|
|
|
we point the URL to the :meth:`~django.views.generic.base.View.as_view` class
|
|
|
|
method instead, which provides a function-like entry to class-based views::
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2012-08-25 08:11:56 -04:00
|
|
|
# urls.py
|
2016-10-20 19:29:04 +02:00
|
|
|
from django.urls import path
|
2012-08-25 08:11:56 -04:00
|
|
|
from some_app.views import AboutView
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2014-04-01 20:46:34 -04:00
|
|
|
urlpatterns = [
|
2023-02-28 20:53:28 +01:00
|
|
|
path("about/", AboutView.as_view()),
|
2014-04-01 20:46:34 -04:00
|
|
|
]
|
2012-06-11 10:34:00 +02:00
|
|
|
|
|
|
|
|
|
|
|
For more information on how to use the built in generic views, consult the next
|
2015-08-19 02:01:36 +06:00
|
|
|
topic on :doc:`generic class-based views</topics/class-based-views/generic-display>`.
|
2012-06-11 10:34:00 +02:00
|
|
|
|
2012-09-08 18:45:02 -04:00
|
|
|
.. _supporting-other-http-methods:
|
|
|
|
|
|
|
|
Supporting other HTTP methods
|
|
|
|
-----------------------------
|
|
|
|
|
|
|
|
Suppose somebody wants to access our book library over HTTP using the views
|
|
|
|
as an API. The API client would connect every now and then and download book
|
|
|
|
data for the books published since last visit. But if no new books appeared
|
|
|
|
since then, it is a waste of CPU time and bandwidth to fetch the books from the
|
|
|
|
database, render a full response and send it to the client. It might be
|
|
|
|
preferable to ask the API when the most recent book was published.
|
|
|
|
|
|
|
|
We map the URL to book list view in the URLconf::
|
|
|
|
|
2016-10-20 19:29:04 +02:00
|
|
|
from django.urls import path
|
2012-09-08 18:45:02 -04:00
|
|
|
from books.views import BookListView
|
|
|
|
|
2014-04-01 20:46:34 -04:00
|
|
|
urlpatterns = [
|
2023-02-28 20:53:28 +01:00
|
|
|
path("books/", BookListView.as_view()),
|
2014-04-01 20:46:34 -04:00
|
|
|
]
|
2012-09-08 18:45:02 -04:00
|
|
|
|
|
|
|
And the view::
|
|
|
|
|
|
|
|
from django.http import HttpResponse
|
|
|
|
from django.views.generic import ListView
|
|
|
|
from books.models import Book
|
|
|
|
|
2023-02-28 20:53:28 +01:00
|
|
|
|
2012-09-08 18:45:02 -04:00
|
|
|
class BookListView(ListView):
|
|
|
|
model = Book
|
|
|
|
|
|
|
|
def head(self, *args, **kwargs):
|
2023-02-28 20:53:28 +01:00
|
|
|
last_book = self.get_queryset().latest("publication_date")
|
2020-09-15 12:43:37 +02:00
|
|
|
response = HttpResponse(
|
|
|
|
# RFC 1123 date format.
|
2023-02-28 20:53:28 +01:00
|
|
|
headers={
|
|
|
|
"Last-Modified": last_book.publication_date.strftime(
|
|
|
|
"%a, %d %b %Y %H:%M:%S GMT"
|
|
|
|
)
|
|
|
|
},
|
2020-09-15 12:43:37 +02:00
|
|
|
)
|
2012-09-08 18:45:02 -04:00
|
|
|
return response
|
|
|
|
|
2019-06-17 16:54:55 +02:00
|
|
|
If the view is accessed from a ``GET`` request, an object list is returned in
|
|
|
|
the response (using the ``book_list.html`` template). But if the client issues
|
|
|
|
a ``HEAD`` request, the response has an empty body and the ``Last-Modified``
|
|
|
|
header indicates when the most recent book was published. Based on this
|
|
|
|
information, the client may or may not download the full object list.
|
2022-04-07 07:05:59 +02:00
|
|
|
|
|
|
|
.. _async-class-based-views:
|
|
|
|
|
|
|
|
Asynchronous class-based views
|
|
|
|
==============================
|
|
|
|
|
|
|
|
As well as the synchronous (``def``) method handlers already shown, ``View``
|
|
|
|
subclasses may define asynchronous (``async def``) method handlers to leverage
|
|
|
|
asynchronous code using ``await``::
|
|
|
|
|
|
|
|
import asyncio
|
|
|
|
from django.http import HttpResponse
|
|
|
|
from django.views import View
|
|
|
|
|
2023-02-28 20:53:28 +01:00
|
|
|
|
2022-04-07 07:05:59 +02:00
|
|
|
class AsyncView(View):
|
|
|
|
async def get(self, request, *args, **kwargs):
|
|
|
|
# Perform io-blocking view logic using await, sleep for example.
|
|
|
|
await asyncio.sleep(1)
|
|
|
|
return HttpResponse("Hello async world!")
|
|
|
|
|
|
|
|
Within a single view-class, all user-defined method handlers must be either
|
|
|
|
synchronous, using ``def``, or all asynchronous, using ``async def``. An
|
|
|
|
``ImproperlyConfigured`` exception will be raised in ``as_view()`` if ``def``
|
|
|
|
and ``async def`` declarations are mixed.
|
|
|
|
|
|
|
|
Django will automatically detect asynchronous views and run them in an
|
|
|
|
asynchronous context. You can read more about Django's asynchronous support,
|
|
|
|
and how to best use async views, in :doc:`/topics/async`.
|