Sheet comment management.

pull/9/head
ruibaby 2019-05-15 10:53:20 +08:00
parent e9e2c7a2d0
commit 289996aa95
4 changed files with 506 additions and 431 deletions

View File

@ -20,6 +20,36 @@ function latestComment(target, top, status) {
})
}
commentApi.queryComment = (target, params) => {
return service({
url: `${baseUrl}/${target}/comments`,
params: params,
method: 'get'
})
}
commentApi.updateStatus = (target, commentId, status) => {
return service({
url: `${baseUrl}/${target}/comments/${commentId}/status/${status}`,
method: 'put'
})
}
commentApi.delete = (target, commentId) => {
return service({
url: `${baseUrl}/${target}/comments/${commentId}`,
method: 'delete'
})
}
commentApi.create = (target, comment) => {
return service({
url: `${baseUrl}/${target}/comments`,
data: comment,
method: 'post'
})
}
/**
* Creates a comment.
* @param {String} target

View File

@ -527,7 +527,13 @@ body {
.ant-card-wider-padding {
.ant-card-body {
padding: 24px !important;
padding: 16px !important;
}
}
.comment-tab-wrapper{
.ant-card-body {
padding: 0 !important;
}
}

View File

@ -1,446 +1,35 @@
<template>
<page-view>
<a-card :bordered="false">
<div class="table-page-search-wrapper">
<a-form layout="inline">
<a-row :gutter="48">
<a-col
:md="6"
:sm="24"
>
<a-form-item label="关键词">
<a-input v-model="queryParam.keyword" />
</a-form-item>
</a-col>
<a-col
:md="6"
:sm="24"
>
<a-form-item label="评论状态">
<a-select
v-model="queryParam.status"
placeholder="请选择评论状态"
@change="loadComments(isSearch)"
>
<a-select-option
v-for="status in Object.keys(commentStatus)"
:key="status"
:value="status"
>{{ commentStatus[status].text }}</a-select-option>
</a-select>
</a-form-item>
</a-col>
<a-col
:md="12"
:sm="24"
>
<span class="table-page-search-submitButtons">
<a-button
type="primary"
@click="loadComments(isSearch)"
>查询</a-button>
<a-button
style="margin-left: 8px;"
@click="handleResetParam"
>重置</a-button>
</span>
</a-col>
</a-row>
</a-form>
</div>
<div class="table-operator">
<a-dropdown v-show="queryParam.status!=null && queryParam.status!=''">
<a-menu slot="overlay">
<a-menu-item
key="1"
v-if="queryParam.status ==='AUDITING'"
>
<a
href="javascript:void(0);"
@click="handlePublishMore"
>
通过
</a>
</a-menu-item>
<a-menu-item
key="2"
v-if="queryParam.status === 'PUBLISHED' || queryParam.status ==='AUDITING'"
>
<a
href="javascript:void(0);"
@click="handleRecycleMore"
>
移到回收站
</a>
</a-menu-item>
<a-menu-item
key="3"
v-if="queryParam.status === 'RECYCLE'"
>
<a
href="javascript:void(0);"
@click="handleDeleteMore"
>
永久删除
</a>
</a-menu-item>
</a-menu>
<a-button>
批量操作
<a-icon type="down" />
</a-button>
</a-dropdown>
</div>
<div style="margin-top:15px">
<a-table
:rowKey="comment => comment.id"
:rowSelection="{
onChange: onSelectionChange,
getCheckboxProps: getCheckboxProps
}"
:columns="columns"
:dataSource="formattedComments"
:loading="commentsLoading"
:pagination="false"
<div class="card-container">
<a-tabs type="card">
<a-tab-pane
key="1"
tab="文章"
>
<p
class="comment-content-wrapper"
slot="content"
slot-scope="content"
v-html="content"
>
</p>
<span
slot="status"
slot-scope="statusProperty"
>
<a-badge :status="statusProperty.status" />
{{ statusProperty.text }}
</span>
<a
slot="post"
slot-scope="post"
:href="options.blog_url+'/archives/'+post.url"
target="_blank"
>{{ post.title }}</a>
<span
slot="createTime"
slot-scope="createTime"
>{{ createTime | timeAgo }}</span>
<span
slot="action"
slot-scope="text, record"
>
<a-dropdown
:trigger="['click']"
v-if="record.status === 'AUDITING'"
>
<a
href="javascript:void(0);"
class="ant-dropdown-link"
>通过</a>
<a-menu slot="overlay">
<a-menu-item key="1">
<a
href="javascript:void(0);"
@click="handleEditStatusClick(record.id,'PUBLISHED')"
>通过</a>
</a-menu-item>
<a-menu-item key="2">
<a
href="javascript:void(0);"
@click="handleReplyAndPassClick(record)"
>通过并回复</a>
</a-menu-item>
</a-menu>
</a-dropdown>
<a
href="javascript:void(0);"
v-else-if="record.status === 'PUBLISHED'"
@click="handleReplyClick(record)"
>回复</a>
<a-popconfirm
:title="'你确定要还原该评论?'"
@confirm="handleEditStatusClick(record.id,'PUBLISHED')"
okText="确定"
cancelText="取消"
v-else-if="record.status === 'RECYCLE'"
>
<a href="javascript:;">还原</a>
</a-popconfirm>
<a-divider type="vertical" />
<a-popconfirm
:title="'你确定要将该评论移到回收站?'"
@confirm="handleEditStatusClick(record.id,'RECYCLE')"
okText="确定"
cancelText="取消"
v-if="record.status === 'PUBLISHED' || record.status === 'AUDITING'"
>
<a href="javascript:;">回收站</a>
</a-popconfirm>
<a-popconfirm
:title="'你确定要永久删除该评论?'"
@confirm="handleDeleteClick(record.id)"
okText="确定"
cancelText="取消"
v-else-if="record.status === 'RECYCLE'"
>
<a href="javascript:;">删除</a>
</a-popconfirm>
</span>
</a-table>
<div class="page-wrapper">
<a-pagination
class="pagination"
:total="pagination.total"
:pageSizeOptions="['1', '2', '5', '10', '20', '50', '100']"
showSizeChanger
@showSizeChange="handlePaginationChange"
@change="handlePaginationChange"
/>
</div>
</div>
</a-card>
<a-modal
v-if="selectComment"
:title="'回复给:'+selectComment.author"
v-model="replyCommentVisible"
@close="onReplyClose"
>
<template slot="footer">
<a-button
key="submit"
type="primary"
@click="handleCreateClick"
<comment-tab type="posts"></comment-tab>
</a-tab-pane>
<a-tab-pane
key="2"
tab="页面"
>
回复
</a-button>
</template>
<a-form layout="vertical">
<a-form-item>
<a-input
type="textarea"
:autosize="{ minRows: 8 }"
v-model="replyComment.content"
/>
</a-form-item>
</a-form>
</a-modal>
<comment-tab type="sheets"></comment-tab>
</a-tab-pane>
</a-tabs>
</div>
</page-view>
</template>
<script>
import { PageView } from '@/layouts'
import postCommentApi from '@/api/postComment'
import optionApi from '@/api/option'
import marked from 'marked'
const columns = [
{
title: '昵称',
dataIndex: 'author'
},
{
title: '内容',
dataIndex: 'content',
scopedSlots: { customRender: 'content' }
},
{
title: '状态',
className: 'status',
dataIndex: 'statusProperty',
scopedSlots: { customRender: 'status' }
},
{
title: '评论页面',
dataIndex: 'post',
scopedSlots: { customRender: 'post' }
},
{
title: '日期',
dataIndex: 'createTime',
scopedSlots: { customRender: 'createTime' }
},
{
title: '操作',
dataIndex: 'action',
width: '150px',
scopedSlots: { customRender: 'action' }
}
]
import CommentTab from './CommentTab'
export default {
name: 'CommentList',
components: {
PageView
PageView,
CommentTab
},
data() {
return {
columns,
replyCommentVisible: false,
pagination: {
current: 1,
pageSize: 10,
sort: null
},
queryParam: {
page: 0,
size: 10,
sort: null,
keyword: null,
status: null
},
selectedRowKeys: [],
selectedRows: [],
comments: [],
selectComment: {},
replyComment: {},
commentsLoading: false,
commentStatus: postCommentApi.commentStatus,
options: [],
keys: ['blog_url']
}
return {}
},
computed: {
formattedComments() {
return this.comments.map(comment => {
comment.statusProperty = this.commentStatus[comment.status]
comment.content = marked(comment.content, { sanitize: true })
return comment
})
}
},
created() {
this.loadComments()
this.loadOptions()
},
methods: {
loadComments(isSearch) {
this.commentsLoading = true
// Set from pagination
this.queryParam.page = this.pagination.current - 1
this.queryParam.size = this.pagination.pageSize
this.queryParam.sort = this.pagination.sort
if (isSearch) {
this.queryParam.page = 0
}
postCommentApi.query(this.queryParam).then(response => {
this.comments = response.data.data.content
this.pagination.total = response.data.data.total
this.commentsLoading = false
})
},
loadOptions() {
optionApi.listAll(this.keys).then(response => {
this.options = response.data.data
})
},
handleEditComment(id) {
this.$message.success('编辑')
},
handleEditStatusClick(commentId, status) {
postCommentApi.updateStatus(commentId, status).then(response => {
this.$message.success('操作成功!')
this.loadComments()
})
},
handleDeleteClick(commentId) {
postCommentApi.delete(commentId).then(response => {
this.$message.success('删除成功!')
this.loadComments()
})
},
handleReplyAndPassClick(comment) {
this.handleReplyClick(comment)
this.handleEditStatusClick(comment.id, 'PUBLISHED')
},
handleReplyClick(comment) {
this.selectComment = comment
this.replyCommentVisible = true
this.replyComment.parentId = comment.id
this.replyComment.postId = comment.post.id
},
handleCreateClick() {
postCommentApi.create(this.replyComment).then(response => {
this.$message.success('回复成功!')
this.replyComment = {}
this.selectComment = {}
this.replyCommentVisible = false
this.loadComments()
})
},
handlePaginationChange(page, pageSize) {
this.$log.debug(`Current: ${page}, PageSize: ${pageSize}`)
this.pagination.current = page
this.pagination.pageSize = pageSize
this.loadComments()
},
handleResetParam() {
this.queryParam.keyword = null
this.queryParam.status = null
this.loadComments()
},
handlePublishMore() {
if (this.selectedRowKeys.length <= 0) {
this.$message.success('请至少选择一项!')
}
for (let index = 0; index < this.selectedRowKeys.length; index++) {
const element = this.selectedRowKeys[index]
postCommentApi.updateStatus(element, 'PUBLISHED').then(response => {
this.$log.debug(`commentId: ${element}, status: PUBLISHED`)
})
}
this.loadComments()
},
handleRecycleMore() {
if (this.selectedRowKeys.length <= 0) {
this.$message.success('请至少选择一项!')
}
for (let index = 0; index < this.selectedRowKeys.length; index++) {
const element = this.selectedRowKeys[index]
postCommentApi.updateStatus(element, 'RECYCLE').then(response => {
this.$log.debug(`commentId: ${element}, status: RECYCLE`)
})
}
this.loadComments()
},
handleDeleteMore() {
if (this.selectedRowKeys.length <= 0) {
this.$message.success('请至少选择一项!')
}
for (let index = 0; index < this.selectedRowKeys.length; index++) {
const element = this.selectedRowKeys[index]
postCommentApi.delete(element).then(response => {
this.$log.debug(`delete: ${element}`)
})
}
this.loadComments()
},
onReplyClose() {
this.replyComment = {}
this.selectComment = {}
this.replyCommentVisible = false
},
onSelectionChange(selectedRowKeys) {
this.selectedRowKeys = selectedRowKeys
this.$log.debug(`SelectedRowKeys: ${selectedRowKeys}`)
},
getCheckboxProps(comment) {
return {
props: {
disabled: comment.status === 'RECYCLE',
name: comment.author
}
}
}
}
methods: {}
}
</script>

View File

@ -0,0 +1,450 @@
<template>
<div class="comment-tab-wrapper">
<a-card :bordered="false">
<div class="table-page-search-wrapper">
<a-form layout="inline">
<a-row :gutter="48">
<a-col
:md="6"
:sm="24"
>
<a-form-item label="关键词">
<a-input v-model="queryParam.keyword" />
</a-form-item>
</a-col>
<a-col
:md="6"
:sm="24"
>
<a-form-item label="评论状态">
<a-select
v-model="queryParam.status"
placeholder="请选择评论状态"
@change="handleQuery"
>
<a-select-option
v-for="status in Object.keys(commentStatus)"
:key="status"
:value="status"
>{{ commentStatus[status].text }}</a-select-option>
</a-select>
</a-form-item>
</a-col>
<a-col
:md="12"
:sm="24"
>
<span class="table-page-search-submitButtons">
<a-button
type="primary"
@click="handleQuery"
>查询</a-button>
<a-button
style="margin-left: 8px;"
@click="handleResetParam"
>重置</a-button>
</span>
</a-col>
</a-row>
</a-form>
</div>
<div class="table-operator">
<a-dropdown v-show="queryParam.status!=null && queryParam.status!=''">
<a-menu slot="overlay">
<a-menu-item
key="1"
v-if="queryParam.status ==='AUDITING'"
>
<a
href="javascript:void(0);"
@click="handlePublishMore"
>
通过
</a>
</a-menu-item>
<a-menu-item
key="2"
v-if="queryParam.status === 'PUBLISHED' || queryParam.status ==='AUDITING'"
>
<a
href="javascript:void(0);"
@click="handleRecycleMore"
>
移到回收站
</a>
</a-menu-item>
<a-menu-item
key="3"
v-if="queryParam.status === 'RECYCLE'"
>
<a
href="javascript:void(0);"
@click="handleDeleteMore"
>
永久删除
</a>
</a-menu-item>
</a-menu>
<a-button>
批量操作
<a-icon type="down" />
</a-button>
</a-dropdown>
</div>
<div style="margin-top:15px">
<a-table
:rowKey="comment => comment.id"
:rowSelection="{
onChange: onSelectionChange,
getCheckboxProps: getCheckboxProps
}"
:columns="columns"
:dataSource="formattedComments"
:loading="loading"
:pagination="false"
>
<p
class="comment-content-wrapper"
slot="content"
slot-scope="content"
v-html="content"
>
</p>
<span
slot="status"
slot-scope="statusProperty"
>
<a-badge :status="statusProperty.status" />
{{ statusProperty.text }}
</span>
<a
slot="post"
slot-scope="post"
:href="options.blog_url+'/archives/'+post.url"
target="_blank"
>{{ post.title }}</a>
<span
slot="createTime"
slot-scope="createTime"
>{{ createTime | timeAgo }}</span>
<span
slot="action"
slot-scope="text, record"
>
<a-dropdown
:trigger="['click']"
v-if="record.status === 'AUDITING'"
>
<a
href="javascript:void(0);"
class="ant-dropdown-link"
>通过</a>
<a-menu slot="overlay">
<a-menu-item key="1">
<a
href="javascript:void(0);"
@click="handleEditStatusClick(record.id,'PUBLISHED')"
>通过</a>
</a-menu-item>
<a-menu-item key="2">
<a
href="javascript:void(0);"
@click="handleReplyAndPassClick(record)"
>通过并回复</a>
</a-menu-item>
</a-menu>
</a-dropdown>
<a
href="javascript:void(0);"
v-else-if="record.status === 'PUBLISHED'"
@click="handleReplyClick(record)"
>回复</a>
<a-popconfirm
:title="'你确定要还原该评论?'"
@confirm="handleEditStatusClick(record.id,'PUBLISHED')"
okText="确定"
cancelText="取消"
v-else-if="record.status === 'RECYCLE'"
>
<a href="javascript:;">还原</a>
</a-popconfirm>
<a-divider type="vertical" />
<a-popconfirm
:title="'你确定要将该评论移到回收站?'"
@confirm="handleEditStatusClick(record.id,'RECYCLE')"
okText="确定"
cancelText="取消"
v-if="record.status === 'PUBLISHED' || record.status === 'AUDITING'"
>
<a href="javascript:;">回收站</a>
</a-popconfirm>
<a-popconfirm
:title="'你确定要永久删除该评论?'"
@confirm="handleDeleteClick(record.id)"
okText="确定"
cancelText="取消"
v-else-if="record.status === 'RECYCLE'"
>
<a href="javascript:;">删除</a>
</a-popconfirm>
</span>
</a-table>
<div class="page-wrapper">
<a-pagination
class="pagination"
:total="pagination.total"
:pageSizeOptions="['1', '2', '5', '10', '20', '50', '100']"
showSizeChanger
@showSizeChange="handlePaginationChange"
@change="handlePaginationChange"
/>
</div>
</div>
</a-card>
<a-modal
v-if="selectComment"
:title="'回复给:'+selectComment.author"
v-model="replyCommentVisible"
@close="onReplyClose"
>
<template slot="footer">
<a-button
key="submit"
type="primary"
@click="handleCreateClick"
>
回复
</a-button>
</template>
<a-form layout="vertical">
<a-form-item>
<a-input
type="textarea"
:autosize="{ minRows: 8 }"
v-model="replyComment.content"
/>
</a-form-item>
</a-form>
</a-modal>
</div>
</template>
<script>
import commentApi from '@/api/comment'
import optionApi from '@/api/option'
import marked from 'marked'
const columns = [
{
title: '昵称',
dataIndex: 'author'
},
{
title: '内容',
dataIndex: 'content',
scopedSlots: { customRender: 'content' }
},
{
title: '状态',
className: 'status',
dataIndex: 'statusProperty',
scopedSlots: { customRender: 'status' }
},
{
title: '评论页面',
dataIndex: 'post',
scopedSlots: { customRender: 'post' }
},
{
title: '日期',
dataIndex: 'createTime',
scopedSlots: { customRender: 'createTime' }
},
{
title: '操作',
dataIndex: 'action',
width: '150px',
scopedSlots: { customRender: 'action' }
}
]
export default {
name: 'CommentTab',
props: {
type: {
type: String,
required: false,
default: 'post',
validator: function(value) {
return ['posts', 'sheets', 'journals'].indexOf(value) !== -1
}
}
},
data() {
return {
columns,
replyCommentVisible: false,
pagination: {
current: 1,
pageSize: 10,
sort: null
},
queryParam: {
page: 0,
size: 10,
sort: null,
keyword: null,
status: null
},
selectedRowKeys: [],
selectedRows: [],
comments: [],
selectComment: {},
replyComment: {},
loading: false,
commentStatus: commentApi.commentStatus,
options: [],
keys: ['blog_url']
}
},
created() {
this.loadComments()
this.loadOptions()
},
computed: {
formattedComments() {
return this.comments.map(comment => {
comment.statusProperty = this.commentStatus[comment.status]
comment.content = marked(comment.content, { sanitize: true })
return comment
})
}
},
methods: {
loadComments() {
this.loading = true
this.queryParam.page = this.pagination.current - 1
this.queryParam.size = this.pagination.pageSize
this.queryParam.sort = this.pagination.sort
commentApi.queryComment(this.type, this.queryParam).then(response => {
this.comments = response.data.data.content
this.pagination.total = response.data.data.total
this.loading = false
})
},
handleQuery() {
this.queryParam.page = 0
this.loadComments()
},
loadOptions() {
optionApi.listAll(this.keys).then(response => {
this.options = response.data.data
})
},
handleEditComment(id) {
this.$message.success('编辑')
},
handleEditStatusClick(commentId, status) {
commentApi.updateStatus(this.type, commentId, status).then(response => {
this.$message.success('操作成功!')
this.loadComments()
})
},
handleDeleteClick(commentId) {
commentApi.delete(this.type, commentId).then(response => {
this.$message.success('删除成功!')
this.loadComments()
})
},
handleReplyAndPassClick(comment) {
this.handleReplyClick(comment)
this.handleEditStatusClick(comment.id, 'PUBLISHED')
},
handleReplyClick(comment) {
this.selectComment = comment
this.replyCommentVisible = true
this.replyComment.parentId = comment.id
this.replyComment.postId = comment.post.id
},
handleCreateClick() {
commentApi.create(this.type, this.replyComment).then(response => {
this.$message.success('回复成功!')
this.replyComment = {}
this.selectComment = {}
this.replyCommentVisible = false
this.loadComments()
})
},
handlePaginationChange(page, pageSize) {
this.$log.debug(`Current: ${page}, PageSize: ${pageSize}`)
this.pagination.current = page
this.pagination.pageSize = pageSize
this.loadComments()
},
handleResetParam() {
this.queryParam.keyword = null
this.queryParam.status = null
this.loadComments()
},
handlePublishMore() {
if (this.selectedRowKeys.length <= 0) {
this.$message.success('请至少选择一项!')
}
for (let index = 0; index < this.selectedRowKeys.length; index++) {
const element = this.selectedRowKeys[index]
commentApi.updateStatus(this.type, element, 'PUBLISHED').then(response => {
this.$log.debug(`commentId: ${element}, status: PUBLISHED`)
})
}
this.loadComments()
},
handleRecycleMore() {
if (this.selectedRowKeys.length <= 0) {
this.$message.success('请至少选择一项!')
}
for (let index = 0; index < this.selectedRowKeys.length; index++) {
const element = this.selectedRowKeys[index]
commentApi.updateStatus(this.type, element, 'RECYCLE').then(response => {
this.$log.debug(`commentId: ${element}, status: RECYCLE`)
})
}
this.loadComments()
},
handleDeleteMore() {
if (this.selectedRowKeys.length <= 0) {
this.$message.success('请至少选择一项!')
}
for (let index = 0; index < this.selectedRowKeys.length; index++) {
const element = this.selectedRowKeys[index]
commentApi.delete(this.type, element).then(response => {
this.$log.debug(`delete: ${element}`)
})
}
this.loadComments()
},
onReplyClose() {
this.replyComment = {}
this.selectComment = {}
this.replyCommentVisible = false
},
onSelectionChange(selectedRowKeys) {
this.selectedRowKeys = selectedRowKeys
this.$log.debug(`SelectedRowKeys: ${selectedRowKeys}`)
},
getCheckboxProps(comment) {
return {
props: {
disabled: comment.status === 'RECYCLE',
name: comment.author
}
}
}
}
}
</script>