Django入门与实践-第22章:基于类的视图
#boards/views.py from django.views.generic import UpdateView from django.utils import timezone class PostUpdateView(UpdateView): model = Post fields = ('message', ) template_name = 'edit_post.html' pk_url_kwarg = 'post_pk' context_object_name = 'post' def form_valid(self, form): post = form.save(commit=False) post.updated_by = self.request.user post.updated_at = timezone.now() post.save() return redirect('topic_posts', pk=post.topic.board.pk, topic_pk=post.topic.pk) #myproject/urls.py url(r'^boards/(?P<pk>\d+)/topics/(?P<topic_pk>\d+)/posts/(?P<post_pk>\d+)/edit/$', views.PostUpdateView.as_view(), name='edit_post'), <!--templates/topic_posts.html--> {% if post.created_by == user %} <div class="mt-3"> <a href="{% url 'edit_post' post.topic.board.pk post.topic.pk post.pk %}" class="btn btn-primary btn-sm"role="button">Edit </a> </div> {% endif %} <!--templates/edit_post.html--> {% extends 'base.html' %} {% block title %}Edit post{% endblock %} {% block breadcrumb %} <li class="breadcrumb-item"><a href="{% url 'home' %}">Boards</a></li> <li class="breadcrumb-item"><a href="{% url 'board_topics' post.topic.board.pk %}">{{ post.topic.board.name }}</a></li> <li class="breadcrumb-item"><a href="{% url 'topic_posts' post.topic.board.pk post.topic.pk %}">{{ post.topic.subject }}</a></li> <li class="breadcrumb-item active">Edit post</li> {% endblock %} {% block content %} <form method="post" class="mb-4" novalidate> {% csrf_token %} {% include 'includes/form.html' %} <button type="submit" class="btn btn-success">Save changes</button> <a href="{% url 'topic_posts' post.topic.board.pk post.topic.pk %}" class="btn btn-outline-secondary" role="button">Cancel</a> </form> {% endblock %}
首先,我们修复 @login_required 装饰器的问题。
在 CBV 上使用视图装饰器的方式有一些不同。我们需要额外的导入:
#boards/views.py
from django.contrib.auth.decorators import login_required
from django.shortcuts import redirect
from django.views.generic import UpdateView
from django.utils import timezone
from django.utils.decorators import method_decorator
from .models import Post
@method_decorator(login_required, name='dispatch')
class PostUpdateView(UpdateView):
model = Post
fields = ('message', )
template_name = 'edit_post.html'
pk_url_kwarg = 'post_pk'
context_object_name = 'post'
def form_valid(self, form):
post = form.save(commit=False)
post.updated_by = self.request.user
post.updated_at = timezone.now()
post.save()
return redirect('topic_posts', pk=post.topic.board.pk, topic_pk=post.topic.pk)
#好的,我们解决了 @login_required 的问题,现在我们必须处理其他用户可以编辑所有帖子的问题。
#解决这个问题最简单的方法是重写UpdateView的 get_queryset 方法。
#boards/views.py
@method_decorator(login_required, name='dispatch')
class PostUpdateView(UpdateView):
model = Post
fields = ('message', )
template_name = 'edit_post.html'
pk_url_kwarg = 'post_pk'
context_object_name = 'post'
def get_queryset(self):
queryset = super().get_queryset()
return queryset.filter(created_by=self.request.user)
def form_valid(self, form):
post = form.save(commit=False)
post.updated_by = self.request.user
post.updated_at = timezone.now()
post.save()
return redirect('topic_posts', pk=post.topic.board.pk, topic_pk=post.topic.pk)
#我们现在可以利用 CBV 的功能来重构这些现有的视图。以主页为例,我们就先从数据库中抓取所有的 boards 并将其罗列在HTML中:
#boards/views.py
from django.views.generic import ListView
class BoardListView(ListView):
model = Board
context_object_name = 'boards'
template_name = 'home.html'
#那么我们得修改一下 urls.py 模块中的引用:
#myproject/urls.py
url(r'^$', views.BoardListView.as_view(), name='home'),
到头来,所有的Django视图其实都是函数。即便是CBV。 在类的底层,它完成了所有的功能并最终返回一个视图函数。 FBV是Django视图中最简单的表示:它就是一个接收 HttpRequest 对象并返回一个 HttpResponse 的函数。 CBV是每个Django视图都被定义为⼀个扩展 django.views.generic.view抽象类后的 Python 类。 GCBV是可以解决一些具体问题的内置的CBV集合,比如列表视图,创建,更新和删除视图等。 from django.views.generic import View class NewPostView(View): def post(self, request): form = PostForm(request.POST) if form.is_valid(): form.save() return redirect('post_list') return render(request, 'new_post.html', {'form': form}) def get(self, request): form = PostForm() return render(request, 'new_post.html', {'form': form}) 我们在 urls.py 模块中引用CBV的方式也会有点不同: urlpatterns = [ url(r'^new_post/$', views.NewPostView.as_view(), name='new_post'), ] 在这里,我们需要调用 as_view() 这个类方法,它将返回⼀个符合url模式视图函数。 还可以创建一些通用视图来完成这些任务,以便我们可以在整个项目中重复利用它。 你需要知道的关于CBV的基本就是这些。就这么简单。
基于类的通过视图(GCBV)接下来关于GCBV。 这是一个不同的情况。正如我前面提到的,GCBV是内置CBV的常见用例。 它们的实现往往会大量使用多重继承(混合继承)和其他面向对象的策略。 #views.py from django.views.generic import CreateView class NewPostView(CreateView): model = Post form_class = PostForm success_url = reverse_lazy('post_list') template_name = 'new_post.html' 这里我们使用了一个用于创建模型对象的通用视图。 它会处理所有表单处理并在表单有效时保存对象。 #boards/views.py from django.shortcuts import redirect from django.views.generic import UpdateView from django.utils import timezone class PostUpdateView(UpdateView): model = Post fields = ('message', ) template_name = 'edit_post.html' pk_url_kwarg = 'post_pk' context_object_name = 'post' def form_valid(self, form): post = form.save(commit=False) post.updated_by = self.request.user post.updated_at = timezone.now() post.save() return redirect('topic_posts', pk=post.topic.board.pk, topic_pk=post.topic.pk) 首先,我们修复 @login_required 装饰器的问题。 在 CBV 上使用视图装饰器的方式有一些不同。我们需要额外的导入: #boards/views.py from django.contrib.auth.decorators import login_required from django.shortcuts import redirect from django.views.generic import UpdateView from django.utils import timezone from django.utils.decorators import method_decorator from .models import Post @method_decorator(login_required, name='dispatch') class PostUpdateView(UpdateView): model = Post fields = ('message', ) template_name = 'edit_post.html' pk_url_kwarg = 'post_pk' context_object_name = 'post' def form_valid(self, form): post = form.save(commit=False) post.updated_by = self.request.user post.updated_at = timezone.now() post.save() return redirect('topic_posts', pk=post.topic.board.pk, topic_pk=post.topic.pk) #列表视图 我们现在可以利用 CBV 的功能来重构一些现有的视图。 以主页为例,我们就先从数据库中抓取所有的 boards 并将其罗列在HTML中: #boards/views.py from django.shortcuts import render from .models import Board def home(request): boards = Board.objects.all() return render(request, 'home.html', {'boards': boards}) 下面是我们如何使用 GCBV 为模型列表来重写它: #boards/views.py from django.views.generic import ListView from .models import Board class BoardListView(ListView): model = Board context_object_name = 'boards' template_name = 'home.html' 那么我们得修改一下 urls.py 模块中的引用: #myproject/urls.py from django.conf.urls import url from boards import views urlpatterns = [ url(r'^$', views.BoardListView.as_view(), name='home'), # ... ]