Django评论系统

目录

创建评论模型

将新模型添加到管理站点

创建一个表单来提交评论和验证输入数据  

显示评论总数和评论列表

创建评论表单,当表单成功提交时,显示一条成功消息


创建评论模型

blog/models.py

class Comment(models.Model):post = models.ForeignKey(Post,on_delete=models.CASCADE,related_name='comments')name = models.CharField(max_length=80)email = models.EmailField()body = models.TextField()created = models.DateTimeField(auto_now_add=True)updated = models.DateTimeField(auto_now=True)active = models.BooleanField(default=True)class Meta:ordering = ('created',)def __str__(self):return 'Comment by {} on {}'.format(self.name, self.post)

📌related_name让我们可以使用属性命名相互关联的对象。通过Post.comments.all()检索文章所有评论。如果没有定义related_name属性,Django将使用小写的模型名,后面跟着_set (comment_set)

更新数据库

python .\\manage.py makemigrations blog 
python .\\manage.py migrate

将新模型添加到管理站点

from .models import Post,Comment#...@admin.register(Comment)
class CommentAdmin(admin.ModelAdmin):list_display = ('name','email','post','created','active')list_filter = ('active','created','updated')search_fields = ('name','email','body')

模型关联 API 用法示例 | Django 文档 | Django (djangoproject.com)

创建一个表单来提交评论和验证输入数据  

Django有两个基类来构建表单:Form和ModelForm。之前使用了第一种方法让您的用户通过电子邮件共享帖子。关于Form类更多内容,请看Django发送QQ邮件-CSDN博客

在本例中,使用  ModelForm,因为必须从Comment模型动态地构建表单。
blog/forms.py

from django import forms
from .models import Comment#...class CommentForm(forms.ModelForm):class Meta:model = Commentfields = ('name','email','body')

添加一个视图来处理表单并将新注释保存到数据库中  
编辑blog/views.py

from .forms import EmailPostForm,CommentForm
#...def post_detail(request,year,month,day,post):post = get_object_or_404(Post,slug=post,status='published',publish__year=year,publish__month=month,publish__day=day)comments = post.comments.filter(active=True)new_comment = Noneif request.method == 'POST':comment_form = CommentForm(data=request.POST)if comment_form.is_valid():new_comment = comment_form.save(commit=False)new_comment.post = postnew_comment.save()else:comment_form = CommentForm()template = "blog/post/detail.html"context = {"post":post,"comments":comments,"new_comment":new_comment,"comment_form":comment_form}return render(request,template,context)

将new_comment变量设置为None。将在创建新注释时使用该变量。如果视图被GET请求调用,用comment_form = CommentForm()构建一个表单实例。如果请求是POST,使用提交的数据实例化表单,并使用is_valid()方法验证它。如果表单无效,则呈现带有验证错误的模板。

如果表单有效,通过调用表单的save()方法创建一个新的Comment对象,并将其赋值给new_comment变量

  • new_comment = comment_form.save(commit=False),save()方法创建表单链接到的模型实例,并将其保存到数据库中。如果使用commit=False,则创建了模型实例,但还没有将其保存到数据库中。
  • new_comment.post = post,当把当前的文章分配给新建的评论(post)。
  • new_comment.save(),写入数据库。

📌save()方法可用于ModelForm,但不能用于Form实例,因为它们没有链接到任何模型。

显示评论总数和评论列表

显示评论总数
/blog/templates/post/detail.html

    {% with comments.count as total_comments %}<h2>{{ total_comments }} comment {{ total_comments|pluralize  }}</h2>{% endwith%}

我们在模板中使用Django ORM,执行QuerySet comments.count()。

📌注意Django模板语言不使用圆括号来调用方法。


{% with %}标签允许我们为一个新变量赋值,该变量将在{% endwith %}标签之前可用。

📌{% with %}模板标签对于避免多次访问数据库或访问昂贵的方法非常有用。

显示评论列表
/blog/templates/post/detail.html

    {% for comment in comments %}<div class="comment"><p class="info">Comment {{ forloop.counter}} by {{ comment.name}}{{ comment.created }}</p>{{ comment.body|linebreaks  }}</div>{% empty %}<p>There are no comments yet.</p>{% endfor %}

使用{% for %}模板标签来循环遍历评论。如果评论列表为空,我们将显示一条默认消息,通知用户这篇文章还没有评论。{{forloop.counter}}包含每次迭代的循环计数器。然后,显示发布评论的用户的姓名、日期和评论正文。

创建评论表单,当表单成功提交时,显示一条成功消息

/blog/templates/post/detail.html

    {% if new_comment %}<h2>Your commet has been added.</h2>{% else %}<h2>Add a new comment</h2><form action="." method="post">{{ comment_form.as_p}}{% csrf_token %}<p><input type="submit" value="Add comment"></p></form>{% endif %}

如果new_comment对象存在,我们将显示一条成功消息,因为注释已成功创建。否则,将为评论模型每个字段呈现带有段落<p>元素的表单,并包含POST请求所需的CSRF令牌。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/593500.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

apollo自动录bag包脚本(包含清理)

需求 制作一个apollo自动录包脚本&#xff0c;类似行车记录仪方便出问题定位。 该脚本实现以下功能&#xff1a; 自动创建录制文件夹创建以日期为命名的文件夹录制bag包全部录制或去除点云等数据量比较大的话题进行简单录制设置bag包最大占用空间每隔一分钟自动计算该文件夹占…

php之 校验多个时间段是否重复

参考网址 https://www.kancloud.cn/xiaobaoxuetp/mywork/3069416 https://segmentfault.com/a/1190000020487996 PHP判断多个时间段是否存在跨天或重复叠加的场景 /*** PHP计算两个时间段是否有交集&#xff08;边界重叠不算&#xff09;** param string $beginTime1 开始时间…

Certum ev多域名证书的优势

多域名证书作为一种能够为多个域名提供安全保护的证书类型&#xff0c;越来越受到企业的青睐。Certum作为一个成立了二十几年的CA认证机构&#xff0c;旗下的EV多域名SSL证书产品已经保护了多家企业的网站。Certum旗下的EV多域名证书作为一种能够为多个域名提供安全保护的证书类…

Python 使用 pymssql 连接 SQL Server 报错:DB-Lib error message 20002, severity 9

文章目录 版本说明排查过程参考个人简介 版本说明 Python 3.8SQL Server 2008pymssql 2.2.11 排查过程 最近给一个学妹看一个 Python 使用 pymssql 连接 SQL Server 报错问题&#xff0c;具体报错信息如下&#xff1a; Error: (20002, bDB-Lib error message 20002, severi…

【C语言】Linux socket 编程

一、Socket 通信过程 在 Linux 系统中&#xff0c;socket 是一种特殊的文件描述符&#xff0c;用于在网络中的不同主机间或者同一台主机中的不同进程间进行双向通信。它是通信链路的端点&#xff0c;可以看作是网络通信的接口。Socket 通信过程主要分为以下几个步骤&#xff1a…

RocketMQ5.0顺序消息设计实现

前言 顺序消息是 RocketMQ 提供的一种高级消息类型&#xff0c;支持消费者按照发送消息的先后顺序获取消息&#xff0c;从而实现业务场景中的顺序处理。 顺序消息的顺序关系通过消息组&#xff08;MessageGroup&#xff09;判定和识别&#xff0c;发送顺序消息时需要为每条消息…

RocketMQ5.0Pop消费模式

前言 RocketMQ 5.0 消费者引入了一种新的消费模式&#xff1a;Pop 消费模式&#xff0c;目的是解决 Push 消费模式的一些痛点。 RocketMQ 4.x 之前&#xff0c;消费模式分为两种&#xff1a; Pull&#xff1a;拉模式&#xff0c;消费者自行拉取消息、上报消费结果Push&#x…

C++字符串操作

1. 字符串比较 strcmp 1. 字符串比较 strcmp 头文件 string.h&#xff1b;变量需要传指针&#xff1b;返回>0 则第一个字符串比第二个字符串大&#xff0c;反之则小&#xff0c;0则表示两个字符串相同。 # include <iostream> # include <stdio.h> # include …

面试 Vue 框架八股文十问十答第一期

面试 Vue 框架八股文十问十答第一期 作者&#xff1a;程序员小白条&#xff0c;个人博客 相信看了本文后&#xff0c;对你的面试是有一定帮助的&#xff01;关注专栏后就能收到持续更新&#xff01; ⭐点赞⭐收藏⭐不迷路&#xff01;⭐ 1&#xff09;MVVM 的理解 MVVM (Mod…

LDD学习笔记 -- 用户空间 内核空间

LDD学习笔记 -- 用户空间 & 内核空间 内核空间用户空间用户空间和内核空间交互 操作系统上下文中User space和Kernel space概念 用户空间&#xff1a;Restricted Mode&#xff0c;用户级编程内核空间&#xff1a;Privileged Mode&#xff0c;内核级代码&#xff08;linux …

深入理解Vue生命周期钩子及其应用

Vue.js其独有的生命周期系统允许我们在组件的不同阶段执行自定义代码。在本文中&#xff0c;我们将深入探讨一个简单的Vue组件&#xff0c;通过观察其生命周期钩子的执行顺序&#xff0c;以及如何在特定时刻插入自己的逻辑。 Vue组件代码 <template><div><p&g…

华为OD机试 - 抢7游戏(Java JS Python C)

题目描述 A、B两个人玩抢7游戏,游戏规则为: A先报一个起始数字 X(10 ≤ 起始数字 ≤ 10000),B报下一个数字 Y (X - Y < 3),A再报一个数字 Z(Y - Z < 3),以此类推,直到其中一个抢到7,抢到7即为胜者; 在B赢得比赛的情况下,一共有多少种组合? 输入描述 …

分布式(8)

目录 36.什么是TCC&#xff1f; 37.分布式系统中常用的缓存方案有哪些&#xff1f; 38.分布式系统缓存的更新模式&#xff1f; 39.分布式缓存的淘汰策略&#xff1f; 40.Java中定时任务有哪些&#xff1f;如何演化的&#xff1f; 36.什么是TCC&#xff1f; TCC&#xff08…

Spring 事务实现

Spring 事务实现 Spring 事务使用 Transactional注解配置项事务传播行为PROPAGATION_REQUIRED当前方法必须在事务中&#xff0c;没有就创建&#xff0c;有就加入。PROPAGATION_SUPPORTS有事务就加入&#xff0c;没有就以非事务方式执行。PROPAGATION_MANDATORY有事务就加入&a…

【算法挨揍日记】day41——【模板】01背包、416. 分割等和子集

【模板】01背包_牛客题霸_牛客网你有一个背包&#xff0c;最多能容纳的体积是V。 现在有n个物品&#xff0c;第i个物品的体积为 ,。题目来自【牛客题霸】https://www.nowcoder.com/practice/fd55637d3f24484e96dad9e992d3f62e?tpId230&tqId2032484&ru/exam/oj&qru…

Flutter基础

一、关键字 class&#xff1a;用于定义一个新的类&#xff1b; extends: 用于指定一个类继承另一个类&#xff1b; mixin: 用于将一个类的代码片段添加到另一个类中&#xff0c;实现代码复用&#xff1b; abstract: 用于声明一个抽象类或抽象方法&#xff0c;不能直接实例化&a…

C++ map容器

通俗一点讲map其实就是python的字典(学会python字典 c_map)!!! map和unordered_map都是C中的关联容器&#xff0c;用于存储键值对。其主要区别在于底层实现方式和性能表现。 1、底层实现方式   map内部使用红黑树&#xff08;一种自平衡二叉查找树&#xff09;来实现&…

HarmonyOS-ArkTS基本语法及声明式UI描述

初识ArkTS语言 ArkTS是HarmonyOS优选的主力应用开发语言。ArkTS围绕应用开发在TypeScript&#xff08;简称TS&#xff09;生态基础上做了进一步扩展&#xff0c;继承了TS的所有特性&#xff0c;是TS的超集。因此&#xff0c;在学习ArkTS语言之前&#xff0c;建议开发者具备TS语…

机器学习常用算法模型总结

文章目录 1.基础篇&#xff1a;了解机器学习1.1 什么是机器学习1.2 机器学习的场景1.2.1 模式识别1.2.2 数据挖掘1.2.3 统计学习1.2.4 自然语言处理1.2.5 计算机视觉1.2.6 语音识别 1.3 机器学习与深度学习1.4 机器学习和人工智能1.5 机器学习的数学基础特征值和特征向量的定义…

Shell 文本处理常用命令

1、Sed sed 即 Stream EDitor&#xff0c;和 vi 不同&#xff0c;sed是基于行的文本编辑器。 Sed是从文件或管道中读取一行&#xff0c;处理一行&#xff0c;输出一行&#xff1b;再读取一行&#xff0c;再处理一行&#xff0c;再输出一行&#xff0c;直到最后一行。 # 查看文…