| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427 |
- <template>
- <el-card class="comment-card shadow-box">
- <div class="comment-container">
- <div class="comment-header">
- <span class="total-count">{{ totalSize }} 评论</span>
- <div class="sort-options">
- <span :class="{ active: sortType === 'hot' }" @click="handleSort('hot')">最热</span>
- <el-divider direction="vertical" />
- <span :class="{ active: sortType === 'time' }" @click="handleSort('time')">最新</span>
- </div>
- </div>
- <div class="main-reply-box">
- <el-avatar
- :size="48"
- :src="currentUser.avatar || defaultAvatar"
- class="clickable-avatar"
- @click.native="goToUserHome(currentUser.userId)"
- />
- <div class="reply-input-wrapper">
- <el-input
- v-model="newComment"
- type="textarea"
- :rows="3"
- placeholder="发一条友善的评论吧"
- resize="none"
- />
- <el-button type="primary" class="post-btn" @click="postMainComment">发表评论</el-button>
- </div>
- </div>
- <div
- v-infinite-scroll="loadMoreMainComments"
- class="comment-list"
- :infinite-scroll-disabled="mainScrollDisabled"
- :infinite-scroll-distance="20"
- >
- <div v-for="item in dataList" :key="item.id" class="comment-item">
- <div class="comment-root">
- <el-avatar
- :size="48"
- :src="item.user.avatar || defaultAvatar"
- class="clickable-avatar"
- @click.native="goToUserHome(item.user.userId || item.user.id)"
- />
- <div class="comment-content">
- <div class="user-info">
- <span class="user-name clickable-link" @click="goToUserHome(item.user.userId || item.user.id)">
- {{ item.user.name }}
- </span>
- </div>
- <p class="text">{{ item.content }}</p>
- <div class="comment-footer">
- <span class="time">{{ formatDate(item.createAt) }}</span>
- <span class="action" @click="showReplyInput(item, item)">回复</span>
- </div>
- <div v-if="item.total > 0" class="reply-list">
- <div v-for="reply in item.children.slice(0, 3)" :key="reply.id" class="reply-item">
- <el-avatar
- :size="24"
- :src="reply.user.avatar || defaultAvatar"
- class="reply-avatar clickable-avatar"
- @click.native="goToUserHome(reply.user.userId || reply.user.id)"
- />
- <div class="reply-right">
- <span class="user-name clickable-link" @click="goToUserHome(reply.user.userId || reply.user.id)">
- {{ reply.user.name }}
- </span>
- <span v-if="reply.targetId && reply.targetId !== item.id">
- 回复
- <span class="target-name clickable-link" @click="goToUserHome(reply.targetUserId)">
- @{{ reply.targetUsername }}
- </span> :
- </span>
- <span class="reply-content">{{ reply.content }}</span>
- <span class="sub-action" @click="showReplyInput(item, reply)">回复</span>
- </div>
- </div>
- <div v-if="item.total > 3" class="view-more-bar">
- 共 {{ item.total }} 条回复,
- <span class="click-more" @click="openReplyDialog(item)">点击查看更多</span>
- </div>
- </div>
- <div v-if="activeRootId === item.id && !dialogVisible" class="inner-reply-box">
- <el-input v-model="replyText" size="small" :placeholder="replyPlaceholder" />
- <el-button type="primary" size="small" @click="postReply">发布</el-button>
- <el-button type="text" size="small" @click="activeRootId = null">取消</el-button>
- </div>
- </div>
- </div>
- <el-divider />
- </div>
- <p v-if="loading" class="list-status">加载中...</p>
- <p v-if="noMoreMain && dataList.length > 0" class="list-status">没有更多评论了</p>
- <el-empty v-if="!loading && dataList.length === 0" description="暂无评论,快来抢沙发" />
- </div>
- <el-dialog :visible.sync="dialogVisible" title="评论详情" width="600px" :append-to-body="true" custom-class="reply-dialog" @close="activeRootId = null">
- <div v-if="currentParent" class="dialog-scroll-area">
- <div class="parent-info">
- <el-avatar
- :size="40"
- :src="currentParent.user.avatar || defaultAvatar"
- class="clickable-avatar"
- @click.native="goToUserHome(currentParent.user.userId || currentParent.user.id)"
- />
- <div class="parent-text">
- <div class="user-name clickable-link" @click="goToUserHome(currentParent.user.userId || currentParent.user.id)">
- {{ currentParent.user.name }}
- </div>
- <div class="content">{{ currentParent.content }}</div>
- <div class="reply-footer">
- <span class="time">{{ formatDate(currentParent.createAt) }}</span>
- <span class="action" @click="showReplyInput(currentParent, currentParent)">回复</span>
- </div>
- </div>
- </div>
- <el-divider content-position="left">全部回复 ({{ currentParent.total }})</el-divider>
- <div class="dialog-reply-list">
- <div v-for="sub in dialogReplies" :key="sub.id" class="dialog-reply-item">
- <el-avatar
- :size="32"
- :src="sub.user.avatar || defaultAvatar"
- class="clickable-avatar"
- @click.native="goToUserHome(sub.user.userId || sub.user.id)"
- />
- <div class="reply-body">
- <span class="user-name clickable-link" @click="goToUserHome(sub.user.userId || sub.user.id)">
- {{ sub.user.name }}
- </span>
- <span v-if="sub.targetId && sub.targetId !== currentParent.id">
- 回复
- <span class="target-name clickable-link" @click="goToUserHome(sub.targetUserId)">
- @{{ sub.targetUsername }}
- </span>
- </span>: {{ sub.content }}
- <div class="reply-footer">
- {{ formatDate(sub.createAt) }}
- <span class="action" @click="showReplyInput(currentParent, sub)">回复</span>
- </div>
- </div>
- </div>
- </div>
- <div class="dialog-pagination">
- <el-pagination
- small
- background
- layout="prev, pager, next"
- :total="currentParent.total"
- :page-size="childPageSize"
- :current-page.sync="childCurrentPage"
- @current-change="handleChildPageChange"
- />
- </div>
- </div>
- <div v-if="dialogVisible && activeRootId === currentParent.id" class="dialog-inner-reply">
- <el-divider />
- <div class="inner-reply-box">
- <el-input v-model="replyText" size="small" :placeholder="replyPlaceholder" />
- <el-button type="primary" size="small" @click="postReply">发布</el-button>
- <el-button type="text" size="small" @click="activeRootId = null">取消</el-button>
- </div>
- </div>
- </el-dialog>
- </div>
- </el-card>
- </template>
- <script>
- import { getChildComment, getComment, publishComment } from '@/api/comment'
- export default {
- name: 'UserCommentCard',
- props: {
- videoId: { type: String, required: true },
- currentUser: { type: Object, default: () => ({ userId: -1, name: '游客', avatar: '' }) }
- },
- data() {
- return {
- dataList: [],
- totalSize: 0,
- currentPage: 0,
- pageSize: 10,
- loading: false,
- noMoreMain: false,
- sortType: 'hot',
- dialogVisible: false,
- currentParent: null,
- dialogReplies: [],
- childCurrentPage: 1,
- childPageSize: 10,
- newComment: '',
- replyText: '',
- activeRootId: null,
- replyTarget: null,
- replyPlaceholder: '',
- defaultAvatar: 'https://cube.elemecdn.com/3/7c/3ea6beec64369c2642b92c6726f1epng.png'
- }
- },
- computed: {
- mainScrollDisabled() {
- return this.loading || this.noMoreMain
- }
- },
- watch: {
- videoId: {
- immediate: true,
- handler(val) {
- if (val) this.initCommentList()
- }
- }
- },
- methods: {
- goToUserHome(userId) {
- this.$message.info('goto userhome ' + userId)
- },
- initCommentList() {
- this.dataList = []
- this.currentPage = 0
- this.noMoreMain = false
- this.loadMoreMainComments()
- },
- async loadMoreMainComments() {
- if (this.loading || this.noMoreMain) return
- this.loading = true
- this.currentPage++
- try {
- const resp = await getComment({
- videoId: this.videoId,
- pn: this.currentPage,
- pageSize: this.pageSize,
- sortBy: this.sortType
- })
- if (resp.code === 0) {
- const newList = resp.data.list || []
- this.dataList = [...this.dataList, ...newList]
- this.totalSize = resp.data.totalSize
- if (newList.length < this.pageSize || this.dataList.length >= this.totalSize) {
- this.noMoreMain = true
- }
- }
- } catch (e) {
- console.error('加载失败', e)
- } finally {
- this.loading = false
- }
- },
- handleSort(type) {
- this.sortType = type
- this.initCommentList()
- },
- showReplyInput(root, target) {
- this.activeRootId = root.id
- this.replyTarget = target
- this.replyPlaceholder = `回复 @${target.user.name} :`
- this.replyText = ''
- },
- postMainComment() {
- if (!this.newComment.trim()) return
- publishComment({
- videoId: this.videoId,
- content: this.newComment,
- parentId: 0,
- targetId: 0
- }).then(resp => {
- if (resp.code === 0) {
- this.$message.success('评论成功')
- this.newComment = ''
- this.initCommentList()
- }
- })
- },
- postReply() {
- if (!this.replyText.trim()) return
- publishComment({
- videoId: this.videoId,
- content: this.replyText,
- parentId: this.activeRootId,
- targetId: this.replyTarget.id,
- targetUsername: this.replyTarget.user.name
- }).then(resp => {
- if (resp.code === 0) {
- this.$message.success('回复成功')
- this.replyText = ''
- // 不再直接置 null,如果想让用户连续回复可以保持,但通常交互是关闭
- this.activeRootId = null
- if (this.dialogVisible) {
- this.fetchDialogReplies(this.currentParent.id, this.childCurrentPage)
- } else {
- // 如果在主列表回复,可能需要刷新子列表或重新获取
- this.initCommentList()
- }
- }
- })
- },
- async openReplyDialog(parentItem) {
- this.currentParent = parentItem
- this.childCurrentPage = 1
- this.dialogVisible = true
- this.activeRootId = null // 重置输入框状态
- this.fetchDialogReplies(parentItem.id, 1)
- },
- fetchDialogReplies(commentId, pageNumber) {
- getChildComment({ commentId, pn: pageNumber }).then(resp => {
- if (resp.code === 0) {
- this.dialogReplies = resp.data.list
- }
- })
- },
- handleChildPageChange(page) {
- this.fetchDialogReplies(this.currentParent.id, page)
- },
- formatDate(ts) {
- if (!ts) return ''
- const date = new Date(ts)
- return `${date.getFullYear()}-${date.getMonth() + 1}-${date.getDate()}`
- }
- }
- }
- </script>
- <style scoped lang="scss">
- .comment-card { margin-top: 20px; border: none; }
- .comment-container { color: #18191c; }
- .clickable-avatar {
- cursor: pointer;
- transition: opacity 0.2s;
- &:hover { opacity: 0.8; }
- }
- .clickable-link {
- cursor: pointer;
- &:hover { color: #00aeec !important; }
- }
- .comment-header {
- margin-bottom: 24px; display: flex; align-items: center;
- .total-count { font-size: 18px; font-weight: 500; margin-right: 20px; }
- .sort-options span { font-size: 14px; color: #9499a0; cursor: pointer; &.active { color: #18191c; font-weight: bold; } }
- }
- .main-reply-box {
- display: flex; gap: 15px; margin-bottom: 30px;
- .reply-input-wrapper { flex: 1; .post-btn { margin-top: 10px; float: right; } }
- }
- .comment-list { min-height: 200px; }
- .list-status { text-align: center; color: #9499a0; font-size: 13px; padding: 20px 0; }
- .comment-root {
- display: flex; gap: 15px;
- .comment-content {
- flex: 1;
- .user-name { font-size: 13px; font-weight: bold; color: #61666d; }
- .text { font-size: 15px; line-height: 1.6; margin: 8px 0; }
- .comment-footer {
- font-size: 13px; color: #9499a0; display: flex; gap: 20px; margin-bottom: 8px;
- .action { cursor: pointer; &:hover { color: #00aeec; } }
- }
- }
- }
- .reply-list {
- background: #f6f7f8; border-radius: 6px; padding: 12px;
- .reply-item {
- display: flex;
- gap: 8px;
- font-size: 14px; line-height: 22px; margin-bottom: 8px;
- .reply-avatar { flex-shrink: 0; margin-top: 2px; }
- .reply-right { flex: 1; }
- .user-name { color: #61666d; font-weight: bold; margin-right: 5px; }
- .target-name { color: #00aeec; font-weight: 500; margin: 0 4px; }
- .sub-action {
- margin-left: 10px; font-size: 12px; color: #9499a0; cursor: pointer;
- display: none;
- }
- &:hover .sub-action { display: inline-block; }
- }
- .view-more-bar {
- font-size: 13px; color: #9499a0; margin-top: 8px; padding-left: 32px;
- .click-more { color: #00aeec; cursor: pointer; font-weight: 500; }
- }
- }
- .inner-reply-box {
- margin-top: 10px; display: flex; gap: 8px;
- background: #fff; padding: 10px; border: 1px solid #e3e5e7; border-radius: 6px;
- }
- .dialog-scroll-area {
- max-height: 50vh; overflow-y: auto; padding-right: 8px;
- .parent-info {
- display: flex; gap: 12px;
- .parent-text {
- flex: 1;
- .user-name { font-weight: bold; color: #61666d; margin-bottom: 4px; }
- .content { font-size: 15px; line-height: 1.6; margin-bottom: 8px; }
- .reply-footer { font-size: 12px; color: #9499a0; .action { margin-left: 15px; cursor: pointer; &:hover { color: #00aeec; } } }
- }
- }
- .dialog-reply-item {
- display: flex; gap: 10px; margin-bottom: 15px;
- .reply-body {
- flex: 1; font-size: 14px;
- .user-name { font-weight: bold; color: #61666d; }
- .target-name { color: #00aeec; margin: 0 4px; }
- .reply-footer { font-size: 12px; color: #9499a0; margin-top: 5px;
- .action { margin-left: 10px; cursor: pointer; &:hover { color: #00aeec; } }
- }
- }
- }
- .dialog-pagination { display: flex; justify-content: center; margin-top: 20px; }
- }
- /* 弹窗专用回复框样式 */
- .dialog-inner-reply {
- padding-top: 0;
- .el-divider--horizontal { margin: 12px 0; }
- }
- </style>
|