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

Django入门与实践-第22章:基于类的视图的更多相关文章

  1. Django入门与实践-第13章:表单处理(完结)

    http://127.0.0.1:8000/boards/1/ http://127.0.0.1:8000/boards/2/ http://127.0.0.1:8000/boards/3/ http ...

  2. Django入门与实践-第26章:个性化工具(完结)

    http://127.0.0.1:8000/boards/1/topics/62/reply/ 我觉得只添加内置的个性化(humanize)包就会很不错. 它包含一组为数据添加“人性化(human t ...

  3. Django入门与实践-第14章:用户注册(完结)

    http://127.0.0.1:8000/signup/ django-admin startapp accounts INSTALLED_APPS = [ 'accounts', ] # mypr ...

  4. Django入门与实践-第12章:复用模板(完结)

    http://127.0.0.1:8000/http://127.0.0.1:8000/boards/1/http://127.0.0.1:8000/boards/2/http://127.0.0.1 ...

  5. Django入门与实践-第11章:URL 分发(完结)

    http://127.0.0.1:8000http://127.0.0.1:8000/boards/1/http://127.0.0.1:8000/boards/2/http://127.0.0.1: ...

  6. Django入门与实践-第25章:Markdown 支持(完结)

    http://127.0.0.1:8000/boards/1/topics/102/reply/ 让我们在文本区域添加 Markdown 支持来改善用户体验. 你会看到要实现这个功能非常简单. 首先, ...

  7. Django入门与实践-第24章:我的账户视图(完结)

    http://127.0.0.1:8000/settings/account/ #好的,那么,这部分将是我们最后的一个视图.之后,我们将专心来改进现有功能. #accounts/views.py fr ...

  8. Django入门与实践-第23章:分页实现(完结)

    http://127.0.0.1:8000/boards/1/ #从现在起,我们将在 board_topics 这个视图中来操作. python manage.py shell from django ...

  9. Django入门与实践-第20章:QuerySets(查询结果集)(完结)

    http://127.0.0.1:8000/boards/1/ #boards/models.py from django.utils.text import Truncator class Topi ...

随机推荐

  1. win10 QQ远程协助部分界面点不了

    win10 QQ远程协助部分界面点不了. 把对方电脑的电脑管家全部退出,退出了也不行. 是win10的防火墙?安全策略?

  2. WebRTC内置debug工具,详细参数解读 chrome://webrtc-internals/

    为了确保这篇文章所写内容尽可能的准确,我决定请来Philipp Hancke来作为此篇文章的共同作者. 当你想要找到你WebRTC产品中的问题时,webrtc-internals是一个非常棒的工具,因 ...

  3. Tocmat 启动错误 Port 8005 required by tomcat v7.0 server at localhost is already in use

    1: netstat -ano!findstr 8005 2: taskkill /pid 6476 /f

  4. iOS toll-free bridge

    https://developer.apple.com/library/ios/documentation/CoreFoundation/Conceptual/CFDesignConcepts/Art ...

  5. 【转】.net 实现 语音搜索(仅限WebKit内核浏览器)

    <input type="text" class="text" name="value_2" id="value_2&quo ...

  6. iKcamp|基于Koa2搭建Node.js实战(含视频)☞ 规范与部署

    沪江CCtalk视频地址:https://www.cctalk.com/v/15114923889450 规范与部署 懒人推动社会进步. 本篇中,我们会讲述三个知识点 定制书写规范 开发环境运行 如何 ...

  7. Windows自带的端口转发工具netsh使用方法_DOS/BAT

    Windows自带的端口转发工具netsh使用方法_DOS/BAT   作者:用户 来源:互联网 时间:2017-02-22 17:24:30 netsh 端口转发 摘要: 下面的代码在windows ...

  8. Windows10系统一键结束所有运行程序

    当电脑及其卡顿的时候,想打开任务管理器关掉所有运行的程序的时候,也会变得及其困难.因此之前你如果有犀利的小程序设置,这都不会是问题. 1)空白处右键-新建-快捷方式 2)将下列代码复制到下列框中(注意 ...

  9. Typora快捷键的使用

    无序列表:输入-之后输入空格 有序列表:输入数字+“.”之后输入空格 任务列表:-[空格]空格 文字 标题:ctrl+数字 表格:ctrl+t 生成目录:[TOC]按回车 选中一整行:ctrl+l 选 ...

  10. 大型运输行业实战_day09_2_站间互售实现

    1.添加站间互售入口 对应的html代码 <button onclick="otherStation()">站间互售</button> 对应的js发送函数 ...