mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Update to [11025]. This time, actually include the new generic views documentation.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@11026 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		
							
								
								
									
										503
									
								
								docs/topics/generic-views.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										503
									
								
								docs/topics/generic-views.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,503 @@ | |||||||
|  | .. _topics-generic-views: | ||||||
|  |  | ||||||
|  | ============= | ||||||
|  | Generic views | ||||||
|  | ============= | ||||||
|  |  | ||||||
|  | Writing Web applications can be monotonous, because we repeat certain patterns | ||||||
|  | again and again. Django tries to take away some of that monotony at the model | ||||||
|  | and template layers, but Web developers also experience this boredom at the view | ||||||
|  | level. | ||||||
|  |  | ||||||
|  | Django's *generic views* were developed to ease that pain. They take certain | ||||||
|  | common idioms and patterns found in view development and abstract them so that | ||||||
|  | you can quickly write common views of data without having to write too much | ||||||
|  | code. | ||||||
|  |  | ||||||
|  | We can recognize certain common tasks, like displaying a list of objects, and | ||||||
|  | write code that displays a list of *any* object. Then the model in question can | ||||||
|  | be passed as an extra argument to the URLconf. | ||||||
|  |  | ||||||
|  | Django ships with generic views to do the following: | ||||||
|  |  | ||||||
|  |     * Perform common "simple" tasks: redirect to a different page and | ||||||
|  |       render a given template. | ||||||
|  |  | ||||||
|  |     * Display list and detail pages for a single object. If we were creating an | ||||||
|  |       application to manage conferences then a ``talk_list`` view and a | ||||||
|  |       ``registered_user_list`` view would be examples of list views. A single | ||||||
|  |       talk page is an example of what we call a "detail" view. | ||||||
|  |  | ||||||
|  |     * Present date-based objects in year/month/day archive pages, | ||||||
|  |       associated detail, and "latest" pages. The Django Weblog's | ||||||
|  |       (http://www.djangoproject.com/weblog/) year, month, and | ||||||
|  |       day archives are built with these, as would be a typical | ||||||
|  |       newspaper's archives. | ||||||
|  |  | ||||||
|  |     * Allow users to create, update, and delete objects -- with or | ||||||
|  |       without authorization. | ||||||
|  |  | ||||||
|  | Taken together, these views provide easy interfaces to perform the most common | ||||||
|  | tasks developers encounter. | ||||||
|  |  | ||||||
|  | Using generic views | ||||||
|  | =================== | ||||||
|  |  | ||||||
|  | All of these views are used by creating configuration dictionaries in | ||||||
|  | your URLconf files and passing those dictionaries as the third member of the | ||||||
|  | URLconf tuple for a given pattern. | ||||||
|  |  | ||||||
|  | For example, here's a simple URLconf you could use to present a static "about" | ||||||
|  | page:: | ||||||
|  |  | ||||||
|  |     from django.conf.urls.defaults import * | ||||||
|  |     from django.views.generic.simple import direct_to_template | ||||||
|  |  | ||||||
|  |     urlpatterns = patterns('', | ||||||
|  |         ('^about/$', direct_to_template, { | ||||||
|  |             'template': 'about.html' | ||||||
|  |         }) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  | Though this might seem a bit "magical" at first glance  -- look, a view with no | ||||||
|  | code! --, actually the ``direct_to_template`` view simply grabs information from | ||||||
|  | the extra-parameters dictionary and uses that information when rendering the | ||||||
|  | view. | ||||||
|  |  | ||||||
|  | Because this generic view -- and all the others -- is a regular view functions | ||||||
|  | like any other, we can reuse it inside our own views. As an example, let's | ||||||
|  | extend our "about" example to map URLs of the form ``/about/<whatever>/`` to | ||||||
|  | statically rendered ``about/<whatever>.html``. We'll do this by first modifying | ||||||
|  | the URLconf to point to a view function: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     from django.conf.urls.defaults import * | ||||||
|  |     from django.views.generic.simple import direct_to_template | ||||||
|  |     **from mysite.books.views import about_pages** | ||||||
|  |  | ||||||
|  |     urlpatterns = patterns('', | ||||||
|  |         ('^about/$', direct_to_template, { | ||||||
|  |             'template': 'about.html' | ||||||
|  |         }), | ||||||
|  |         **('^about/(\w+)/$', about_pages),** | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  | Next, we'll write the ``about_pages`` view:: | ||||||
|  |  | ||||||
|  |     from django.http import Http404 | ||||||
|  |     from django.template import TemplateDoesNotExist | ||||||
|  |     from django.views.generic.simple import direct_to_template | ||||||
|  |  | ||||||
|  |     def about_pages(request, page): | ||||||
|  |         try: | ||||||
|  |             return direct_to_template(request, template="about/%s.html" % page) | ||||||
|  |         except TemplateDoesNotExist: | ||||||
|  |             raise Http404() | ||||||
|  |  | ||||||
|  | Here we're treating ``direct_to_template`` like any other function. Since it | ||||||
|  | returns an ``HttpResponse``, we can simply return it as-is. The only slightly | ||||||
|  | tricky business here is dealing with missing templates. We don't want a | ||||||
|  | nonexistent template to cause a server error, so we catch | ||||||
|  | ``TemplateDoesNotExist`` exceptions and return 404 errors instead. | ||||||
|  |  | ||||||
|  | .. admonition:: Is there a security vulnerability here? | ||||||
|  |  | ||||||
|  |     Sharp-eyed readers may have noticed a possible security hole: we're | ||||||
|  |     constructing the template name using interpolated content from the browser | ||||||
|  |     (``template="about/%s.html" % page``). At first glance, this looks like a | ||||||
|  |     classic *directory traversal* vulnerability. But is it really? | ||||||
|  |  | ||||||
|  |     Not exactly. Yes, a maliciously crafted value of ``page`` could cause | ||||||
|  |     directory traversal, but although ``page`` *is* taken from the request URL, | ||||||
|  |     not every value will be accepted. The key is in the URLconf: we're using | ||||||
|  |     the regular expression ``\w+`` to match the ``page`` part of the URL, and | ||||||
|  |     ``\w`` only accepts letters and numbers. Thus, any malicious characters | ||||||
|  |     (dots and slashes, here) will be rejected by the URL resolver before they | ||||||
|  |     reach the view itself. | ||||||
|  |  | ||||||
|  | Generic views of objects | ||||||
|  | ======================== | ||||||
|  |  | ||||||
|  | The ``direct_to_template`` certainly is useful, but Django's generic views | ||||||
|  | really shine when it comes to presenting views on your database content. Because | ||||||
|  | it's such a common task, Django comes with a handful of built-in generic views | ||||||
|  | that make generating list and detail views of objects incredibly easy. | ||||||
|  |  | ||||||
|  | Let's take a look at one of these generic views: the "object list" view. We'll | ||||||
|  | be using these models:: | ||||||
|  |  | ||||||
|  |     # models.py | ||||||
|  |     from django.db import models | ||||||
|  |  | ||||||
|  |     class Publisher(models.Model): | ||||||
|  |         name = models.CharField(max_length=30) | ||||||
|  |         address = models.CharField(max_length=50) | ||||||
|  |         city = models.CharField(max_length=60) | ||||||
|  |         state_province = models.CharField(max_length=30) | ||||||
|  |         country = models.CharField(max_length=50) | ||||||
|  |         website = models.URLField() | ||||||
|  |  | ||||||
|  |         def __unicode__(self): | ||||||
|  |             return self.name | ||||||
|  |  | ||||||
|  |         class Meta: | ||||||
|  |             ordering = ["-name"] | ||||||
|  |  | ||||||
|  |     class Book(models.Model): | ||||||
|  |         title = models.CharField(max_length=100) | ||||||
|  |         authors = models.ManyToManyField('Author') | ||||||
|  |         publisher = models.ForeignKey(Publisher) | ||||||
|  |         publication_date = models.DateField() | ||||||
|  |  | ||||||
|  | To build a list page of all books, we'd use a URLconf along these lines:: | ||||||
|  |  | ||||||
|  |     from django.conf.urls.defaults import * | ||||||
|  |     from django.views.generic import list_detail | ||||||
|  |     from mysite.books.models import Publisher | ||||||
|  |  | ||||||
|  |     publisher_info = { | ||||||
|  |         "queryset" : Publisher.objects.all(), | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     urlpatterns = patterns('', | ||||||
|  |         (r'^publishers/$', list_detail.object_list, publisher_info) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  | That's all the Python code we need to write. We still need to write a template, | ||||||
|  | however. We could explicitly tell the ``object_list`` view which template to use | ||||||
|  | by including a ``template_name`` key in the extra arguments dictionary, but in | ||||||
|  | the absence of an explicit template Django will infer one from the object's | ||||||
|  | name. In this case, the inferred template will be | ||||||
|  | ``"books/publisher_list.html"`` -- the "books" part comes from the name of the | ||||||
|  | app that defines the model, while the "publisher" bit is just the lowercased | ||||||
|  | version of the model's name. | ||||||
|  |  | ||||||
|  | .. highlightlang:: html+django | ||||||
|  |  | ||||||
|  | This template will be rendered against a context containing a variable called | ||||||
|  | ``object_list`` that contains all the book objects. A very simple template | ||||||
|  | might look like the following:: | ||||||
|  |  | ||||||
|  |     {% extends "base.html" %} | ||||||
|  |  | ||||||
|  |     {% block content %} | ||||||
|  |         <h2>Publishers</h2> | ||||||
|  |         <ul> | ||||||
|  |             {% for publisher in object_list %} | ||||||
|  |                 <li>{{ publisher.name }}</li> | ||||||
|  |             {% endfor %} | ||||||
|  |         </ul> | ||||||
|  |     {% endblock %} | ||||||
|  |  | ||||||
|  | That's really all there is to it. All the cool features of generic views come | ||||||
|  | from changing the "info" dictionary passed to the generic view. The | ||||||
|  | :ref:`generic views reference<ref-generic-views>` documents all the generic | ||||||
|  | views and all their options in detail; the rest of this document will consider | ||||||
|  | some of the common ways you might customize and extend generic views. | ||||||
|  |  | ||||||
|  | Extending generic views | ||||||
|  | ======================= | ||||||
|  |  | ||||||
|  | .. highlightlang:: python | ||||||
|  |  | ||||||
|  | There's no question that using generic views can speed up development | ||||||
|  | substantially. In most projects, however, there comes a moment when the | ||||||
|  | generic views no longer suffice. Indeed, the most common question asked by new | ||||||
|  | Django developers is how to make generic views handle a wider array of | ||||||
|  | situations. | ||||||
|  |  | ||||||
|  | Luckily, in nearly every one of these cases, there are ways to simply extend | ||||||
|  | generic views to handle a larger array of use cases. These situations usually | ||||||
|  | fall into a handful of patterns dealt with in the sections that follow. | ||||||
|  |  | ||||||
|  | Making "friendly" template contexts | ||||||
|  | ----------------------------------- | ||||||
|  |  | ||||||
|  | You might have noticed that our sample publisher list template stores all the | ||||||
|  | books in a variable named ``object_list``. While this works just fine, it isn't | ||||||
|  | all that "friendly" to template authors: they have to "just know" that they're | ||||||
|  | dealing with books here. A better name for that variable would be | ||||||
|  | ``publisher_list``; that variable's content is pretty obvious. | ||||||
|  |  | ||||||
|  | We can change the name of that variable easily with the ``template_object_name`` | ||||||
|  | argument: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     publisher_info = { | ||||||
|  |         "queryset" : Publisher.objects.all(), | ||||||
|  |         **"template_object_name" : "publisher",** | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     urlpatterns = patterns('', | ||||||
|  |         (r'^publishers/$', list_detail.object_list, publisher_info) | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  | Providing a useful ``template_object_name`` is always a good idea. Your | ||||||
|  | coworkers who design templates will thank you. | ||||||
|  |  | ||||||
|  | Adding extra context | ||||||
|  | -------------------- | ||||||
|  |  | ||||||
|  | Often you simply need to present some extra information beyond that provided by | ||||||
|  | the generic view. For example, think of showing a list of all the other | ||||||
|  | publishers on each publisher detail page. The ``object_detail`` generic view | ||||||
|  | provides the publisher to the context, but it seems there's no way to get a list | ||||||
|  | of *all* publishers in that template. | ||||||
|  |  | ||||||
|  | But there is: all generic views take an extra optional parameter, | ||||||
|  | ``extra_context``. This is a dictionary of extra objects that will be added to | ||||||
|  | the template's context. So, to provide the list of all publishers on the detail | ||||||
|  | detail view, we'd use an info dict like this: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     from mysite.books.models import Publisher, **Book** | ||||||
|  |  | ||||||
|  |     publisher_info = { | ||||||
|  |         "queryset" : Publisher.objects.all(), | ||||||
|  |         "template_object_name" : "publisher", | ||||||
|  |         **"extra_context" : {"book_list" : Book.objects.all()}** | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | This would populate a ``{{ book_list }}`` variable in the template context. | ||||||
|  | This pattern can be used to pass any information down into the template for the | ||||||
|  | generic view. It's very handy. | ||||||
|  |  | ||||||
|  | However, there's actually a subtle bug here -- can you spot it? | ||||||
|  |  | ||||||
|  | The problem has to do with when the queries in ``extra_context`` are evaluated. | ||||||
|  | Because this example puts ``Publisher.objects.all()`` in the URLconf, it will | ||||||
|  | be evaluated only once (when the URLconf is first loaded). Once you add or | ||||||
|  | remove publishers, you'll notice that the generic view doesn't reflect those | ||||||
|  | changes until you reload the Web server (see :ref:`caching-and-querysets` | ||||||
|  | for more information about when QuerySets are cached and evaluated). | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |     This problem doesn't apply to the ``queryset`` generic view argument. Since | ||||||
|  |     Django knows that particular QuerySet should *never* be cached, the generic | ||||||
|  |     view takes care of clearing the cache when each view is rendered. | ||||||
|  |  | ||||||
|  | The solution is to use a callback in ``extra_context`` instead of a value. Any | ||||||
|  | callable (i.e., a function) that's passed to ``extra_context`` will be evaluated | ||||||
|  | when the view is rendered (instead of only once). You could do this with an | ||||||
|  | explicitly defined function: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     def get_books(): | ||||||
|  |         return Book.objects.all() | ||||||
|  |  | ||||||
|  |     publisher_info = { | ||||||
|  |         "queryset" : Publisher.objects.all(), | ||||||
|  |         "template_object_name" : "publisher", | ||||||
|  |         "extra_context" : **{"book_list" : get_books}** | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | or you could use a less obvious but shorter version that relies on the fact that | ||||||
|  | ``Publisher.objects.all`` is itself a callable: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     publisher_info = { | ||||||
|  |         "queryset" : Publisher.objects.all(), | ||||||
|  |         "template_object_name" : "publisher", | ||||||
|  |         "extra_context" : **{"book_list" : Book.objects.all}** | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | Notice the lack of parentheses after ``Book.objects.all``; this references | ||||||
|  | the function without actually calling it (which the generic view will do later). | ||||||
|  |  | ||||||
|  | Viewing subsets of objects | ||||||
|  | -------------------------- | ||||||
|  |  | ||||||
|  | Now let's take a closer look at this ``queryset`` key we've been using all | ||||||
|  | along. Most generic views take one of these ``queryset`` arguments -- it's how | ||||||
|  | the view knows which set of objects to display (see :ref:`topics-db-queries` for | ||||||
|  | more information about ``QuerySet`` objects, and see the | ||||||
|  | :ref:`generic views reference<ref-generic-views>` for the complete details). | ||||||
|  |  | ||||||
|  | To pick a simple example, we might want to order a list of books by | ||||||
|  | publication date, with the most recent first: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     book_info = { | ||||||
|  |         "queryset" : Book.objects.all().order_by("-publication_date"), | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     urlpatterns = patterns('', | ||||||
|  |         (r'^publishers/$', list_detail.object_list, publisher_info), | ||||||
|  |         **(r'^books/$', list_detail.object_list, book_info),** | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  |  | ||||||
|  | That's a pretty simple example, but it illustrates the idea nicely. Of course, | ||||||
|  | you'll usually want to do more than just reorder objects. If you want to | ||||||
|  | present a list of books by a particular publisher, you can use the same | ||||||
|  | technique: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     **acme_books = {** | ||||||
|  |         **"queryset": Book.objects.filter(publisher__name="Acme Publishing"),** | ||||||
|  |         **"template_name" : "books/acme_list.html"** | ||||||
|  |     **}** | ||||||
|  |  | ||||||
|  |     urlpatterns = patterns('', | ||||||
|  |         (r'^publishers/$', list_detail.object_list, publisher_info), | ||||||
|  |         **(r'^books/acme/$', list_detail.object_list, acme_books),** | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  | Notice that along with a filtered ``queryset``, we're also using a custom | ||||||
|  | template name. If we didn't, the generic view would use the same template as the | ||||||
|  | "vanilla" object list, which might not be what we want. | ||||||
|  |  | ||||||
|  | Also notice that this isn't a very elegant way of doing publisher-specific | ||||||
|  | books. If we want to add another publisher page, we'd need another handful of | ||||||
|  | lines in the URLconf, and more than a few publishers would get unreasonable. | ||||||
|  | We'll deal with this problem in the next section. | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |     If you get a 404 when requesting ``/books/acme/``, check to ensure you | ||||||
|  |     actually have a Publisher with the name 'ACME Publishing'.  Generic | ||||||
|  |     views have an ``allow_empty`` parameter for this case.  See the | ||||||
|  |     :ref:`generic views reference<ref-generic-views>` for more details. | ||||||
|  |  | ||||||
|  | Complex filtering with wrapper functions | ||||||
|  | ---------------------------------------- | ||||||
|  |  | ||||||
|  | Another common need is to filter down the objects given in a list page by some | ||||||
|  | key in the URL. Earlier we hard-coded the publisher's name in the URLconf, but | ||||||
|  | what if we wanted to write a view that displayed all the books by some arbitrary | ||||||
|  | publisher? We can "wrap" the ``object_list`` generic view to avoid writing a lot | ||||||
|  | of code by hand. As usual, we'll start by writing a URLconf: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     from mysite.books.views import books_by_publisher | ||||||
|  |  | ||||||
|  |     urlpatterns = patterns('', | ||||||
|  |         (r'^publishers/$', list_detail.object_list, publisher_info), | ||||||
|  |         **(r'^books/(\w+)/$', books_by_publisher),** | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  | Next, we'll write the ``books_by_publisher`` view itself:: | ||||||
|  |  | ||||||
|  |     from django.http import Http404 | ||||||
|  |     from django.views.generic import list_detail | ||||||
|  |     from mysite.books.models import Book, Publisher | ||||||
|  |  | ||||||
|  |     def books_by_publisher(request, name): | ||||||
|  |  | ||||||
|  |         # Look up the publisher (and raise a 404 if it can't be found). | ||||||
|  |         try: | ||||||
|  |             publisher = Publisher.objects.get(name__iexact=name) | ||||||
|  |         except Publisher.DoesNotExist: | ||||||
|  |             raise Http404 | ||||||
|  |  | ||||||
|  |         # Use the object_list view for the heavy lifting. | ||||||
|  |         return list_detail.object_list( | ||||||
|  |             request, | ||||||
|  |             queryset = Book.objects.filter(publisher=publisher), | ||||||
|  |             template_name = "books/books_by_publisher.html", | ||||||
|  |             template_object_name = "books", | ||||||
|  |             extra_context = {"publisher" : publisher} | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  | This works because there's really nothing special about generic views -- they're | ||||||
|  | just Python functions. Like any view function, generic views expect a certain | ||||||
|  | set of arguments and return ``HttpResponse`` objects. Thus, it's incredibly easy | ||||||
|  | to wrap a small function around a generic view that does additional work before | ||||||
|  | (or after; see the next section) handing things off to the generic view. | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |     Notice that in the preceding example we passed the current publisher being | ||||||
|  |     displayed in the ``extra_context``. This is usually a good idea in wrappers | ||||||
|  |     of this nature; it lets the template know which "parent" object is currently | ||||||
|  |     being browsed. | ||||||
|  |  | ||||||
|  | Performing extra work | ||||||
|  | --------------------- | ||||||
|  |  | ||||||
|  | The last common pattern we'll look at involves doing some extra work before | ||||||
|  | or after calling the generic view. | ||||||
|  |  | ||||||
|  | Imagine we had a ``last_accessed`` field on our ``Author`` object that we were | ||||||
|  | using to keep track of the last time anybody looked at that author:: | ||||||
|  |  | ||||||
|  |     # models.py | ||||||
|  |  | ||||||
|  |     class Author(models.Model): | ||||||
|  |         salutation = models.CharField(max_length=10) | ||||||
|  |         first_name = models.CharField(max_length=30) | ||||||
|  |         last_name = models.CharField(max_length=40) | ||||||
|  |         email = models.EmailField() | ||||||
|  |         headshot = models.ImageField(upload_to='/tmp') | ||||||
|  |         last_accessed = models.DateTimeField() | ||||||
|  |  | ||||||
|  | The generic ``object_detail`` view, of course, wouldn't know anything about this | ||||||
|  | field, but once again we could easily write a custom view to keep that field | ||||||
|  | updated. | ||||||
|  |  | ||||||
|  | First, we'd need to add an author detail bit in the URLconf to point to a | ||||||
|  | custom view: | ||||||
|  |  | ||||||
|  | .. parsed-literal:: | ||||||
|  |  | ||||||
|  |     from mysite.books.views import author_detail | ||||||
|  |  | ||||||
|  |     urlpatterns = patterns('', | ||||||
|  |         #... | ||||||
|  |         **(r'^authors/(?P<author_id>\d+)/$', author_detail),** | ||||||
|  |     ) | ||||||
|  |  | ||||||
|  | Then we'd write our wrapper function:: | ||||||
|  |  | ||||||
|  |     import datetime | ||||||
|  |     from mysite.books.models import Author | ||||||
|  |     from django.views.generic import list_detail | ||||||
|  |     from django.shortcuts import get_object_or_404 | ||||||
|  |  | ||||||
|  |     def author_detail(request, author_id): | ||||||
|  |         # Look up the Author (and raise a 404 if she's not found) | ||||||
|  |         author = get_object_or_404(Author, pk=author_id) | ||||||
|  |  | ||||||
|  |         # Record the last accessed date | ||||||
|  |         author.last_accessed = datetime.datetime.now() | ||||||
|  |         author.save() | ||||||
|  |  | ||||||
|  |         # Show the detail page | ||||||
|  |         return list_detail.object_detail( | ||||||
|  |             request, | ||||||
|  |             queryset = Author.objects.all(), | ||||||
|  |             object_id = author_id, | ||||||
|  |         ) | ||||||
|  |  | ||||||
|  | .. note:: | ||||||
|  |  | ||||||
|  |     This code won't actually work unless you create a | ||||||
|  |     ``books/author_detail.html`` template. | ||||||
|  |  | ||||||
|  | We can use a similar idiom to alter the response returned by the generic view. | ||||||
|  | If we wanted to provide a downloadable plain-text version of the list of | ||||||
|  | authors, we could use a view like this:: | ||||||
|  |  | ||||||
|  |     def author_list_plaintext(request): | ||||||
|  |         response = list_detail.object_list( | ||||||
|  |             request, | ||||||
|  |             queryset = Author.objects.all(), | ||||||
|  |             mimetype = "text/plain", | ||||||
|  |             template_name = "books/author_list.txt" | ||||||
|  |         ) | ||||||
|  |         response["Content-Disposition"] = "attachment; filename=authors.txt" | ||||||
|  |         return response | ||||||
|  |  | ||||||
|  | This works because the generic views return simple ``HttpResponse`` objects | ||||||
|  | that can be treated like dictionaries to set HTTP headers. This | ||||||
|  | ``Content-Disposition`` business, by the way, instructs the browser to | ||||||
|  | download and save the page instead of displaying it in the browser. | ||||||
		Reference in New Issue
	
	Block a user