显示当前文集的文章列表

视图,代码:

from rest_framework.generics import ListAPIView
from .serializers import ArticleModelSerializer
from .models import Article
from rest_framework.response import Response
from rest_framework import status
class ArticleOfCollectionAPIView(ListAPIView):"""文集下的文章"""serializer_class = ArticleModelSerializerpermission_classes = [IsAuthenticated]  # 必须是登陆用户才能访问过来def list(self, request,  *args, **kwargs):user = request.usercollection_id = request.query_params.get("collection_id")try:ArticleCollection.objects.get(pk=collection_id)except ArticleCollection.DoesNotExist:return Response({"message":"错误的文集ID"}, status=status.HTTP_400_BAD_REQUEST)queryset = Article.objects.filter(user=user,collection_id=collection_id).order_by("orders", "-id")queryset = self.filter_queryset(queryset)serializer = self.get_serializer(queryset, many=True)return Response(serializer.data)

序列化器,代码:

from .models import Article
class ArticleModelSerializer(serializers.ModelSerializer):"""文章模型序列化器"""class Meta:model = Articlefields = ["id","name","content","html_content","collection","pub_date","is_public"]

路由代码:

from django.urls import path,re_path
from . import views
urlpatterns = [path("collection/", views.CollecionAPIView.as_view()),re_path("^collection/(?P<pk>\d+)/$", views.CollecionAPIView.as_view()),re_path("^collection/article/$", views.ArticleOfCollectionAPIView.as_view()),
]

现有的接口实现了以后,会发现没有文章,所以我们需要把文章相关模型注册到xadmin中,让我们可以在xadmin中添加测试文章数据。

article/adminx.py,代码:

import xadminfrom .models import ArticleCollection
class ArticleCollectionModelAdmin(object):"""文集"""list_display = ["id","name"]list_editable = ["name"]
xadmin.site.register(ArticleCollection,ArticleCollectionModelAdmin)from .models import Article
class ArticleModelAdmin(object):"""文章"""list_display=["id","name"]
xadmin.site.register(Article, ArticleModelAdmin)

完成上面注册以后,修改文章子应用在xadmin中的名称显示,

article/apps.py,代码:

from django.apps import AppConfigclass ArticleConfig(AppConfig):name = 'article'verbose_name="文章管理"

article/__init__.py,代码:

default_app_config = "article.apps.ArticleConfig"

访问xadmin的时候报错,错误提示:__str__方法相关错误。解决方案:

  1. users/models.py下模型的__str__方法代码调整:
from django.db import models
from django.contrib.auth.models import AbstractUser
class User(AbstractUser):mobile = models.CharField(max_length=15, null=True, unique=True, help_text="手机号码",verbose_name="手机号码")wxchat = models.CharField(max_length=100, null=True, unique=True, help_text="微信账号", verbose_name="微信账号")qq_number = models.CharField(max_length=11, null=True, unique=True, help_text="QQ号", verbose_name="QQ号")alipay = models.CharField(max_length=100, null=True, unique=True, help_text="支付宝账号", verbose_name="支付宝账号")# 保存文件的子目录 ImageField和FileField字段类型内置了文件上传处理类avatar = models.ImageField(upload_to="avatar", null=True, default=None, verbose_name="头像")money  = models.DecimalField(max_digits=8, decimal_places=2,default=0, help_text="账户余额", verbose_name="账户余额")nickname = models.CharField(max_length=20, null=True, unique=True, help_text="用户昵称",verbose_name="用户昵称")class Meta:db_table = "rr_users"verbose_name = "用户信息"verbose_name_plural = verbose_namedef __str__(self):return self.nickname if self.nickname else self.username

在utils/models.py,代码:

from django.db import models
class BaseModel(models.Model):"""公共模型"""name = models.CharField(null=True, default="暂无", blank=True, max_length=150, verbose_name='名称')orders = models.IntegerField(default=0, verbose_name='显示顺序')is_show = models.BooleanField(default=False, verbose_name="是否上架")is_delete = models.BooleanField(default=False, verbose_name="逻辑删除")created_time = models.DateTimeField(null=True,blank=True, auto_now_add=True, verbose_name="添加时间")updated_time = models.DateTimeField(null=True,blank=True, auto_now=True, verbose_name="更新时间")class Meta:# 设置当前模型在数据迁移的时候不要为它创建表abstract = Truedef __str__(self):return self.name if self.name else ""

前端展示当前文集的文章列表,代码:

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public}"></i><div class="_3P4JX poOXI" v-if="key==current_article"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn"><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title=""><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" title="随笔"><span class="">随笔</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单}},filters:{truncate(content){return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;}}},components: {mavonEditor},methods:{edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script>

切换文集以后,重新获取新的文集对应的文章列表。

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public}"></i><div class="_3P4JX poOXI" v-if="key==current_article"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn"><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title=""><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" title="随笔"><span class="">随笔</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单}},filters:{truncate(content){return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);},current_collection(){// 切换文集this.get_article_of_collection();}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;}}},components: {mavonEditor},methods:{edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script>

显示当前文章的操作菜单【扩展】

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public}"></i><div class="_3P4JX poOXI" v-if="key==current_article" @click.stop="is_show_article_menu=!is_show_article_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_article_menu}"><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title=""><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" :title="collection.name" v-if="key!=current_collection" v-for="collection,key in collection_list"><span class="">{{collection.name}}</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单is_show_article_menu: false, // 是否显示文章的菜单}},filters:{truncate(content){return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);},current_collection(){// 切换文集this.get_article_of_collection();// 关闭文章菜单this.is_show_article_menu = false;},current_article(){// 切换文章this.is_show_article_menu = false;}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;// 关闭文章菜单this.is_show_article_menu = false;}}},components: {mavonEditor},methods:{edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script><style scoped>/* 在原有css代码基础上调整以下样式 */.rQQG7{height: 100%;display: block;width: 40%;border-right: 1px solid #d9d9d9;}._3x4X_{top: 0;right: 100%;position: absolute;}._24i7u {flex-shrink: 0;padding: 0 80px 10px 40px;margin-bottom: 0;border: none;font-size: 30px;font-weight: 400;line-height: 30px;box-shadow: none;color: #595959;background-color: transparent;outline: none;border-radius: 0;overflow: hidden;text-overflow: ellipsis;white-space: nowrap;position: absolute;top: 0;right: 0;width: 60%;
}#editor {margin: auto;width: 60%;position: absolute;right: 0;top: 44px;height: 580px;}
</style>

作业:完善当前文集下的文章列表的BUG

当前文集如果默认是0,那么在用户新建文件以后,显示的文章列表错误的。

所以,我们需要在用户创建文集以后,设置当前文章列表为空列表。

添加文章

视图代码:

from rest_framework.generics import ListAPIView
from .serializers import ArticleModelSerializer
from .models import Article
from rest_framework.response import Response
from rest_framework import status
class ArticleOfCollectionAPIView(ListAPIView, CreateAPIView):"""文集下的文章"""serializer_class = ArticleModelSerializerpermission_classes = [IsAuthenticated]  # 必须是登陆用户才能访问过来def list(self, request,  *args, **kwargs):user = request.usercollection_id = request.query_params.get("collection_id")try:ArticleCollection.objects.get(pk=collection_id)except ArticleCollection.DoesNotExist:return Response({"message":"错误的文集ID"}, status=status.HTTP_400_BAD_REQUEST)queryset = Article.objects.filter(user=user,collection_id=collection_id).order_by("orders", "-id")queryset = self.filter_queryset(queryset)serializer = self.get_serializer(queryset, many=True)return Response(serializer.data)

序列化器,代码:

from .models import Article
class ArticleModelSerializer(serializers.ModelSerializer):"""文章模型序列化器"""position = serializers.IntegerField(write_only=True, default=0, allow_null=True, label="添加文章的位置", help_text="在文章列表的前面插入添加则为0, 在文章列表的后面追加添加则为1")class Meta:model = Articlefields = ["id","position","name","content","html_content","collection","pub_date","is_public"]read_only_fields = ["id","content","html_content","pub_date","is_public"]def validate(self, data):name = data.get("name")if len(name)<1:raise serializers.ValidationError("对不起,文章标题不能为空!")return datadef create(self, validated_data):"""添加文章"""try:article = Article.objects.create(name=validated_data.get("name"),user=self.context["request"].user,content="",html_content="",collection=validated_data.get("collection"),is_public=False,is_show=True,orders=0)except:raise serializers.ValidationError("对不起,文章添加失败!")if validated_data.get("position"):"""如果用户设置在文章列表后面追加添加文章,则让文章的排序值跟着id来变化越来越大排序值越大的文章,越往后排列"""article.orders = article.idarticle.save()return article

路由代码与上面的文集的文章列表共用一个路由。

客户端发送ajax,代码:

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5" @click.stop="add_article(0)"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public}"></i><div class="_3P4JX poOXI" v-if="key==current_article" @click.stop="is_show_article_menu=!is_show_article_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_article_menu}"><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title=""><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" :title="collection.name" v-if="key!=current_collection" v-for="collection,key in collection_list"><span class="">{{collection.name}}</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3" @click.stop="add_article(1)"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单is_show_article_menu: false, // 是否显示文章的菜单position: 0,  // 添加文章的位置,默认前面插入,值为0}},filters:{truncate(content){if(content === null){return 0;}return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);},current_collection(){// 切换文集this.get_article_of_collection();// 关闭文章菜单this.is_show_article_menu = false;},current_article(){// 切换文章this.is_show_article_menu = false;}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;// 关闭文章菜单this.is_show_article_menu = false;}}},components: {mavonEditor},methods:{edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},add_article(position){// 添加文章this.$axios.post(`${this.$settings.Host}/article/collection/article/`,{collection: this.collection_list[this.current_collection].id,name: this.get_datetime(),position: position,},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("文章添加成功!");// 给本地的aticle追加/插入新建的文章if(position){// 追加this.article_list.push(response.data);}else{// 插入this.article_list.unshift(response.data);}}).catch(error=>{this.$message.error("对不起,添加文章失败!");});},get_datetime(){// 获取客户端日期格式let datetime = new Date();let Y = datetime.getFullYear();let m = datetime.getMonth()+1;m = m<10?"0"+m:m;let d = datetime.getDate();d = d<10?"0"+d:d;return `${Y}-${m}-${d}`;},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script>

文章发布

默认情况下,我们的文章属于未发布状态下的,这时候除了作者本人和后台管理员,其他人看不到的。

  1. 在写文章页面,我们需要给用户区分哪些文章已经发布了,哪些文章没有发布,在前面的文章列表中我们已经完成了这个功能。
  2. 在写文章页面,提供一个api,用于修改发布状态。同时,需要提供发布后的文章页面。

切换文章的发布状态

服务端提供切换文章发布状态的功能。、

视图代码:

from rest_framework.views import APIView
class ArticleAPIView(APIView):permission_classes = [IsAuthenticated]def patch(self,request,pk):"""切换文章的发布状态"""try:article = Article.objects.get(user=request.user,pk=pk)except Article.DoesNotExist:return Response({"message":"当前文章不存在!"}, status=status.HTTP_400_BAD_REQUEST)article.is_public = not article.is_publicarticle.save()return Response(status=status.HTTP_200_OK)

路由代码:

from django.urls import path,re_path
from . import views
urlpatterns = [path("collection/", views.CollecionAPIView.as_view()),re_path("^collection/(?P<pk>\d+)/$", views.CollecionAPIView.as_view()),path("collection/article/", views.ArticleOfCollectionAPIView.as_view()),re_path("^public/(?P<pk>\d+)/$", views.ArticleAPIView.as_view()),
]

前端代码:

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5" @click.stop="add_article(0)"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public}"></i><div class="_3P4JX poOXI" v-if="key==current_article" @click.stop="is_show_article_menu=!is_show_article_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_article_menu}"><li class="_2po2r cRfUr" title="" v-if="!article.is_public" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title="" v-if="!article.is_public"><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title="" v-if="!article.is_public"><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title="" v-if="article.is_public" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>取消发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" :title="collection.name" v-if="key!=current_collection" v-for="collection,key in collection_list"><span class="">{{collection.name}}</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3" @click.stop="add_article(1)"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单is_show_article_menu: false, // 是否显示文章的菜单position: 0,  // 添加文章的位置,默认前面插入,值为0}},filters:{truncate(content){if(content === null){return 0;}return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);},current_collection(){// 切换文集this.get_article_of_collection();// 关闭文章菜单this.is_show_article_menu = false;},current_article(){// 切换文章this.is_show_article_menu = false;}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;// 关闭文章菜单this.is_show_article_menu = false;}}},components: {mavonEditor},methods:{edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},add_article(position){// 添加文章this.$axios.post(`${this.$settings.Host}/article/collection/article/`,{collection: this.collection_list[this.current_collection].id,name: this.get_datetime(),position: position,},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("文章添加成功!");// 给本地的aticle追加/插入新建的文章if(position){// 追加this.article_list.push(response.data);}else{// 插入this.article_list.unshift(response.data);}}).catch(error=>{this.$message.error("对不起,添加文章失败!");});},get_datetime(){// 获取客户端日期格式let datetime = new Date();let Y = datetime.getFullYear();let m = datetime.getMonth()+1;m = m<10?"0"+m:m;let d = datetime.getDate();d = d<10?"0"+d:d;return `${Y}-${m}-${d}`;},pub_article(){// 切换文章的发布状态let article = this.article_list[this.current_article];this.$axios.patch(`${this.$settings.Host}/article/public/${article.id}/`,{},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{// 切换成功article.is_public=!article.is_public;// 关闭菜单this.is_show_article_menu = false;}).catch(error=>{this.$message.error("切换文章发布状态失败!");})},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script>

移动文章

首先修复文章菜单中的文集列表,在css代码中这个位置放置以下代码:

._2_WAp ._2KzJx, ._2_WAp ._3x4X_ {position: absolute;right: 100%;top: 0;display: none;
}
._2_WAp:hover ._2KzJx, ._2_WAp:hover ._3x4X_ {display: block;
}

服务端提供修改文章的文集ID的接口,在上面切换文章发布状态的视图类中,增加代码:

from rest_framework.views import APIView
class ArticleAPIView(APIView):permission_classes = [IsAuthenticated]def patch(self,request,pk):"""切换文章的发布状态"""try:article = Article.objects.get(user=request.user,pk=pk)except Article.DoesNotExist:return Response({"message":"当前文章不存在!"}, status=status.HTTP_400_BAD_REQUEST)article.is_public = not article.is_publicarticle.save()return Response(status=status.HTTP_200_OK)def put(self,request,pk):"""移动文章"""try:article = Article.objects.get(user=request.user, pk=pk)except Article.DoesNotExist:return Response({"message": "当前文章不存在!"}, status=status.HTTP_400_BAD_REQUEST)collection_id = request.data.get("collection_id")try:collection = ArticleCollection.objects.get(user=request.user, pk=collection_id)except ArticleCollection.DoesNotExist:return Response({"message": "当前文集不存在!"}, status=status.HTTP_400_BAD_REQUEST)article.collection = collectionarticle.save()return Response(status=status.HTTP_200_OK)

路由与上面发布文章状态切换接口一致。客户端实现切换文集代码效果:

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5" @click.stop="add_article(0)"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public}"></i><div class="_3P4JX poOXI" v-if="key==current_article" @click.stop="is_show_article_menu=!is_show_article_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_article_menu}"><li class="_2po2r cRfUr" title="" v-if="!article.is_public" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title="" v-if="!article.is_public"><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title="" v-if="!article.is_public"><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title="" v-if="article.is_public" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>取消发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" :title="collection.name" v-if="key!=current_collection" v-for="collection,key in collection_list" @click.stop="move_article(collection.id)"><span class="">{{collection.name}}</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3" @click.stop="add_article(1)"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单is_show_article_menu: false, // 是否显示文章的菜单position: 0,  // 添加文章的位置,默认前面插入,值为0}},filters:{truncate(content){if(content === null){return 0;}return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);},current_collection(){// 切换文集this.get_article_of_collection();// 关闭文章菜单this.is_show_article_menu = false;},current_article(){// 切换文章this.is_show_article_menu = false;}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;// 关闭文章菜单this.is_show_article_menu = false;}}},components: {mavonEditor},methods:{edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},add_article(position){// 添加文章this.$axios.post(`${this.$settings.Host}/article/collection/article/`,{collection: this.collection_list[this.current_collection].id,name: this.get_datetime(),position: position,},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("文章添加成功!");// 给本地的aticle追加/插入新建的文章if(position){// 追加this.article_list.push(response.data);}else{// 插入this.article_list.unshift(response.data);}}).catch(error=>{this.$message.error("对不起,添加文章失败!");});},get_datetime(){// 获取客户端日期格式let datetime = new Date();let Y = datetime.getFullYear();let m = datetime.getMonth()+1;m = m<10?"0"+m:m;let d = datetime.getDate();d = d<10?"0"+d:d;return `${Y}-${m}-${d}`;},pub_article(){// 切换文章的发布状态let article = this.article_list[this.current_article];this.$axios.patch(`${this.$settings.Host}/article/public/${article.id}/`,{},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{// 切换成功article.is_public=!article.is_public;// 关闭菜单this.is_show_article_menu = false;}).catch(error=>{this.$message.error("切换文章发布状态失败!");})},move_article(collection_id){// 移动文章this.is_show_article_menu = false;let article = this.article_list[this.current_article];this.$axios.put(`${this.$settings.Host}/article/public/${article.id}/`,{collection_id, // collection_id: collection_id的简写,},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{this.article_list.splice(this.current_article,1);}).catch(error=>{this.$message.error("移动文章失败!");});},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script>

定时发布[扩展知识点]

原理:使用celery完成定时任务!
步骤:
1. 当用户点选了定时发布, 页面中弹出一个选择时间的窗口。
2. 当用户设置完成发布时间以后,点击“确认”以后,把这个时间和文章id发送到服务端。
3. 服务端中文章模型的pub_date记录这个定时发布时间。
4. 在celery中创建一个定时任务,在每个固定时间段,检查文章表中,对应时间段的pub_date把对应的文章进行发布。

前端增加一个选择时间的弹窗,代码:

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5" @click.stop="add_article(0)"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public}"></i><div class="_3P4JX poOXI" v-if="key==current_article" @click.stop="is_show_article_menu=!is_show_article_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_article_menu}"><li class="_2po2r cRfUr" title="" v-if="!article.is_public" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title="" v-if="!article.is_public" @click.stop="is_show_interval_windows=true;is_show_article_menu=false;"><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title="" v-if="!article.is_public"><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title="" v-if="article.is_public" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>取消发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" :title="collection.name" v-if="key!=current_collection" v-for="collection,key in collection_list" @click.stop="move_article(collection.id)"><span class="">{{collection.name}}</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3" @click.stop="add_article(1)"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div><!-- 定时发布弹窗 --><div class="interval_box" v-if="is_show_interval_windows"><transition name="el-fade-in-linear"><div class="transition-box"><div class="_2tIvb"><div class="ZTNas" aria-label="Close" @click.stop="is_show_interval_windows=false"><i class="fa fa-close"></i></div><div class="_1KgC3"><div class="-K8Re">定时发文 </div></div><div class="_1LROK PWCkH"><div class="wzwGh">选择定时发布的时间:</div><div class="On4jq"><el-date-pickerv-model="pub_date"type="datetime"format="yyyy-MM-dd HH:mm"placeholder="选择日期时间"></el-date-picker></div><div class="_3mEYS">本文章将于<span class="yfqan">{{timeformat(pub_date)}}</span>发布。</div></div><div class="RzhZ5 pCffa"><button type="button" class="_3zXcJ" @click.stop="is_show_interval_windows=false;"><span>取 消</span></button><button type="button" class="_3zXcJ _3QfkW" @click.stop="interval_article"><span>确 认</span></button></div></div></div></transition></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单is_show_article_menu: false, // 是否显示文章的菜单position: 0,  // 添加文章的位置,默认前面插入,值为0is_show_interval_windows: false, // 选择定时发布文章的时间窗口pub_date: new Date().toLocaleDateString(),  // 定时发布当前文章的时间默认值为当前时间}},filters:{truncate(content){if(content === null){return 0;}return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);},current_collection(){// 切换文集this.get_article_of_collection();// 关闭文章菜单this.is_show_article_menu = false;},current_article(){// 切换文章this.is_show_article_menu = false;}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;// 关闭文章菜单this.is_show_article_menu = false;}}},components: {mavonEditor},methods:{interval_article(){// 定时发布文章let pub_date = this.pub_date;let article_id = this.article_list[this.current_article].id;// 发送ajax},edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},add_article(position){// 添加文章this.$axios.post(`${this.$settings.Host}/article/collection/article/`,{collection: this.collection_list[this.current_collection].id,name: this.get_datetime(),position: position,},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("文章添加成功!");// 给本地的aticle追加/插入新建的文章if(position){// 追加this.article_list.push(response.data);}else{// 插入this.article_list.unshift(response.data);}}).catch(error=>{this.$message.error("对不起,添加文章失败!");});},get_datetime(){// 获取客户端日期格式let datetime = new Date();let Y = datetime.getFullYear();let m = datetime.getMonth()+1;m = m<10?"0"+m:m;let d = datetime.getDate();d = d<10?"0"+d:d;return `${Y}-${m}-${d}`;},timeformat(time){// 获取客户端日期时间格式time = new Date(time);return `${time.getFullYear()}-${time.getMonth()+1}-${time.getDate()} ${time.getHours()}:${time.getMinutes()}`;},pub_article(){// 切换文章的发布状态let article = this.article_list[this.current_article];this.$axios.patch(`${this.$settings.Host}/article/public/${article.id}/`,{},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{// 切换成功article.is_public=!article.is_public;// 关闭菜单this.is_show_article_menu = false;}).catch(error=>{this.$message.error("切换文章发布状态失败!");})},move_article(collection_id){// 移动文章this.is_show_article_menu = false;let article = this.article_list[this.current_article];this.$axios.put(`${this.$settings.Host}/article/public/${article.id}/`,{collection_id, // collection_id: collection_id的简写,},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{this.article_list.splice(this.current_article,1);}).catch(error=>{this.$message.error("移动文章失败!");});},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script><style scoped>/* 页面原来的样式不要删除 *//* 新增以下定时发布文章的相关样式 */.interval_box{display: flex;height: 280px;width: 400px;position: fixed;top: 0;left: 0;right: 0;bottom: 0;margin: auto;background: #fff;border-radius: 4px;z-index: 2000;}.NVdZF{display:block}._20JYe {position: fixed;top: 0;right: 0;left: 0;bottom: 0;background-color: hsla(0, 0%, 100%, .7);height: 100%;z-index: 1000;filter: alpha(opacity=30)}body.reader-night-mode ._20JYe {background-color: hsla(0, 0%, 40%, .7)}._3WS2u {display: none}._23VW8 {position: fixed;overflow: auto;top: 0;right: 0;bottom: 0;left: 0;z-index: 1010;-webkit-overflow-scrolling: touch;outline: 0}._3cgNz {position: relative;background-color: #fff;width: auto;border: 0;border-radius: 6px;margin: 0 auto 24px;background-clip: padding-box;-webkit-box-shadow: 0 2px 8px rgba(0, 0, 0, .2);box-shadow: 0 2px 8px rgba(0, 0, 0, .2)}body.reader-night-mode ._3cgNz {background-color: #3d3d3d}.cjahm {position: absolute;top: 50%;left: 50%;-webkit-transform: translate(-50%, -50%);-ms-transform: translate(-50%, -50%);transform: translate(-50%, -50%)}.cjahm.move-up-appear.move-up-appear-active, .cjahm.move-up-enter.move-up-enter-active {-webkit-animation-name: _2oyZp;animation-name: _2oyZp}.cjahm.move-up-leave.move-up-leave-active {-webkit-animation-name: _3g8pf;animation-name: _3g8pf}._2tIvb {position: relative;border-radius: 6px;background-color: #eee;}body.reader-night-mode ._2tIvb {background-color: #3d3d3d}.ZTNas {position: absolute;right: 0;top: 0;width: 48px;height: 48px;line-height: 46px;text-align: center;font-size: 16px;cursor: pointer;color: #999;-webkit-transition: color .3s ease;-o-transition: color .3s ease;transition: color .3s ease}.ZTNas:hover {color: #4d4d4d}._1KgC3 {padding: 13px 16px;border-radius: 4px 4px 0 0;color: #595959;border-bottom: 1px solid #d9d9d9}body.reader-night-mode ._1KgC3 {border-color: #2e2e2e}.-K8Re {margin: 0;font-size: 16px;line-height: 21px;font-weight: 500;color: #4d4d4d}body.reader-night-mode .-K8Re {color: #b3b3b3}._3O4M2 {font-size: 13px;padding-left: 10px;color: #999}._3O4M2 a {color: #3294d0}.PWCkH {padding: 16px;font-size: 12px;line-height: 1.5;color: #595959}body.reader-night-mode .PWCkH {color: #b3b3b3}.pCffa {padding: 0 16px 16px;text-align: right;border-radius: 0 0 2px 2px}._26mjB {display: block}.HhIYk {zoom: 1}.HhIYk:after, .HhIYk:before {content: " ";display: table}.HhIYk:after {clear: both;visibility: hidden;font-size: 0;height: 0}._37SYn {font-size: 13px;line-height: 20px;padding: 30px 30px 20px;color: #333}body.reader-night-mode ._37SYn {color: #b3b3b3}._2BmdS {padding: 0 16px 16px;text-align: right;border-radius: 0 0 2px 2px}._26mjB .PWCkH {padding: 0;border-radius: 2px}._26mjB .PWCkH a {color: #3194d0}._26mjB .PWCkH a:active, ._26mjB .PWCkH a:hover {color: #2b86bc}._26mjB ._38pIX {padding: 30px 16px;border: 0}._26mjB ._38pIX input {display: block;border-radius: 4px;width: 100%;line-height: 20px;padding: 5px 10px;font-size: 15px;background-color: transparent;border: 1px solid #ccc}body.reader-night-mode ._26mjB ._38pIX input {border-color: #2e2e2e}._26mjB .ZTNas {width: 32px;height: 32px;line-height: 32px}.HSpeJ .PWCkH {border: 0}
</style>

服务端提供修改pub_date发布时间的api接口

配置文件中, 调整市区,代码:

# 必须设置为True,否则后面提交发布时间到服务端会报错!
USE_TZ = True

视图代码:

class ArticleIntervalAPIView(APIView):"""定时发布文章"""permission_classes = [IsAuthenticated]def put(self,request,pk):try:article = Article.objects.get(pk=pk)except Article.DoesNotExist:return Response("对不起,当前文章不存在!", status=status.HTTP_400_BAD_REQUEST)pub_date = request.data.get("pub_date")article.pub_date = pub_datearticle.save()return Response("操作成功!")

路由,代码:

urlpatterns = [....re_path("^interval/(?P<pk>\d+)/$", views.ArticleIntervalAPIView.as_view()),
]

客户端发送ajax请求,代码:

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5" @click.stop="add_article(0)"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public,interval:article.pub_date}"></i><div class="_3P4JX poOXI" v-if="key==current_article" @click.stop="is_show_article_menu=!is_show_article_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_article_menu}"><li class="_2po2r cRfUr" title="" v-if="!(article.is_public||article.pub_date)" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title="" v-if="!(article.is_public||article.pub_date)" @click.stop="is_show_interval_windows=true;is_show_article_menu=false;"><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title="" v-if="!(article.is_public||article.pub_date)"><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title="" v-if="article.is_public||article.pub_date" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>取消发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" :title="collection.name" v-if="key!=current_collection" v-for="collection,key in collection_list" @click.stop="move_article(collection.id)"><span class="">{{collection.name}}</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3" @click.stop="add_article(1)"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div><!-- 定时发布弹窗 --><div class="interval_box" v-if="is_show_interval_windows"><transition name="el-fade-in-linear"><div class="transition-box"><div class="_2tIvb"><div class="ZTNas" aria-label="Close" @click.stop="is_show_interval_windows=false"><i class="fa fa-close"></i></div><div class="_1KgC3"><div class="-K8Re">定时发文 </div></div><div class="_1LROK PWCkH"><div class="wzwGh">选择定时发布的时间:</div><div class="On4jq"><el-date-pickerv-model="pub_date"type="datetime"format="yyyy-MM-dd HH:mm"placeholder="选择日期时间"></el-date-picker></div><div class="_3mEYS">本文章将于<span class="yfqan">{{timeformat(pub_date)}}</span>发布。</div></div><div class="RzhZ5 pCffa"><button type="button" class="_3zXcJ" @click.stop="is_show_interval_windows=false;"><span>取 消</span></button><button type="button" class="_3zXcJ _3QfkW" @click.stop="interval_article"><span>确 认</span></button></div></div></div></transition></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单is_show_article_menu: false, // 是否显示文章的菜单position: 0,  // 添加文章的位置,默认前面插入,值为0is_show_interval_windows: false, // 选择定时发布文章的时间窗口pub_date: new Date().toLocaleDateString(),  // 定时发布当前文章的时间默认值为当前时间}},filters:{truncate(content){if(content === null){return 0;}return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);},current_collection(){// 切换文集this.get_article_of_collection();// 关闭文章菜单this.is_show_article_menu = false;},current_article(){// 切换文章this.is_show_article_menu = false;}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;// 关闭文章菜单this.is_show_article_menu = false;}}},components: {mavonEditor},methods:{interval_article(){// 定时发布文章let article_id = this.article_list[this.current_article].id;// 发送ajaxthis.$axios.put(`${this.$settings.Host}/article/interval/${article_id}/`,{pub_date: this.pub_date,},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.article_list[this.current_article].pub_date = this.pub_date;this.$message.success("文章设置成功!");}).catch(error=>{this.$message.error("操作失败!请联系客服工作人员!");});this.is_show_interval_windows=false;// 关闭定时窗口},edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},add_article(position){// 添加文章this.$axios.post(`${this.$settings.Host}/article/collection/article/`,{collection: this.collection_list[this.current_collection].id,name: this.get_datetime(),position: position,},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("文章添加成功!");// 给本地的aticle追加/插入新建的文章if(position){// 追加this.article_list.push(response.data);}else{// 插入this.article_list.unshift(response.data);}}).catch(error=>{this.$message.error("对不起,添加文章失败!");});},get_datetime(){// 获取客户端日期格式let datetime = new Date();let Y = datetime.getFullYear();let m = datetime.getMonth()+1;m = m<10?"0"+m:m;let d = datetime.getDate();d = d<10?"0"+d:d;return `${Y}-${m}-${d}`;},timeformat(time){// 获取客户端日期时间格式time = new Date(time);return `${time.getFullYear()}-${time.getMonth()+1}-${time.getDate()} ${time.getHours()}:${time.getMinutes()}`;},pub_article(){// 切换文章的发布状态let article = this.article_list[this.current_article];this.$axios.patch(`${this.$settings.Host}/article/public/${article.id}/`,{},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{// 切换成功article.is_public=!article.is_public;// 关闭菜单this.is_show_article_menu = false;}).catch(error=>{this.$message.error("切换文章发布状态失败!");})},move_article(collection_id){// 移动文章this.is_show_article_menu = false;let article = this.article_list[this.current_article];this.$axios.put(`${this.$settings.Host}/article/public/${article.id}/`,{collection_id, // collection_id: collection_id的简写,},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{this.article_list.splice(this.current_article,1);}).catch(error=>{this.$message.error("移动文章失败!");});},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script>
<style>
/*  注意:在素材目录下找到sprite.9d24217.png放到static/image目录中 */
._25Ilv .interval {background: url(/static/image/sprite.9d24217.png) no-repeat -74px -25px;background-size: 250px;
}
</style>

当用户取消发布时,我们应该把定时发布也取消掉,也就是说把当前文章的pub_date清空!

视图中切换文章发布状态视图接口代码进行调整,代码:

from rest_framework.views import APIView
class ArticleAPIView(APIView):permission_classes = [IsAuthenticated]def patch(self,request,pk):"""切换文章的发布状态"""try:article = Article.objects.get(user=request.user,pk=pk)except Article.DoesNotExist:return Response({"message":"当前文章不存在!"}, status=status.HTTP_400_BAD_REQUEST)# 判断如果是取消发布,则取消定时发布# 1. 定时发布if article.is_public == False and article.pub_date is not None:# 原来定时发布的,现在取消article.pub_date = Noneelif article.is_public == False and article.pub_date is None:# 原来没有发布的,现在立即发布article.is_public = Trueelif article.is_public == True:# 原来发布的,现在取消article.is_public = Falsearticle.save()return Response(status=status.HTTP_200_OK)

客户端状态在取消发布以后调整显示状态,代码:

<template><div class="write" v-if="is_show_page"><div class="_2v5v5"><div class="_3zibT"><router-link to="/">回首页</router-link></div><div class="_1iZMb"><div class="_33Zlg" @click="collection_form=true"><i class="fa fa-plus"></i><span>新建文集</span></div><div class="_2G97m"><form class="M8J6Q" :class="collection_form?'_2a1Rp':'_1mU5v'"><input type="text" placeholder="请输入文集名..." v-model="collection_name" class="_1CtV4"><button type="submit" class="dwU8Q _3zXcJ _3QfkW" @click.stop.prevent="add_collection"><span>提 交</span></button><button type="button" class="vIzwB _3zXcJ" @click.stop.prevent="collection_form=false"><span>取 消</span></button></form></div></div><ul class="_3MbJ4 _3t059"><li class="_3DM7w" :class="{_31PCv:current_collection==key}" @click.stop="current_collection=key;is_show_collection_menu=false;" :title="collection.name" v-for="collection,key in collection_list"><div class="_3P4JX _2VLy-" v-if="current_collection==key" @click.stop="is_show_collection_menu=!is_show_collection_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_collection_menu}"><li class="_2po2r cRfUr" title=""><span class="" @click.stop="edit_collection"><i class="fa fa-pencil-square-o _22XWG"></i>修改文集</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文集</span></li></ul></span></div><span>{{collection.name}}</span></li></ul><div style="height: 50px;"></div><div role="button" class="h-5Am"><span class="ant-dropdown-trigger"><i class="fa fa-bars"></i><span>设置</span></span><span class="Yv5Zx">遇到问题<i class="fa fa-question-circle-o"></i></span></div></div><div class="rQQG7"><div class="_3revO _2mnPN"><div class="_3br9T"><div><div class="_1GsW5" @click.stop="add_article(0)"><i class="fa fa-plus-circle"></i><span> 新建文章</span></div><ul class="_2TxA-"><li class="_25Ilv" :class="{_33nt7:key==current_article}" @click.stop="current_article=key;" :title="article.name" v-for="article,key in article_list"><i class="_13kgp" :class="{_2m93u:article.is_public,interval:article.pub_date}"></i><div class="_3P4JX poOXI" v-if="key==current_article" @click.stop="is_show_article_menu=!is_show_article_menu"><i class="fa fa-gear"></i><span><ul class="_2V8zt _3FcHm _2w9pn" :class="{NvfK4:is_show_article_menu}"><li class="_2po2r cRfUr" title="" v-if="!(article.is_public||article.pub_date)" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>直接发布</span></li><li class="_2po2r cRfUr" title="" v-if="!(article.is_public||article.pub_date)" @click.stop="is_show_interval_windows=true;is_show_article_menu=false;"><span class=""><i class="fa fa-clock-o _22XWG"></i>定时发布</span></li><li class="_2po2r cRfUr" title="" v-if="!(article.is_public||article.pub_date)"><span class="_20tIi"><i class="iconfont ic-paid _22XWG"></i>发布为付费文章</span></li><li class="_2po2r cRfUr" title="" v-if="article.is_public||article.pub_date" @click.stop="pub_article"><span class=""><i class="fa fa-share _22XWG"></i>取消发布</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="iconfont ic-set _22XWG"></i>设置发布样式</span></li><li class="_3nZXj _2_WAp _3df2u _2po2r cRfUr" title=""><span class=""><i class="fa fa-folder-open _22XWG"></i>移动文章<div class="_3x4X_"><ul class="_2KzJx oGKRI _3DXDE _2w9pn"><li class="_2po2r cRfUr" :title="collection.name" v-if="key!=current_collection" v-for="collection,key in collection_list" @click.stop="move_article(collection.id)"><span class="">{{collection.name}}</span></li></ul></div></span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-history _22XWG"></i>历史版本</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-trash-o _22XWG"></i>删除文章</span></li><li class="_2po2r cRfUr" title=""><span class=""><i class="fa fa-ban _22XWG"></i>设置禁止转载</span></li></ul></span></div><span class="NariC">{{article.name}}</span><span class="hLzJv">{{article.content|truncate}}</span><span class="_29C-V" v-if="key==current_article">字数:{{article.content.length}}</span></li>
<!--              <li class="_25Ilv" title="2020-01-12">-->
<!--                <i class="_13kgp"></i>-->
<!--                <span class="NariC">2020-01-12</span>-->
<!--                <span class="hLzJv">题目:有四个数字:1、2、3、4,能组成多少个互不相同且无重复数字的三位数?各是多少?--><!--题目:企业发放的奖金根据利润提成</span>-->
<!--              </li>--></ul><div class="_2cVn3" @click.stop="add_article(1)"><i class="fa fa-plus"></i><span> 在下方新建文章</span></div></div></div></div><input type="text" class="_24i7u" value="2020-01-12"><div id="editor"><mavon-editorstyle="height: 100%"v-model="editorContent":ishljs="true"ref=md@imgAdd="imgAdd"@imgDel="imgDel"></mavon-editor></div></div><!-- 定时发布弹窗 --><div class="interval_box" v-if="is_show_interval_windows"><transition name="el-fade-in-linear"><div class="transition-box"><div class="_2tIvb"><div class="ZTNas" aria-label="Close" @click.stop="is_show_interval_windows=false"><i class="fa fa-close"></i></div><div class="_1KgC3"><div class="-K8Re">定时发文 </div></div><div class="_1LROK PWCkH"><div class="wzwGh">选择定时发布的时间:</div><div class="On4jq"><el-date-pickerv-model="pub_date"type="datetime"format="yyyy-MM-dd HH:mm"placeholder="选择日期时间"></el-date-picker></div><div class="_3mEYS">本文章将于<span class="yfqan">{{timeformat(pub_date)}}</span>发布。</div></div><div class="RzhZ5 pCffa"><button type="button" class="_3zXcJ" @click.stop="is_show_interval_windows=false;"><span>取 消</span></button><button type="button" class="_3zXcJ _3QfkW" @click.stop="interval_article"><span>确 认</span></button></div></div></div></transition></div></div>
</template>
<script>import { mavonEditor } from 'mavon-editor'import 'mavon-editor/dist/css/index.css'import "../../static/font-awesome/css/font-awesome.css";export default {name: "Writer",data(){return {is_show_page: false, // 是否显示页面collection_list:[],  // 文集列表article_list:[],     // 文章列表current_collection: 0, // 当前选中的文集下标,默认为0current_article:0,     // 当前选中的文章下标,默认为0editorContent:"",img_file:[],collection_form:false,collection_name:"",is_show_collection_menu: false, // 是否显示文集的菜单is_show_article_menu: false, // 是否显示文章的菜单position: 0,  // 添加文章的位置,默认前面插入,值为0is_show_interval_windows: false, // 选择定时发布文章的时间窗口pub_date: new Date().toLocaleDateString(),  // 定时发布当前文章的时间默认值为当前时间}},filters:{truncate(content){if(content === null){return 0;}return content.substr(0,30);}},watch:{editorContent(){console.log(this.editorContent);},current_collection(){// 切换文集this.get_article_of_collection();// 关闭文章菜单this.is_show_article_menu = false;},current_article(){// 切换文章this.is_show_article_menu = false;}},created(){// 判断登录this.$settings.check_user_login(this,"警告","您尚未登录!", "跳转到登录", "/login");if(this.token){// 显示页面this.is_show_page = true;// 获取当前用户的文集列表this.get_collection();}},mounted(){if(this.is_show_page){document.querySelector("#editor").style.height = document.documentElement.clientHeight - document.querySelector("._24i7u").clientHeight + "px";// 点选页面其他位置,关闭菜单document.onclick = (event)=>{// 关闭文集菜单this.is_show_collection_menu = false;// 关闭文章菜单this.is_show_article_menu = false;}}},components: {mavonEditor},methods:{interval_article(){// 定时发布文章let article_id = this.article_list[this.current_article].id;// 发送ajaxthis.$axios.put(`${this.$settings.Host}/article/interval/${article_id}/`,{pub_date: this.pub_date,},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.article_list[this.current_article].pub_date = this.pub_date;this.$message.success("文章设置成功!");}).catch(error=>{this.$message.error("操作失败!请联系客服工作人员!");});this.is_show_interval_windows=false;// 关闭定时窗口},edit_collection(){// 修改文集this.is_show_collection_menu=false; // 关闭文集的操作菜单this.$prompt('请输入新文集名', '提示', {confirmButtonText: '保存',cancelButtonText: '取消',inputPattern: /.{1,}/,inputErrorMessage: '文集名称不能为空!',inputValue: this.collection_list[this.current_collection].name,}).then(({ value }) => {// 点击确定,需要把当前文集名称提交到服务端进行修改let collection_id = this.collection_list[this.current_collection].id;this.$axios.put(`${this.$settings.Host}/article/collection/${collection_id}/`,{name: value,},{headers:{Authorization:"jwt " + this.token,}}).then(response=>{// 服务端ajax请求操作成功,则客户端的name也要发生改变this.collection_list[this.current_collection].name = value;}).catch(error=>{this.$message.error(error.response.data);})}).catch(() => {});},add_collection(){// 添加文集if(this.collection_name.length<1){this.$message.error("文集名称不能为空!");return;}// 发送ajax请求this.$axios.post(`${this.$settings.Host}/article/collection/`,{name: this.collection_name},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("添加文集成功!");this.collection_name = "";this.collection_form = false; // 隐藏添加文集的表单// 把服务端中添加返回的文集信息,保存到collection_list中this.collection_list.unshift(response.data);}).catch(error=>{this.$message.error(error.response.data);});},get_collection(){// 获取用户的文集列表this.$axios.get(`${this.$settings.Host}/article/collection/`,{headers:{Authorization: "jwt " + this.token, // 必须在左边加上 "jwt ",空格!!!}}).then(response=>{this.collection_list = response.data;// 获取当前选中文集的文章列表this.get_article_of_collection();}).catch(error=>{this.$message.error("对不起,无法获取当前用户的文集列表!");});},get_article_of_collection(){// 获取当前文集的文章列表this.$axios.get(`${this.$settings.Host}/article/collection/article/`,{params:{collection_id: this.collection_list[this.current_collection].id,},headers:{Authorization:"jwt " + this.token,}}).then(response=>{this.article_list = response.data;}).catch(error=>{this.$message.error(error.response.data.message);})},add_article(position){// 添加文章this.$axios.post(`${this.$settings.Host}/article/collection/article/`,{collection: this.collection_list[this.current_collection].id,name: this.get_datetime(),position: position,},{headers:{Authorization: "jwt " + this.token,}}).then(response=>{this.$message.success("文章添加成功!");// 给本地的aticle追加/插入新建的文章if(position){// 追加this.article_list.push(response.data);}else{// 插入this.article_list.unshift(response.data);}}).catch(error=>{this.$message.error("对不起,添加文章失败!");});},get_datetime(){// 获取客户端日期格式let datetime = new Date();let Y = datetime.getFullYear();let m = datetime.getMonth()+1;m = m<10?"0"+m:m;let d = datetime.getDate();d = d<10?"0"+d:d;return `${Y}-${m}-${d}`;},timeformat(time){// 获取客户端日期时间格式time = new Date(time);return `${time.getFullYear()}-${time.getMonth()+1}-${time.getDate()} ${time.getHours()}:${time.getMinutes()}`;},pub_article(){// 切换文章的发布状态let article = this.article_list[this.current_article];this.$axios.patch(`${this.$settings.Host}/article/public/${article.id}/`,{},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{// 切换成功if(article.pub_date || article.is_public){// 取消发布article.is_public=false;article.pub_date=null;}else{// 立即发布article.is_public=true;}// 关闭菜单this.is_show_article_menu = false;}).catch(error=>{this.$message.error("切换文章发布状态失败!");})},move_article(collection_id){// 移动文章this.is_show_article_menu = false;let article = this.article_list[this.current_article];this.$axios.put(`${this.$settings.Host}/article/public/${article.id}/`,{collection_id, // collection_id: collection_id的简写,},{headers:{Authorization: "jwt "+this.token,}}).then(response=>{this.article_list.splice(this.current_article,1);}).catch(error=>{this.$message.error("移动文章失败!");});},// 绑定@imgAdd eventimgAdd(pos, $file){// 添加文件},imgDel(pos) {// 删除文件}}}
</script>

celery的定时任务,每分钟执行一次定时发布操作,让pub_date时间到了,则更新对应文章的发布状态。

在mycelery中创建article任务目录,在目录下创建任务文件tasks.py,编写异步任务:

from mycelery.main import app
from article.models import Article
from datetime import datetime
@app.task(name="interval_pub_article")
def interval_pub_article():"""定时发布文章"""article_list = Article.objects.exclude(pub_date=None)for article in article_list:pub_date_timestamp = int(article.pub_date.timestamp())current_timestamp = int(datetime.now().timestamp() +8 * 60 * 60)if pub_date_timestamp <= current_timestamp:article.pub_date = Nonearticle.is_public = Truearticle.save()

注册异步任务到main.py中。并重启celery,代码:

from celery import Celery
# 初始化celery对象
app = Celery("renran")# 初始化django
import os
os.environ.setdefault('DJANGO_SETTINGS_MODULE', 'renranapi.settings.dev')
import django
django.setup()# 加载配置
app.config_from_object("mycelery.config")# 注册异步任务
# 任务以包进行管理,每一个包必须里面包含了一个tasks.py文件
app.autodiscover_tasks(["mycelery.sms","mycelery.article"])# 在终端下面运行celery,将来在线上服务器中,可以使用supervisor以守护进程的方式启动
# celery -A mycelery.main worker --loglevel=info

接下来,我们就可以使用celery的定时任务调度器,让celery定时执行异步任务。

Celery官方文档中关于定时任务使用的说明:

http://docs.celeryproject.org/en/latest/userguide/periodic-tasks.html

在mycelery中config.py配置异步任务定时执行,代码:

# 任务队列的链接地址
broker_url = 'redis://127.0.0.1:6379/15'
# 结果队列的链接地址
result_backend = 'redis://127.0.0.1:6379/14'# 定时任务调度器相关配置
from .main import app
from celery.schedules import crontab
app.conf.beat_schedule = {# 定时任务列表'pub-article-every-one-minute': {'task': 'interval_pub_article', # 指定定时执行的的异步任务# 'schedule': crontab(),         # 时间间隔,一分钟'schedule': 30.0,               # 时间间隔,默认:秒# 'args': (16, 16)              # 如果任务有固定参数,则可以写在args},
}# 和django框架同步时区
from django.conf import settings
app.conf.timezone = settings.TIME_ZONE# 完成上面的配置以后,重启celery并在新的终端窗口执行命令,运行定时任务的调度器
# celery -A mycelery.main beat  # mycelery.main 是celery的主应用文件

接下来,我们就可以重启Celery并启用Celery的定时任务调度器

先在终端下,运行celery的定时任务程序,以下命令:

celery -A mycelery.main beat  # mycelery.main 是celery的主应用文件

然后再新建一个终端,运行以下命令,上面的命令必须先指定:

celery -A mycelery.main worker --loglevel=info

注意,使用的时候,如果有时区必须先配置好系统时区。

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

相关文章

  1. ANSYS报错总结

    遇到问题后,不定期会更新 1.用maxwell16建的Pruis模型存在问题基于workbench平台,optislang和maxwell联合仿真中,报很多错,提示maxwell的PM1和Rotor模块重合 用ANSYS electronics desktop的ACT extension做效率map图时,程序无法运行,报错 Simplorer与maxwell联合仿真时,…...

    2024/5/4 13:53:53
  2. 8.SpringBoot学习(八)——Spring Boot Actuator监控

    1.简介 1.1 概述Spring Boot includes a number of additional features to help you monitor and manage your application when you push it to production. You can choose to manage and monitor your application by using HTTP endpoints or with JMX. Auditing, health,…...

    2024/4/16 17:10:51
  3. 创建具体的产品,并继承产品抽象类

    工厂方法模式定义了一个创建对象的接口,但由子类决定要实例化的类是哪一个。工厂方法让类把实例化推迟到子类。PS:在设计模式中,“实现一个接口”泛指实现某个超类型(可以是类或接口)的某个方法。要点:通过子类来创建具体的对象。客户只需要知道他们所使用的抽象类型即可…...

    2024/4/12 4:25:13
  4. 简单微博爬取以及分析

    对微博评论进行简单爬取并进行分析 任务: 1.爬取评论和时间(request和re) 2.词频统计(jieba) 3.词云展示(wordcloud) 4.时间分布(matplotlib) #heheyang import requests import re import jieba import wordcloud import time as ti import pandas as pd import nump…...

    2024/4/21 19:07:00
  5. 括号匹配问题

    #include <stdio.h> #include <stdbool.h> #include <string.h> int main(){struct stack{//堆char data[20];int top;}s;char str[20]={(,),),[,[,],{,},{,(,),},]};int len=strlen(str);s.top=-1;//初始化for (int i = 0; i < len; ++i) {if(str[i]==(|…...

    2024/5/4 19:28:46
  6. 项目小功能POI

    我们可以利用Apache的POI项目,帮助我们生成Excel,POI不仅能处理Excel,还可以处理word,PowerPoint等格式的文件 实现Excel在线导出如何利用POI生成Excel首先在生成Excel前,我们需要理解一下Excel文件的组织形式。在POI中式这样理解的 :一个Excel对应workbook 一个workbook是由…...

    2024/5/4 18:05:26
  7. renran-文章模块01

    文章模块 创建子引用 cd renranapi/apps python ../../manage.py startapp article注册子应用 INSTALLED_APPS = [# ....article,]模型代码 aricle./models.py,代码: from django.db import models from renranapi.utils.models import BaseModel from users.models import Use…...

    2024/5/4 19:01:00
  8. 2020.7.20认识类和对象

    1.对象是由静态特征和动态特征构成的 静态特征:属性 动态特征:方法 2.类是具有相同特征的对象的集合(抽象) 对象是实体 类是对象的抽象。对象是类的具体实现 面向对象分为:OOA\OOP\OOD 3.方法: 无返回值的时候关注的重点是过程->存储过程 有返回值的时候关注的重点是结…...

    2024/5/4 16:15:52
  9. Linux中安装redis

    安装步骤 安装gcc环境 通过rpm -qa | grep gcc 查看系统是否自带gcc环境,如果没有使用 yum install gcc-c++ 安装 将下载的Redis安装包上传到root文件夹中 解压redis压缩包 tar -zvxf redis-3.0.0.tar.gz 编译redis源码 cd redis-3.0.0 make 安装redis make install PREFIX=/u…...

    2024/4/22 12:24:54
  10. Jsp基础总结

    执行过程当浏览器访问如Html,css,js,图片等静态等资源等时候,服务器找到相应等资源,通过响应实体,转换成流,直接返回给浏览器,让浏览器解析出来;但是如果执行jsp时,因为jsp里可以写java代码,所以不会直接把jsp中的内容直接发送给浏览器,而是要经过翻译,翻译成serv…...

    2024/4/25 0:45:46
  11. 1069 微博转发抽奖 (20分)

    小明 PAT 考了满分,高兴之余决定发起微博转发抽奖活动,从转发的网友中按顺序每隔 N 个人就发出一个红包。请你编写程序帮助他确定中奖名单。 输入格式: 输入第一行给出三个正整数 M(≤ 1000)、N 和 S,分别是转发的总量、小明决定的中奖间隔、以及第一位中奖者的序号(编号…...

    2024/4/9 1:09:24
  12. 设计模式——桥接模式(Singleton Pattern)

    桥接模式桥接模式1、背景2、定义3、特征4、应用参考 桥接模式 1、背景 在生活中,手机的品牌多种多样,手机的软件功能同样五花八门,现在要求不同的品牌手机具有类似的功能,比如通讯录和游戏,如何设计。 1、根据不同的品牌划分,可以得到如下UML图:这样划分存在问题,如果我…...

    2024/4/20 19:51:52
  13. C++面试总结(持续更新)

    C++面经总结1 C++基础1.1 C和C++的区别?1.2 指针和引用的区别1.3 new/delete与malloc/free的区别 1 C++基础 1.1 C和C++的区别? 回答一:C是面向过程的语言,C++是面向对象的语言 C++使用new和delete对内存分配的运算符,取代了c中的malloc和free C++中有引用的概念,C中没有…...

    2024/4/14 11:34:29
  14. EasyTouch_5插件

    1、导入 #pragma warning disable xxxx //xxxx 警告信息不可见2、使用 2.1、EasyTouch4.X版本写法在Hierarchy视图中创建一个Easytouch对象; 新建一个空物体,挂载测试脚本;using System.Collections; using System.Collections.Generic; using UnityEngine; using Hedgeho…...

    2024/4/8 4:20:58
  15. HELLO WORLD(总结与开始)

    这是我的第一篇CSDN文章。 我一直认为自己不是一个CRUD的程序员,我热爱写代码,我愿意搞清楚代码的每一个细节!从我毕业开始,然而,我已经工作将近5年了。 我毕业的时候,我就想,把大学4年的课程串起来。从硬盘开始,当一条指令被当作一个进程,从他被分配进程控制块,到分…...

    2024/4/23 3:31:45
  16. js监听鼠标滚动事件

    js中监听鼠标滚动事件,事件的名称是“mousewheel”,按正常的事件绑定即可,其中有两点关于firefox浏览器的需要注意:1、firefox浏览器的事件名称是“DOMMouseScroll”2、firefox的滑动值和正常的相反的<!DOCTYPE html> <html class="no-js"><head&…...

    2024/5/4 20:21:55
  17. 开发中常用到的@Valid 注解类型的使用

    一:前言搭建springboot项目,采用Restful接口,当前端调用接口或者是其他项目调用时,我们不能单一靠调用方来控制参数的准确性,自己也要对一些非空的值进行判定。二:传统解决方案就是我们自己在代码中对参数一一进行判空操作Model:/*** @author: liman* @Date: 2020/7/2…...

    2024/4/15 15:13:39
  18. PAT考前知识点梳理

    一些常用math函数floor(x):向下取整 ceil(x):向上取整pow(a, b):求a的b次方 sqrt(x):求根号log(x):即求lnx,若要求以a为底的,则必须利用公式:lnb/lna 来替换pi的精确定义为acos(-1.0)一些常用的字符串函数 sscanf(A.c_str()+i,"%d",KA);//把数字型字符…...

    2024/4/9 1:09:17
  19. java中 == 比较的是什么

    对于基本数据类型比较的是数值是否相等对于引用数据类型== 对比的是两个对象基于内存的引用,如果两个对象的引用完全相同(指向同一个对象内存地址),== 返回true,否则返回false值得注意的是即使是两个含义相同但是内存地址不同的两个对象比较也是false...

    2024/4/9 1:09:16
  20. Java 对象创建过程详解

    Java知识点 Java 对象创建过程详解 在java中 对于一个初始化一个对象的代码为 Date d = new Date() 但是,在底层中的实现,是怎么执行的呢?又分为哪些步骤呢? 例如我们拥有一个这样的类public class Test {int x= 1;public static void main(String[] args) {Test test = ne…...

    2024/4/22 0:02:51

最新文章

  1. 对命令模式的理解

    目录 一、场景1、文本编辑器并不是一个好的例子&#xff0c;设备控制器才是2、设备控制器的demo 二、不用命令模式1、代码2、问题 三、使用命令模式1、代码2、当需求变化时2.1 新增代码2.2 优点 四、进一步思考1、省略对Command的建模可以吗&#xff1f;2、命令模式的价值 一、…...

    2024/5/4 21:16:11
  2. 梯度消失和梯度爆炸的一些处理方法

    在这里是记录一下梯度消失或梯度爆炸的一些处理技巧。全当学习总结了如有错误还请留言&#xff0c;在此感激不尽。 权重和梯度的更新公式如下&#xff1a; w w − η ⋅ ∇ w w w - \eta \cdot \nabla w ww−η⋅∇w 个人通俗的理解梯度消失就是网络模型在反向求导的时候出…...

    2024/3/20 10:50:27
  3. C# 构建可定时关闭的异步提示弹窗

    C# 构建可定时关闭的异步提示弹窗 引言1、调用接口的实现2、自动定时窗口的实现 引言 我们在最常用最简单的提示弹框莫过于MessageBox.Show( )的方法了&#xff0c;但是使用久了之后&#xff0c;你会发现这个MessageBox并不是万能的&#xff0c;有事后并不想客户去点击&#x…...

    2024/5/2 6:14:07
  4. c++类的继承方式

    在 C 中&#xff0c;类的继承方式有三种&#xff1a;公有继承&#xff08;public inheritance&#xff09;、保护继承&#xff08;protected inheritance&#xff09;和私有继承&#xff08;private inheritance&#xff09;。这些继承方式决定了派生类对基类成员的访问权限。 …...

    2024/5/3 14:42:38
  5. 416. 分割等和子集问题(动态规划)

    题目 题解 class Solution:def canPartition(self, nums: List[int]) -> bool:# badcaseif not nums:return True# 不能被2整除if sum(nums) % 2 ! 0:return False# 状态定义&#xff1a;dp[i][j]表示当背包容量为j&#xff0c;用前i个物品是否正好可以将背包填满&#xff…...

    2024/5/4 12:05:22
  6. 【Java】ExcelWriter自适应宽度工具类(支持中文)

    工具类 import org.apache.poi.ss.usermodel.Cell; import org.apache.poi.ss.usermodel.CellType; import org.apache.poi.ss.usermodel.Row; import org.apache.poi.ss.usermodel.Sheet;/*** Excel工具类** author xiaoming* date 2023/11/17 10:40*/ public class ExcelUti…...

    2024/5/4 11:23:32
  7. Spring cloud负载均衡@LoadBalanced LoadBalancerClient

    LoadBalance vs Ribbon 由于Spring cloud2020之后移除了Ribbon&#xff0c;直接使用Spring Cloud LoadBalancer作为客户端负载均衡组件&#xff0c;我们讨论Spring负载均衡以Spring Cloud2020之后版本为主&#xff0c;学习Spring Cloud LoadBalance&#xff0c;暂不讨论Ribbon…...

    2024/5/4 14:46:16
  8. TSINGSEE青犀AI智能分析+视频监控工业园区周界安全防范方案

    一、背景需求分析 在工业产业园、化工园或生产制造园区中&#xff0c;周界防范意义重大&#xff0c;对园区的安全起到重要的作用。常规的安防方式是采用人员巡查&#xff0c;人力投入成本大而且效率低。周界一旦被破坏或入侵&#xff0c;会影响园区人员和资产安全&#xff0c;…...

    2024/5/3 16:00:51
  9. VB.net WebBrowser网页元素抓取分析方法

    在用WebBrowser编程实现网页操作自动化时&#xff0c;常要分析网页Html&#xff0c;例如网页在加载数据时&#xff0c;常会显示“系统处理中&#xff0c;请稍候..”&#xff0c;我们需要在数据加载完成后才能继续下一步操作&#xff0c;如何抓取这个信息的网页html元素变化&…...

    2024/5/4 12:10:13
  10. 【Objective-C】Objective-C汇总

    方法定义 参考&#xff1a;https://www.yiibai.com/objective_c/objective_c_functions.html Objective-C编程语言中方法定义的一般形式如下 - (return_type) method_name:( argumentType1 )argumentName1 joiningArgument2:( argumentType2 )argumentName2 ... joiningArgu…...

    2024/5/3 21:22:01
  11. 【洛谷算法题】P5713-洛谷团队系统【入门2分支结构】

    &#x1f468;‍&#x1f4bb;博客主页&#xff1a;花无缺 欢迎 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! 本文由 花无缺 原创 收录于专栏 【洛谷算法题】 文章目录 【洛谷算法题】P5713-洛谷团队系统【入门2分支结构】&#x1f30f;题目描述&#x1f30f;输入格…...

    2024/5/3 23:17:01
  12. 【ES6.0】- 扩展运算符(...)

    【ES6.0】- 扩展运算符... 文章目录 【ES6.0】- 扩展运算符...一、概述二、拷贝数组对象三、合并操作四、参数传递五、数组去重六、字符串转字符数组七、NodeList转数组八、解构变量九、打印日志十、总结 一、概述 **扩展运算符(...)**允许一个表达式在期望多个参数&#xff0…...

    2024/5/4 14:46:12
  13. 摩根看好的前智能硬件头部品牌双11交易数据极度异常!——是模式创新还是饮鸩止渴?

    文 | 螳螂观察 作者 | 李燃 双11狂欢已落下帷幕&#xff0c;各大品牌纷纷晒出优异的成绩单&#xff0c;摩根士丹利投资的智能硬件头部品牌凯迪仕也不例外。然而有爆料称&#xff0c;在自媒体平台发布霸榜各大榜单喜讯的凯迪仕智能锁&#xff0c;多个平台数据都表现出极度异常…...

    2024/5/4 14:46:11
  14. Go语言常用命令详解(二)

    文章目录 前言常用命令go bug示例参数说明 go doc示例参数说明 go env示例 go fix示例 go fmt示例 go generate示例 总结写在最后 前言 接着上一篇继续介绍Go语言的常用命令 常用命令 以下是一些常用的Go命令&#xff0c;这些命令可以帮助您在Go开发中进行编译、测试、运行和…...

    2024/5/4 14:46:11
  15. 用欧拉路径判断图同构推出reverse合法性:1116T4

    http://cplusoj.com/d/senior/p/SS231116D 假设我们要把 a a a 变成 b b b&#xff0c;我们在 a i a_i ai​ 和 a i 1 a_{i1} ai1​ 之间连边&#xff0c; b b b 同理&#xff0c;则 a a a 能变成 b b b 的充要条件是两图 A , B A,B A,B 同构。 必要性显然&#xff0…...

    2024/5/4 2:14:16
  16. 【NGINX--1】基础知识

    1、在 Debian/Ubuntu 上安装 NGINX 在 Debian 或 Ubuntu 机器上安装 NGINX 开源版。 更新已配置源的软件包信息&#xff0c;并安装一些有助于配置官方 NGINX 软件包仓库的软件包&#xff1a; apt-get update apt install -y curl gnupg2 ca-certificates lsb-release debian-…...

    2024/5/3 16:23:03
  17. Hive默认分割符、存储格式与数据压缩

    目录 1、Hive默认分割符2、Hive存储格式3、Hive数据压缩 1、Hive默认分割符 Hive创建表时指定的行受限&#xff08;ROW FORMAT&#xff09;配置标准HQL为&#xff1a; ... ROW FORMAT DELIMITED FIELDS TERMINATED BY \u0001 COLLECTION ITEMS TERMINATED BY , MAP KEYS TERMI…...

    2024/5/4 12:39:12
  18. 【论文阅读】MAG:一种用于航天器遥测数据中有效异常检测的新方法

    文章目录 摘要1 引言2 问题描述3 拟议框架4 所提出方法的细节A.数据预处理B.变量相关分析C.MAG模型D.异常分数 5 实验A.数据集和性能指标B.实验设置与平台C.结果和比较 6 结论 摘要 异常检测是保证航天器稳定性的关键。在航天器运行过程中&#xff0c;传感器和控制器产生大量周…...

    2024/5/4 13:16:06
  19. --max-old-space-size=8192报错

    vue项目运行时&#xff0c;如果经常运行慢&#xff0c;崩溃停止服务&#xff0c;报如下错误 FATAL ERROR: CALL_AND_RETRY_LAST Allocation failed - JavaScript heap out of memory 因为在 Node 中&#xff0c;通过JavaScript使用内存时只能使用部分内存&#xff08;64位系统&…...

    2024/5/4 16:48:41
  20. 基于深度学习的恶意软件检测

    恶意软件是指恶意软件犯罪者用来感染个人计算机或整个组织的网络的软件。 它利用目标系统漏洞&#xff0c;例如可以被劫持的合法软件&#xff08;例如浏览器或 Web 应用程序插件&#xff09;中的错误。 恶意软件渗透可能会造成灾难性的后果&#xff0c;包括数据被盗、勒索或网…...

    2024/5/4 14:46:05
  21. JS原型对象prototype

    让我简单的为大家介绍一下原型对象prototype吧&#xff01; 使用原型实现方法共享 1.构造函数通过原型分配的函数是所有对象所 共享的。 2.JavaScript 规定&#xff0c;每一个构造函数都有一个 prototype 属性&#xff0c;指向另一个对象&#xff0c;所以我们也称为原型对象…...

    2024/5/4 2:00:16
  22. C++中只能有一个实例的单例类

    C中只能有一个实例的单例类 前面讨论的 President 类很不错&#xff0c;但存在一个缺陷&#xff1a;无法禁止通过实例化多个对象来创建多名总统&#xff1a; President One, Two, Three; 由于复制构造函数是私有的&#xff0c;其中每个对象都是不可复制的&#xff0c;但您的目…...

    2024/5/3 22:03:11
  23. python django 小程序图书借阅源码

    开发工具&#xff1a; PyCharm&#xff0c;mysql5.7&#xff0c;微信开发者工具 技术说明&#xff1a; python django html 小程序 功能介绍&#xff1a; 用户端&#xff1a; 登录注册&#xff08;含授权登录&#xff09; 首页显示搜索图书&#xff0c;轮播图&#xff0…...

    2024/5/4 9:07:39
  24. 电子学会C/C++编程等级考试2022年03月(一级)真题解析

    C/C++等级考试(1~8级)全部真题・点这里 第1题:双精度浮点数的输入输出 输入一个双精度浮点数,保留8位小数,输出这个浮点数。 时间限制:1000 内存限制:65536输入 只有一行,一个双精度浮点数。输出 一行,保留8位小数的浮点数。样例输入 3.1415926535798932样例输出 3.1…...

    2024/5/4 14:46:02
  25. 配置失败还原请勿关闭计算机,电脑开机屏幕上面显示,配置失败还原更改 请勿关闭计算机 开不了机 这个问题怎么办...

    解析如下&#xff1a;1、长按电脑电源键直至关机&#xff0c;然后再按一次电源健重启电脑&#xff0c;按F8健进入安全模式2、安全模式下进入Windows系统桌面后&#xff0c;按住“winR”打开运行窗口&#xff0c;输入“services.msc”打开服务设置3、在服务界面&#xff0c;选中…...

    2022/11/19 21:17:18
  26. 错误使用 reshape要执行 RESHAPE,请勿更改元素数目。

    %读入6幅图像&#xff08;每一幅图像的大小是564*564&#xff09; f1 imread(WashingtonDC_Band1_564.tif); subplot(3,2,1),imshow(f1); f2 imread(WashingtonDC_Band2_564.tif); subplot(3,2,2),imshow(f2); f3 imread(WashingtonDC_Band3_564.tif); subplot(3,2,3),imsho…...

    2022/11/19 21:17:16
  27. 配置 已完成 请勿关闭计算机,win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机...

    win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机”问题的解决方法在win7系统关机时如果有升级系统的或者其他需要会直接进入一个 等待界面&#xff0c;在等待界面中我们需要等待操作结束才能关机&#xff0c;虽然这比较麻烦&#xff0c;但是对系统进行配置和升级…...

    2022/11/19 21:17:15
  28. 台式电脑显示配置100%请勿关闭计算机,“准备配置windows 请勿关闭计算机”的解决方法...

    有不少用户在重装Win7系统或更新系统后会遇到“准备配置windows&#xff0c;请勿关闭计算机”的提示&#xff0c;要过很久才能进入系统&#xff0c;有的用户甚至几个小时也无法进入&#xff0c;下面就教大家这个问题的解决方法。第一种方法&#xff1a;我们首先在左下角的“开始…...

    2022/11/19 21:17:14
  29. win7 正在配置 请勿关闭计算机,怎么办Win7开机显示正在配置Windows Update请勿关机...

    置信有很多用户都跟小编一样遇到过这样的问题&#xff0c;电脑时发现开机屏幕显现“正在配置Windows Update&#xff0c;请勿关机”(如下图所示)&#xff0c;而且还需求等大约5分钟才干进入系统。这是怎样回事呢&#xff1f;一切都是正常操作的&#xff0c;为什么开时机呈现“正…...

    2022/11/19 21:17:13
  30. 准备配置windows 请勿关闭计算机 蓝屏,Win7开机总是出现提示“配置Windows请勿关机”...

    Win7系统开机启动时总是出现“配置Windows请勿关机”的提示&#xff0c;没过几秒后电脑自动重启&#xff0c;每次开机都这样无法进入系统&#xff0c;此时碰到这种现象的用户就可以使用以下5种方法解决问题。方法一&#xff1a;开机按下F8&#xff0c;在出现的Windows高级启动选…...

    2022/11/19 21:17:12
  31. 准备windows请勿关闭计算机要多久,windows10系统提示正在准备windows请勿关闭计算机怎么办...

    有不少windows10系统用户反映说碰到这样一个情况&#xff0c;就是电脑提示正在准备windows请勿关闭计算机&#xff0c;碰到这样的问题该怎么解决呢&#xff0c;现在小编就给大家分享一下windows10系统提示正在准备windows请勿关闭计算机的具体第一种方法&#xff1a;1、2、依次…...

    2022/11/19 21:17:11
  32. 配置 已完成 请勿关闭计算机,win7系统关机提示“配置Windows Update已完成30%请勿关闭计算机”的解决方法...

    今天和大家分享一下win7系统重装了Win7旗舰版系统后&#xff0c;每次关机的时候桌面上都会显示一个“配置Windows Update的界面&#xff0c;提示请勿关闭计算机”&#xff0c;每次停留好几分钟才能正常关机&#xff0c;导致什么情况引起的呢&#xff1f;出现配置Windows Update…...

    2022/11/19 21:17:10
  33. 电脑桌面一直是清理请关闭计算机,windows7一直卡在清理 请勿关闭计算机-win7清理请勿关机,win7配置更新35%不动...

    只能是等着&#xff0c;别无他法。说是卡着如果你看硬盘灯应该在读写。如果从 Win 10 无法正常回滚&#xff0c;只能是考虑备份数据后重装系统了。解决来方案一&#xff1a;管理员运行cmd&#xff1a;net stop WuAuServcd %windir%ren SoftwareDistribution SDoldnet start WuA…...

    2022/11/19 21:17:09
  34. 计算机配置更新不起,电脑提示“配置Windows Update请勿关闭计算机”怎么办?

    原标题&#xff1a;电脑提示“配置Windows Update请勿关闭计算机”怎么办&#xff1f;win7系统中在开机与关闭的时候总是显示“配置windows update请勿关闭计算机”相信有不少朋友都曾遇到过一次两次还能忍但经常遇到就叫人感到心烦了遇到这种问题怎么办呢&#xff1f;一般的方…...

    2022/11/19 21:17:08
  35. 计算机正在配置无法关机,关机提示 windows7 正在配置windows 请勿关闭计算机 ,然后等了一晚上也没有关掉。现在电脑无法正常关机...

    关机提示 windows7 正在配置windows 请勿关闭计算机 &#xff0c;然后等了一晚上也没有关掉。现在电脑无法正常关机以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容&#xff0c;让我们赶快一起来看一下吧&#xff01;关机提示 windows7 正在配…...

    2022/11/19 21:17:05
  36. 钉钉提示请勿通过开发者调试模式_钉钉请勿通过开发者调试模式是真的吗好不好用...

    钉钉请勿通过开发者调试模式是真的吗好不好用 更新时间:2020-04-20 22:24:19 浏览次数:729次 区域: 南阳 > 卧龙 列举网提醒您:为保障您的权益,请不要提前支付任何费用! 虚拟位置外设器!!轨迹模拟&虚拟位置外设神器 专业用于:钉钉,外勤365,红圈通,企业微信和…...

    2022/11/19 21:17:05
  37. 配置失败还原请勿关闭计算机怎么办,win7系统出现“配置windows update失败 还原更改 请勿关闭计算机”,长时间没反应,无法进入系统的解决方案...

    前几天班里有位学生电脑(windows 7系统)出问题了&#xff0c;具体表现是开机时一直停留在“配置windows update失败 还原更改 请勿关闭计算机”这个界面&#xff0c;长时间没反应&#xff0c;无法进入系统。这个问题原来帮其他同学也解决过&#xff0c;网上搜了不少资料&#x…...

    2022/11/19 21:17:04
  38. 一个电脑无法关闭计算机你应该怎么办,电脑显示“清理请勿关闭计算机”怎么办?...

    本文为你提供了3个有效解决电脑显示“清理请勿关闭计算机”问题的方法&#xff0c;并在最后教给你1种保护系统安全的好方法&#xff0c;一起来看看&#xff01;电脑出现“清理请勿关闭计算机”在Windows 7(SP1)和Windows Server 2008 R2 SP1中&#xff0c;添加了1个新功能在“磁…...

    2022/11/19 21:17:03
  39. 请勿关闭计算机还原更改要多久,电脑显示:配置windows更新失败,正在还原更改,请勿关闭计算机怎么办...

    许多用户在长期不使用电脑的时候&#xff0c;开启电脑发现电脑显示&#xff1a;配置windows更新失败&#xff0c;正在还原更改&#xff0c;请勿关闭计算机。。.这要怎么办呢&#xff1f;下面小编就带着大家一起看看吧&#xff01;如果能够正常进入系统&#xff0c;建议您暂时移…...

    2022/11/19 21:17:02
  40. 还原更改请勿关闭计算机 要多久,配置windows update失败 还原更改 请勿关闭计算机,电脑开机后一直显示以...

    配置windows update失败 还原更改 请勿关闭计算机&#xff0c;电脑开机后一直显示以以下文字资料是由(历史新知网www.lishixinzhi.com)小编为大家搜集整理后发布的内容&#xff0c;让我们赶快一起来看一下吧&#xff01;配置windows update失败 还原更改 请勿关闭计算机&#x…...

    2022/11/19 21:17:01
  41. 电脑配置中请勿关闭计算机怎么办,准备配置windows请勿关闭计算机一直显示怎么办【图解】...

    不知道大家有没有遇到过这样的一个问题&#xff0c;就是我们的win7系统在关机的时候&#xff0c;总是喜欢显示“准备配置windows&#xff0c;请勿关机”这样的一个页面&#xff0c;没有什么大碍&#xff0c;但是如果一直等着的话就要两个小时甚至更久都关不了机&#xff0c;非常…...

    2022/11/19 21:17:00
  42. 正在准备配置请勿关闭计算机,正在准备配置windows请勿关闭计算机时间长了解决教程...

    当电脑出现正在准备配置windows请勿关闭计算机时&#xff0c;一般是您正对windows进行升级&#xff0c;但是这个要是长时间没有反应&#xff0c;我们不能再傻等下去了。可能是电脑出了别的问题了&#xff0c;来看看教程的说法。正在准备配置windows请勿关闭计算机时间长了方法一…...

    2022/11/19 21:16:59
  43. 配置失败还原请勿关闭计算机,配置Windows Update失败,还原更改请勿关闭计算机...

    我们使用电脑的过程中有时会遇到这种情况&#xff0c;当我们打开电脑之后&#xff0c;发现一直停留在一个界面&#xff1a;“配置Windows Update失败&#xff0c;还原更改请勿关闭计算机”&#xff0c;等了许久还是无法进入系统。如果我们遇到此类问题应该如何解决呢&#xff0…...

    2022/11/19 21:16:58
  44. 如何在iPhone上关闭“请勿打扰”

    Apple’s “Do Not Disturb While Driving” is a potentially lifesaving iPhone feature, but it doesn’t always turn on automatically at the appropriate time. For example, you might be a passenger in a moving car, but your iPhone may think you’re the one dri…...

    2022/11/19 21:16:57