From e15dbdbc396f61a645c8d8a504b45476f1fcea08 Mon Sep 17 00:00:00 2001 From: xiangpei <xiangpei@timesnew.cn> Date: 星期五, 30 五月 2025 17:16:17 +0800 Subject: [PATCH] 评论点赞功能 --- pages/tabbar/index/home.vue | 824 ++++++++++++++++++++++++++++++++++++++++++++++++++-------- 1 files changed, 709 insertions(+), 115 deletions(-) diff --git a/pages/tabbar/index/home.vue b/pages/tabbar/index/home.vue index bbbe33b..d3b82cc 100644 --- a/pages/tabbar/index/home.vue +++ b/pages/tabbar/index/home.vue @@ -9,17 +9,28 @@ @change="onSwiperChange" > <swiper-item v-for="(item, index) in videoList" :key="item.id"> + <!-- 鎾斁鎸夐挳锛堜粎褰撹棰戞殏鍋滄椂鏄剧ず锛� --> + <view + class="play-icon" + @click="togglePlay(index)" + v-if="!currentVideoIsPlaying" + > + <image src="/static/video/play.png" style="width: 45px;height: 45px" mode="aspectFit"></image> + </view> <video :id="'video'+index" - :src="item.url" + :ref="'video'+index" + :src="item.videoUrl" :autoplay="currentIndex === index" :controls="false" :loop="true" :object-fit="item.objectFit" class="video-item" - @play="onPlay(index)" + @play="onPlay(item.id, index)" @pause="onPause(index)" @ended="onEnded(index)" + @click="togglePlay(index)" + @timeupdate="onTimeUpdate($event)" ></video> <!-- 鎮寕鍟嗗搧閾炬帴灞� --> @@ -27,7 +38,7 @@ <view class="goods-link"> <view class="goods-container"> <!-- 鍟嗗搧鍥剧墖 --> - <image class="goods-image" :src="item.goods.image" mode="aspectFill"></image> + <image class="goods-image" :src="item.goods.imageUrl" mode="aspectFill"></image> <!-- 鍟嗗搧淇℃伅 --> <view class="goods-info"> @@ -36,7 +47,7 @@ <text class="current-price">楼{{item.goods.price}}</text> <text class="original-price" v-if="item.goods.originalPrice">楼{{item.goods.originalPrice}}</text> </view> - <text class="sales-count">{{item.goods.sales}}浜哄凡璐�</text> + <text class="sales-count">{{item.goods.saleNum}}浜哄凡璐�</text> </view> <!-- 璐拱鎸夐挳 --> @@ -51,11 +62,11 @@ <!-- 瑙嗛淇℃伅灞� --> <view class="video-info"> <view> - <text class="video-author">@{{item.author}}</text> + <text class="video-author">@{{item.authorName}}</text> </view> <view style="width: 100%;word-wrap: break-word;white-space: normal;overflow-wrap: break-word;"> <text class="video-title">{{item.title}}</text> - <text class="video-tag" v-for="(tag, index) in item.tags" :key="tag">#{{tag}}</text> + <text class="video-tag" v-for="(tag, index) in item.tagList" :key="tag">#{{tag.tagName}}</text> </view> </view> @@ -64,91 +75,412 @@ <view class="avatar-container"> <image class="avatar" :src="item.authorAvatar" mode="aspectFill"></image> <!-- 鍏虫敞鍥炬爣 - 浣跨敤缁濆瀹氫綅 --> - <view class="follow-icon"> + <view v-if="!item.subscribeThisAuthor" class="follow-icon" @click="subscribeAuth(index, item.authorId)"> <text class="iconfont"></text> </view> </view> - <view class="action-item" @click="toggleCollect(item)"> - <!-- <image :src="item.isCollected ? '/static/collected.png' : '/static/collect.png'"></image> --> + <view class="action-item" @click="toggleCollect(item, index)"> <text class="iconfont" v-if="item.collected"></text> <text class="iconfont" v-else></text> - <text style="font-size: 10px;font-weight: lighter;">{{item.collectCount}}</text> + <text style="font-size: 10px;font-weight: lighter;">{{item.collectNum}}</text> </view> <view class="action-item" @click="showComments(item)"> <text class="iconfont"></text> - <text style="font-size: 10px;font-weight: lighter;">{{item.commentCount}}</text> + <text style="font-size: 10px;font-weight: lighter;">{{item.commentNum}}</text> </view> </view> + </swiper-item> </swiper> + + <!-- 璇勮寮圭獥 --> + <uni-popup ref="commentPopup" type="bottom" :is-mask-click="true" @maskClick="closeCommentPopup"> + <view class="comment-popup"> + <view class="popup-header"> + <text class="popup-title" v-if="!commentForm.replyId">璇勮({{commentsTotal}})</text> + <view class="reply-title" v-else> + <text>鍥炲 @{{commentForm.replyUserNickname}}</text> + <text class="cancel-reply" @click="cancelReply">鍙栨秷</text> + </view> + <text class="iconfont close-icon" @click="closeCommentPopup"></text> + </view> + + <scroll-view class="comment-list" scroll-y :show-scrollbar="false" @scrolltolower="getCommentPage"> + <view v-if="commentLoading" class="loading"> + <uni-load-more status="loading"></uni-load-more> + </view> + + <view v-else-if="comments.length === 0" class="empty"> + 鏆傛棤璇勮锛屽揩鏉ュ彂琛ㄧ涓�鏉¤瘎璁哄惂~ + </view> + + <view v-else class="comment-item" v-for="(comment, index) in comments" :key="comment.id"> + <view style="display: flex;"> + <image class="comment-avatar" :src="comment.userAvatar || '/static/default-avatar.png'"></image> + <view class="comment-content"> + <text class="nickname">{{comment.userNickname}}</text> + <text class="content">{{comment.commentContent}}</text> + <view style="position: relative;"> + <text class="time">{{formatTime(comment.createTime)}}</text> + <text @click="openReply(comment)" class="reply-btu time">鍥炲</text> + <text v-if="!comment.hasThumbsUp" class="thumbs-up time iconfont" @click="thubmsUp(comment.id, index, null)"><text v-show="comment.thumbsUpNum > 0" class="thumbs-num">{{comment.thumbsUpNum}}</text></text> + <text v-else class="thumbs-up time iconfont" @click="cancelThumbsUp(comment.id, index, null)"><text v-show="comment.thumbsUpNum > 0" class="thumbs-num">{{comment.thumbsUpNum}}</text></text> + </view> + </view> + </view> + <!-- 鍥炲鍒楄〃 --> + <view class="reply-list" v-if="comment.replies && comment.replies.length > 0"> + <view class="reply-item" v-for="(reply, replyIndex) in comment.replies" :key="reply.id"> + <view class="reply-content"> + <view style="display: flex;"> + <image class="comment-reply-avatar" :src="reply.replyUserAvatar || '/static/default-avatar.png'"></image> + <text class="nickname">{{reply.userNickname}}</text> + <text v-if="reply.replyUserId && reply.masterCommentId !== reply.replyId" class="reply-to"><text style="margin-right: 10rpx;font-size: 28rpx;" class="iconfont"></text>{{reply.replyUserNickname}}</text> + </view> + <text class="content">{{reply.commentContent}}</text> + <view class="reply-footer"> + <text class="time">{{formatTime(reply.createTime)}}</text> + <text @click="openReply(comment, reply)" class="reply-btu time">鍥炲</text> + <text v-if="!reply.hasThumbsUp" class="thumbs-up time iconfont" @click="thubmsUp(reply.id, index, replyIndex)"><text v-show="reply.thumbsUpNum > 0" class="thumbs-num">{{reply.thumbsUpNum}}</text></text> + <text v-else class="thumbs-up time iconfont" @click="cancelThumbsUp(reply.id, index, replyIndex)"><text v-show="reply.thumbsUpNum > 0" class="thumbs-num">{{reply.thumbsUpNum}}</text></text> + </view> + </view> + </view> + </view> + <view class="view-more-replies" v-if="comment.replyTotalCount > 0 && !comment.expandReply" @click="loadRepliesPage(comment, index)"> + <text class="line">鈥斺��</text>灞曞紑{{comment.replyTotalCount}}鏉″洖澶� 鈫� + </view> + <view class="reply-op" v-if="comment.replyTotalCount > replyCommentQuery.pageNumber * replyCommentQuery.pageSize && comment.expandReply"> + <view @click="loadNextPageReply(index)" class="reply-op-item"><text class="line">鈥斺��</text>灞曞紑鏇村<text class="iconfont textSideIcon"></text></view> + <view @click="retractReplyComment(index)" class="reply-op-item" style="margin-left: 50rpx;">鏀惰捣<text class="iconfont textSideIcon"></text></view> + </view> + <view class="reply-op" v-else-if="comment.replyTotalCount <= replyCommentQuery.pageNumber * replyCommentQuery.pageSize && comment.expandReply"> + <view @click="retractReplyComment(index)" class="reply-op-item"><text class="line">鈥斺��</text>鏀惰捣<text class="iconfont textSideIcon"></text></view> + </view> + </view> + </scroll-view> + <view class="comment-input-area"> + <input + ref="commentInput" + class="comment-input" + v-model="commentForm.commentContent" + :placeholder="commentForm.replyId ? `鍥炲 @${commentForm.replyUserNickname}` : '鍐欎笅浣犵殑璇勮...'" + placeholder-class="placeholder" + /> + <button class="submit-btn" @click="submitComment" :disabled="!commentForm.commentContent.trim()">鍙戦��</button> + </view> + </view> + </uni-popup> + + <custom-tabbar bgColor="#333333" selected="index" selectedTextColor="#ffffff"></custom-tabbar> </view> </template> <script> +import { getRecommendVideos, savePlayRecord, subscribe, getVideoComments, addVideoComment, thubmsUpComment, cancelThubmsUpComment } from "@/api/video.js"; +import { changeCollect } from "@/api/collect.js"; export default { data() { return { - isFullScreen: false, - windowHeight: 0, - currentIndex: 0, // 褰撳墠鎾斁鐨勮棰戠储寮� - videoList: [ - { - url: 'http://vjs.zencdn.net/v/oceans.mp4', - objectFit: 'contain', - title: '鎴戜簡涓�', - author: 'xp', - authorAvatar: 'https://picsum.photos/200/200?random=2', - collected: true, - commentCount: 12, - collectCount: 45, - tags: ["浜斾竴", "鐖辩編椋�", "澹ぇ澶�熷害鍜岀矇绾㈣壊鐨勬仮澶嶉�熷害鐨勫彛琚嬬┖绌�"], - goods: { - name: '鎺ㄦ祦', - price: '10', - originalPrice: '48.9', - sales: 1988, - image: 'https://picsum.photos/200/200?random=2' - } - }, - { - url: 'https://videos.pexels.com/video-files/30900524/13210612_1080_1920_30fps.mp4', - objectFit: 'cover', - title: '鎴戜簡涓�', - author: 'xp', - authorAvatar: 'https://picsum.photos/200/200?random=2', - collected: false, - commentCount: 6, - collectCount: 45, - tags: ["鎴戝枩娆�"], - goods: { - name: '鎺ㄦ祦', - price: '10', - originalPrice: '48.9', - sales: 1988, - image: 'https://picsum.photos/200/200?random=2' - } - }, - ], // 瑙嗛鍒楄〃鏁版嵁 - videoContexts: [], // 瑙嗛涓婁笅鏂囧璞¢泦鍚� - loading: false, // 鏄惁姝e湪鍔犺浇 - page: 1, // 褰撳墠椤电爜 - pageSize: 10 // 姣忛〉鏁伴噺 + commentNoMore: false, // 鏄惁杩樻湁鏇村璇勮 + commentQuery: { + pageNumber: 1, + pageSize: 5, + videoId: '', + masterCommentId: '' + }, + replyCommentQuery: { + pageNumber: 1, + pageSize: 5, + videoId: '', + masterCommentId: '' + }, + commentForm: { // 璇勮琛ㄥ崟鏁版嵁 + id: '', + videoId: '', + commentContent: '', + replyId: '', + replyUserId: '', + replyUserNickname: '', + replyUserAvatar: '', + masterCommentId: null + }, + comments: [], // 璇勮鍒楄〃 + commentsTotal: 0, // 璇勮鎬绘潯鏁� + commentLoading: false, // 璇勮鍔犺浇鐘舵�� + startHidenTime: 0, // 璁板綍鍒囨崲鑷冲叾瀹冮〉闈㈢殑鏃堕棿锛岀敤浜庤绠楄棰戣鐪嬫椂闂村噺鍘荤殑閮ㄥ垎 + totalHidenTime: 0, // 鎬诲叡闅愯棌椤甸潰鐨勬椂闂� + startPauseTime: 0, // 寮�濮嬫殏鍋滅殑鏃堕棿 + totalPauseTime: 0, // 鎬诲叡鏆傚仠鐨勬椂闂� + playRecord: { + videoId: null, + viewDuration: 0, // 杩欎釜瑙嗛鎬诲叡瑙傜湅浜嗗涔� + playAt: 0 ,// 杩欎釜瑙嗛鎾斁鍒板摢浜� + startPlayTime: 0 // 杩欎釜瑙嗛浠庝粈涔堟椂鍊欏紑濮嬫挱鏀剧殑 + }, + currentVideoIsPlaying: true, // 褰撳墠瑙嗛鏄惁姝e湪鎾斁 + isFullScreen: false, + windowHeight: 0, + currentIndex: 0, // 褰撳墠鎾斁鐨勮棰戠储寮� + videoList: [ + + ], // 瑙嗛鍒楄〃鏁版嵁 + videoContexts: [], // 瑙嗛涓婁笅鏂囧璞¢泦鍚� + loading: false, // 鏄惁姝e湪鍔犺浇 + page: 1, // 褰撳墠椤电爜 + pageSize: 10 // 姣忛〉鏁伴噺 } }, + onShow() { + this.loadVideos() + // 濡傛灉瑙嗛鎸変笅鏆傚仠鍚庡垏鎹㈤〉闈㈠啀鍥炲埌椤甸潰鏃讹紝鍙畻鏆傚仠鏃堕棿锛堝洜涓烘殏鍋滄椂闂村拰绂诲紑椤甸潰鏃堕棿鏄噸澶嶇殑锛屽彧绠椾竴涓級 + if(this.startHidenTime !== 0 && this.currentVideoIsPlaying) { + const duration = Date.now() - this.startHidenTime + this.totalHidenTime += duration + } + }, + onHide() { + this.startHidenTime = Date.now() + }, onLoad() { - // this.loadVideos(); + this.loadVideos(); }, onReady() { // 鍒濆鍖栬棰戜笂涓嬫枃 this.initVideoContexts(); }, methods: { + // 鍙栨秷鐐硅禐 + async cancelThumbsUp(id, commentIndex, replyIndex) { + const data = { + refId: id, + thumbsUpType: 'video_comment' + } + cancelThubmsUpComment(data).then(res => { + if(replyIndex != null) { + this.comments[commentIndex].replies[replyIndex].hasThumbsUp = false; + this.comments[commentIndex].replies[replyIndex].thumbsUpNum -= 1; + } else { + this.comments[commentIndex].hasThumbsUp = false; + this.comments[commentIndex].thumbsUpNum -= 1; + } + }) + }, + // 璇勮鐐硅禐 + async thubmsUp(id, commentIndex, replyIndex) { + const data = { + refId: id, + thumbsUpType: 'video_comment' + } + thubmsUpComment(data).then(res => { + if(replyIndex != null) { + this.comments[commentIndex].replies[replyIndex].hasThumbsUp = true; + this.comments[commentIndex].replies[replyIndex].thumbsUpNum += 1; + } else { + this.comments[commentIndex].hasThumbsUp = true; + this.comments[commentIndex].thumbsUpNum += 1; + } + }) + }, + // 鍔犺浇涓嬩竴椤靛洖澶� + loadNextPageReply(index) { + this.replyCommentQuery.pageNumber++; + getVideoComments(this.replyCommentQuery).then(res => { + this.comments[index].replies = [ + ...this.comments[index].replies, + ...res.data.data.filter( + (newItem) => !this.comments[index].replies.some((oldItem) => oldItem.id === newItem.id) + ), + ]; + }) + }, + // 鏀惰捣鍥炲 + retractReplyComment(index) { + this.comments[index].expandReply = false; + this.comments[index].replies = []; + }, + // 鍔犺浇鍥炲 + loadRepliesPage(comment, index) { + this.replyCommentQuery.pageNumber = 1; + this.replyCommentQuery.masterCommentId = comment.id + getVideoComments(this.replyCommentQuery).then(res => { + this.comments[index].replies = res.data.data; + this.comments[index].expandReply = true; + }) + }, + resetCommentForm() { + const videoId = this.commentForm.videoId; + this.commentForm = { // 璇勮琛ㄥ崟鏁版嵁 + id: '', + videoId: '', + commentContent: '', + replyId: '', + replyUserId: '', + replyUserNickname: '', + replyUserAvatar: '', + masterCommentId: null + } + }, + // 鍙栨秷鍥炲 + cancelReply() { + this.resetCommentForm() + }, + // 鎵撳紑鍥炲妗� + openReply(comment, reply = null) { + if(reply) { + comment = reply + } + this.commentForm.masterCommentId = comment.masterCommentId ? comment.masterCommentId : comment.id; + this.commentForm.replyId = comment.id; + this.commentForm.replyUserId = comment.userId; + this.commentForm.replyUserNickname = comment.userNickname; + this.commentForm.replyUserAvatar = comment.userAvatar; + // 鑷姩鑱氱劍杈撳叆妗� + this.$nextTick(() => { + const input = this.$refs.commentInput; + if (input) input.focus(); + }); + }, + // 鏍煎紡鍖栨椂闂� + formatTime(time) { + const date = new Date(time); + const now = new Date(); + const diff = Math.floor((now - date) / 1000); // 绉� + + if (diff < 60) return '鍒氬垰'; + if (diff < 3600) return `${Math.floor(diff / 60)}鍒嗛挓鍓峘; + if (diff < 86400) return `${Math.floor(diff / 3600)}灏忔椂鍓峘; + + return `${date.getMonth() + 1}鏈�${date.getDate()}鏃; + }, + // 鎻愪氦璇勮 + async submitComment() { + if (!this.commentForm.commentContent.trim()) { + uni.showToast({ + title: '璇勮鍐呭涓嶈兘涓虹┖', + icon: 'none' + }); + return; + } + // 鍙戣〃璇勮 + addVideoComment(this.commentForm).then(res => { + if(res.data.code === 200) { + this.resetCommentForm() + + // 濡傛灉鏄瘎璁哄埆浜虹殑鍥炲锛岄偅涔堝氨灏嗚繖涓彂甯冨埌replies閲岄潰 + if(res.data.data.replyId) { + for (const [index, item] of this.comments.entries()) { + if (item.id === res.data.data.replyId) { + item.replies.unshift(res.data.data); + // this.loadRepliesPage(item, index) + break; // 璺冲嚭寰幆 + } + } + } else { + this.comments.unshift(res.data.data); + } + console.log("鏂板鍚�",this.comments); + uni.showToast({ + title: '璇勮鎴愬姛' + }); + // 褰撳墠瑙嗛璇勮鏁板姞涓� + this.commentsTotal += 1; + this.videoList[this.currentIndex].commentNum += 1; + } else { + uni.showToast({ + title: res.data.msg, + icon: 'none' + }); + } + }).catch(() => { + uni.showToast({ + title: '璇勮澶辫触', + icon: 'none' + }); + }) + }, + // 鍏抽棴璇勮寮圭獥 + closeCommentPopup() { + this.$refs.commentPopup.close() + this.showCommentPopup = false; + this.comments = []; + this.resetCommentForm() + this.commentQuery.pageNumber = 1; + this.commentNoMore = false; + }, + // 涓嬫粦璇勮鍖哄姞杞借瘎璁� + async getCommentPage() { + if(this.commentNoMore) { + return; + } + getVideoComments(this.commentQuery).then(res => { + if(this.commentQuery.pageNumber === 1) { + this.comments = res.data.data + } else { + this.comments = [ + ...this.comments, + ...res.data.data.filter( + (newItem) => !this.comments.some((oldItem) => oldItem.id === newItem.id) + ), + ]; + } + if (res.data.data.length < this.commentQuery.pageSize) { + this.commentNoMore = true; + return; + } + this.commentQuery.pageNumber++; + }) + }, + // 鏄剧ず璇勮寮圭獥 + async showComments(item) { + this.commentForm.videoId = item.id; + this.$refs.commentPopup.open(); + this.commentLoading = true; + this.commentQuery.videoId = item.id + this.replyCommentQuery.videoId = item.id + // 棣栨鍔犺浇璇勮鍒嗛〉澶у皬澧炲姞涓�鍊嶏紝浠ヤ骇鐢熸粴鍔ㄦ潯锛屽悗缁彲瑙﹀彂 + this.commentQuery.pageSize *= 2; + getVideoComments(this.commentQuery).then(res => { + this.commentsTotal = res.data.total; + this.comments = res.data.data; + this.commentQuery.pageNumber += 2; + this.commentQuery.pageSize /= 2; + }).catch(() => { + uni.showToast({ + title: '鑾峰彇璇勮澶辫触', + icon: 'none' + }); + }).finally(() => { + this.commentLoading = false; + }) + }, + // 鍏虫敞浣滆�� + subscribeAuth(index, authorId) { + this.videoList.forEach(video => { + if(video.authorId === authorId) { + video.subscribeThisAuthor = true + } + }) + subscribe(authorId).then(res => { + if(res.data.code === 200) { + uni.showToast({ + title: '鍏虫敞鎴愬姛~', + icon: 'none' + }); + } else { + this.videoList.forEach(video => { + if(video.authorId === authorId) { + video.subscribeThisAuthor = false + } + }) + } + }) + }, // 鍒濆鍖栬棰戜笂涓嬫枃 initVideoContexts() { this.videoContexts = this.videoList.map((_, index) => { let videoContent = uni.createVideoContext(`video${index}`, this); - // videoContent.requestFullScreen({ direction: 0 }); return videoContent; }); }, @@ -158,77 +490,139 @@ if (this.loading) return; this.loading = true; - try { - const res = await uni.request({ - url: 'https://your-api.com/videos', - data: { - page: this.page, - pageSize: this.pageSize - } - }); - - if (this.page === 1) { - this.videoList = res.data.list; - } else { - this.videoList = [...this.videoList, ...res.data.list]; - } - - this.page++; - this.$nextTick(() => { - this.initVideoContexts(); - }); - } catch (e) { - console.error('鍔犺浇瑙嗛澶辫触', e); - } finally { - this.loading = false; - } + getRecommendVideos({pageNumber: this.page, pageSize: this.pageSize}).then(res => { + console.log(res, "瑙嗛鏁版嵁"); + if (this.page === 1) { + this.videoList = res.data.data; + } else { + this.videoList = [...this.videoList, ...res.data.data]; + } + + this.page++; + this.$nextTick(() => { + this.initVideoContexts(); + }); + this.loading = false; + }) }, // 婊戝姩鍒囨崲瑙嗛 onSwiperChange(e) { - const oldIndex = this.currentIndex; - this.currentIndex = e.detail.current; - - // 鏆傚仠涓婁竴涓棰� - if (this.videoContexts[oldIndex]) { - this.videoContexts[oldIndex].pause(); - } - - // 鎾斁褰撳墠瑙嗛 - if (this.videoContexts[this.currentIndex]) { - this.videoContexts[this.currentIndex].play(); - } + // 濡傛灉瑙嗛澶勪簬鏆傚仠鐘舵�佸線涓嬪埛瑙嗛锛岄偅涔堥渶瑕佸啀璁$畻涓�娆℃殏鍋滄椂闂� + if(!this.currentVideoIsPlaying) { + if(this.startPauseTime !== 0) { + const duration = Date.now() - this.startPauseTime + this.totalPauseTime += duration + } + } + // 淇濆瓨涓婁竴涓棰戠殑鎾斁璁板綍 + this.savePlayRecord() + const oldIndex = this.currentIndex; + this.currentIndex = e.detail.current; + + // 鏆傚仠涓婁竴涓棰� + if (this.videoContexts[oldIndex]) { + this.videoContexts[oldIndex].pause(); + } + + this.startPauseTime = 0; + // 鎾斁褰撳墠瑙嗛 + if (this.videoContexts[this.currentIndex]) { + this.videoContexts[this.currentIndex].play(); + } }, - // 鐐硅禐/鍙栨秷鐐硅禐 - toggleLike(item) { - item.isLiked = !item.isLiked; - item.likeCount += item.isLiked ? 1 : -1; - - uni.request({ - url: `https://your-api.com/video/${item.id}/like`, - method: item.isLiked ? 'POST' : 'DELETE' - }); + // 鏀惰棌/鍙栨秷鏀惰棌 + toggleCollect(item, index) { + let data = { + refId: item.id, + collectType: 'video' + } + const beforeCollected = item.collected + const beforeCollectNum = item.collectNum + if(item.collected) { + this.videoList[index].collected = false + this.videoList[index].collectNum -= 1 + } else { + this.videoList[index].collected = true + this.videoList[index].collectNum += 1 + } + changeCollect(data).then(res => { + if(res.data.code !== 200) { + this.videoList[index].collected = beforeCollected + this.videoList[index].collectNum = beforeCollectNum + } + }) }, - + // 鍗曞嚮灞忓箷锛氭殏鍋滄垨缁х画鎾斁 + togglePlay(index) { + if(this.currentVideoIsPlaying) { + this.videoContexts[index].pause(); + } else { + this.videoContexts[index].play(); + } + }, // 瑙嗛鎾斁浜嬩欢 - onPlay(index) { - console.log(`瑙嗛 ${index} 寮�濮嬫挱鏀綻); + onPlay(id, index) { + console.log(id, index, "瑙﹀彂鎾斁"); + if(index === this.currentIndex) { + this.currentVideoIsPlaying = true; + } else { + this.currentVideoIsPlaying = false; + return + } + this.playRecord.videoId = id; + // 娌″垵濮嬪寲鎵嶈祴鍊硷紝鍥犱负涓�涓棰戦噸澶嶆挱鏀緊nPlay浼氶噸澶嶈Е鍙� + if(this.playRecord.startPlayTime === 0) { + this.playRecord.startPlayTime = Date.now(); + } + if(this.startPauseTime !== 0) { + const duration = Date.now() - this.startPauseTime + this.totalPauseTime += duration + } }, // 瑙嗛鏆傚仠浜嬩欢 onPause(index) { - console.log(`瑙嗛 ${index} 鏆傚仠`); + console.log(index, "瑙﹀彂鏆傚仠"); + if(index === this.currentIndex) { + this.currentVideoIsPlaying = false; + } else { + this.currentVideoIsPlaying = true; + return + } + this.startPauseTime = Date.now() }, // 瑙嗛缁撴潫浜嬩欢 onEnded(index) { - console.log(`瑙嗛 ${index} 鎾斁缁撴潫`); - // 鑷姩鎾斁涓嬩竴涓紙濡傛灉涓嶅湪鏈�鍚庝竴涓級 - if (index < this.videoList.length - 1) { - this.currentIndex = index + 1; - } - } + // this.currentVideoIsPlaying = false; + }, + + // 璁板綍鎾斁鏃堕暱 + onTimeUpdate(e) { + this.playRecord.playAt = e.detail.currentTime + }, + + // 淇濆瓨鎾斁璁板綍 + async savePlayRecord() { + console.log(Date.now(), this.playRecord.startPlayTime, this.totalHidenTime); + + const data = { + videoId: this.playRecord.videoId, + viewDuration: Date.now() - this.playRecord.startPlayTime - this.totalHidenTime - this.totalPauseTime, + playAt: this.playRecord.playAt + } + this.playRecord = { + videoId: null, + viewDuration: 0, // 杩欎釜瑙嗛鎬诲叡瑙傜湅浜嗗涔� + playAt: 0 ,// 杩欎釜瑙嗛鎾斁鍒板摢浜� + startPlayTime: 0 // 杩欎釜瑙嗛浠庝粈涔堟椂鍊欏紑濮嬫挱鏀剧殑 + } + this.totalHidenTime = 0 + this.totalPauseTime = 0 + savePlayRecord(data) + } } } </script> @@ -252,6 +646,16 @@ width: 100%; height: 100%; object-fit: cover; + } + .play-icon { + position: absolute; + top: 50%; + left: 50%; + transform: translate(-50%, -50%); + width: 45px; + height: 45px; + z-index: 10; + opacity: 0.6; } .video-info { @@ -382,7 +786,7 @@ } .original-price { - font-size: 24rpx; + font-size: 28rpx; color: #999; text-decoration: line-through; } @@ -395,10 +799,200 @@ .buy-button { background: linear-gradient(to right, #ff5a5f, #ff2e4d); color: white; - padding: 10rpx 24rpx; + padding: 10rpx 28rpx; border-radius: 20rpx; font-size: 26rpx; font-weight: bold; } + /* 璇勮寮圭獥鏍峰紡 */ + .comment-popup { + background-color: #fff; + border-radius: 20rpx 20rpx 0 0; + padding-bottom: env(safe-area-inset-bottom); + height: 60vh; + display: flex; + flex-direction: column; + } + .popup-header { + padding: 30rpx; + display: flex; + justify-content: space-between; + align-items: center; + border-bottom: 1rpx solid #f5f5f5; + } + + .popup-title { + font-size: 32rpx; + font-weight: bold; + } + + .close-icon { + /* font-size: 36rpx; */ + color: #999; + } + + .comment-list { + flex: 1; + padding: 0rpx 20rpx 20rpx 20rpx; + box-sizing: border-box; + height: calc(60vh - 260rpx); + } + + .comment-item { + display: flex; + flex-direction: column; + padding: 10rpx 0 20rpx 0; + } + + .comment-avatar { + width: 70rpx; + height: 70rpx; + border-radius: 50%; + margin-right: 10rpx; + } + .comment-reply-avatar { + width: 40rpx; + height: 40rpx; + border-radius: 50%; + margin-right: 10rpx; + } + + .comment-content { + flex: 1; + } + + .nickname { + font-size: 28rpx; + color: #666; + display: block; + margin-bottom: 10rpx; + } + + .content { + font-size: 28rpx; + color: #333; + display: block; + margin-bottom: 10rpx; + } + + .time { + font-size: 28rpx; + color: #999; + } + + .comment-input-area { + display: flex; + padding: 20rpx 30rpx; + align-items: center; + } + + .comment-input { + flex: 1; + background-color: #fff; + height: 80rpx; + border: 1px solid #dcdcdc; + border-radius: 40rpx; + padding: 0 30rpx; + font-size: 28rpx; + } + + .placeholder { + color: #ccc; + } + + .submit-btn { + margin-left: 20rpx; + background-color: #07c160; + color: #fff; + border-radius: 40rpx; + padding: 0 30rpx; + height: 80rpx; + line-height: 80rpx; + font-size: 28rpx; + } + + .loading, .empty { + padding: 40rpx 0; + text-align: center; + color: #999; + } + .reply-list { + margin-top: 20rpx; + padding-left: 80rpx; + } + .reply-op { + margin-top: 10rpx; + padding-left: 80rpx; + display: flex; + font-size: 28rpx; + color: #333; + } + .reply-op-item { + display: flex; + align-items: center; + height: 40rpx; + } + + .reply-item { + display: flex; + margin-bottom: 20rpx; + } + + .reply-content { + flex: 1; + } + + .reply-to { + color: #576b95; + margin: 0 10rpx; + font-size: 28rpx; + } + .reply-title { + display: flex; + align-items: center; + font-size: 28rpx; + color: #333; + } + + .cancel-reply { + margin-left: 20rpx; + color: #576b95; + font-size: 28rpx; + padding: 6rpx 12rpx; + background: #f5f5f5; + border-radius: 20rpx; + } + .view-more-replies { + color: #576b95; + font-size: 28rpx; + padding: 10rpx 0; + padding-left: 80rpx; + } + .comment-footer, .reply-footer { + display: flex; + align-items: center; + font-size: 28rpx; + color: #999; + } + .reply-btu { + margin-left: 30rpx; + } + .thumbs-up { + position: absolute; + right: 20rpx; + font-size: 32rpx; + width: 120rpx; + } + .textSideIcon { + font-size: 36rpx; + margin-left: 5rpx; + } + .line { + margin-right: 10rpx; + color: #cccccc; + } + .thumbs-num { + margin-left: 4rpx; + } </style> \ No newline at end of file -- Gitblit v1.8.0