From 1c0627671cd95695df604a54de0bbb76469c21e9 Mon Sep 17 00:00:00 2001
From: xiangpei <xiangpei@timesnew.cn>
Date: 星期四, 29 五月 2025 10:23:37 +0800
Subject: [PATCH] 真机上视频加载和渲染是异步的,可能导致 onPlay/onPause 事件触发延迟。解决暂停图标和播放不一致问题

---
 pages/tabbar/index/home.vue |  591 +++++++++++++++++++++++++++++++++++++++++++++++-----------
 1 files changed, 478 insertions(+), 113 deletions(-)

diff --git a/pages/tabbar/index/home.vue b/pages/tabbar/index/home.vue
index bbbe33b..4b7f2e1 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,268 @@
 		   <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">&#xe629;</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">&#xe605;</text>
 			<text class="iconfont" v-else>&#xe601;</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">&#xe7f7;</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">璇勮({{commentsTotal}})</text>
+	      <text class="iconfont close-icon" @click="closeCommentPopup">&#xe675;</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 in comments" :key="comment.id">
+	        <image class="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>
+	          <text class="time">{{formatTime(comment.createTime)}}</text>
+	        </view>
+	      </view>
+	    </scroll-view>
+	    
+	    <view class="comment-input-area">
+	      <input 
+	        class="comment-input" 
+	        v-model="commentForm.commentContent" 
+	        placeholder="鍐欎笅浣犵殑璇勮..." 
+	        placeholder-class="placeholder"
+	      />
+	      <button class="submit-btn" @click="submitComment">鍙戦��</button>
+	    </view>
+	  </view>
+	</uni-popup>
+	
+	
 	<custom-tabbar bgColor="#333333" selected="index" selectedTextColor="#ffffff"></custom-tabbar>
   </view>
 </template>
 
 <script>
+import { getRecommendVideos, savePlayRecord, subscribe, getVideoComments, addVideoComment } 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: ''
+		},
+		commentForm: { // 璇勮琛ㄥ崟鏁版嵁
+			id: null,
+			videoId: null,
+			commentContent: '',
+			replyId: 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: {
+		// 鏍煎紡鍖栨椂闂�
+	    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.commentForm = {
+				  			  id: null,
+				  			  videoId: null,
+				  			  commentContent: '',
+				  			  replyId: null
+				  }
+				  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.commentForm = {
+			  id: null,
+			  videoId: null,
+			  commentContent: '',
+			  replyId: null
+		  }
+		  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.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 +346,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 +502,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 {
@@ -400,5 +660,110 @@
 	  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;
+	  padding: 10rpx 0;
+	}
+
+	.avatar {
+	  width: 80rpx;
+	  height: 80rpx;
+	  border-radius: 50%;
+	  margin-right: 20rpx;
+	}
+
+	.comment-content {
+	  flex: 1;
+	}
+
+	.nickname {
+	  font-size: 24rpx;
+	  color: #666;
+	  display: block;
+	  margin-bottom: 10rpx;
+	}
+
+	.content {
+	  font-size: 24rpx;
+	  color: #333;
+	  display: block;
+	  margin-bottom: 10rpx;
+	}
+
+	.time {
+	  font-size: 24rpx;
+	  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;
+	}
 </style>
\ No newline at end of file

--
Gitblit v1.8.0