You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
			
				
					632 lines
				
				14 KiB
			
		
		
			
		
	
	
					632 lines
				
				14 KiB
			| 
											2 months ago
										 | <template> | ||
|  |   <view class="note-detail-container"> | ||
|  |     <!-- 笔记内容区域 --> | ||
|  |     <view class="content-scroll" > | ||
|  |       <!-- 作者信息 --> | ||
|  |       <view class="author-section"> | ||
|  |         <image | ||
|  |           class="author-avatar" | ||
|  |           :src="noteDetail.user.avatar" | ||
|  |           mode="aspectFill" | ||
|  |         /> | ||
|  |         <view class="author-info"> | ||
|  |           <text class="author-name">{{ noteDetail.user.name }}</text> | ||
|  |           <text class="publish-time">{{ | ||
|  |             formatTime(noteDetail.createTime) | ||
|  |           }}</text> | ||
|  |         </view> | ||
|  |         <button | ||
|  |           class="follow-btn" | ||
|  |           :class="{ followed: noteDetail.user.isFollowed }" | ||
|  |           @click="toggleFollow" | ||
|  |         > | ||
|  |           {{ noteDetail.user.isFollowed ? "已关注" : "+ 关注" }} | ||
|  |         </button> | ||
|  |       </view> | ||
|  | 
 | ||
|  |       <!-- 笔记标题 --> | ||
|  |       <view class="note-title"> | ||
|  |         {{ noteDetail.title }} | ||
|  |       </view> | ||
|  | 
 | ||
|  |       <!-- 笔记主图 --> | ||
|  |       <view class="note-image-container" v-if="noteDetail.image"> | ||
|  |         <image | ||
|  |           class="note-image" | ||
|  |           :src="noteDetail.image" | ||
|  |           mode="aspectFill" | ||
|  |           @click="previewImage(noteDetail.image)" | ||
|  |         /> | ||
|  |       </view> | ||
|  | 
 | ||
|  |       <!-- 笔记内容 --> | ||
|  |       <view class="note-content"> | ||
|  |         <text class="content-text">{{ noteDetail.content }}</text> | ||
|  |       </view> | ||
|  | 
 | ||
|  |       <!-- 标签 --> | ||
|  |       <view | ||
|  |         class="tags-section" | ||
|  |         v-if="noteDetail.tags && noteDetail.tags.length" | ||
|  |       > | ||
|  |         <view class="tag-item" v-for="tag in noteDetail.tags" :key="tag"> | ||
|  |           #{{ tag }} | ||
|  |         </view> | ||
|  |       </view> | ||
|  | 
 | ||
|  |       <!-- 互动数据 --> | ||
|  |       <view class="interaction-section"> | ||
|  |         <view class="interaction-item" @click="toggleLike"> | ||
|  |           <text class="interaction-icon" :class="{ liked: noteDetail.isLiked }" | ||
|  |             >♥</text | ||
|  |           > | ||
|  |           <text class="interaction-text">{{ noteDetail.likes }}</text> | ||
|  |         </view> | ||
|  |         <view class="interaction-item" @click="toggleCollect"> | ||
|  |           <text | ||
|  |             class="interaction-icon" | ||
|  |             :class="{ collected: noteDetail.isCollected }" | ||
|  |             >★</text | ||
|  |           > | ||
|  |           <text class="interaction-text">{{ noteDetail.collects }}</text> | ||
|  |         </view> | ||
|  |         <view class="interaction-item" @click="showShareMenu"> | ||
|  |           <text class="interaction-icon">↗</text> | ||
|  |           <text class="interaction-text">分享</text> | ||
|  |         </view> | ||
|  |       </view> | ||
|  | 
 | ||
|  |       <!-- 评论区域 --> | ||
|  |       <view class="comments-section"> | ||
|  |         <view class="comments-header"> | ||
|  |           <text class="comments-title" | ||
|  |             >评论 ({{ noteDetail.comments.length }})</text | ||
|  |           > | ||
|  |         </view> | ||
|  | 
 | ||
|  |         <view | ||
|  |           class="comment-item" | ||
|  |           v-for="comment in noteDetail.comments" | ||
|  |           :key="comment.id" | ||
|  |         > | ||
|  |           <image | ||
|  |             class="comment-avatar" | ||
|  |             :src="comment.user.avatar" | ||
|  |             mode="aspectFill" | ||
|  |           /> | ||
|  |           <view class="comment-content"> | ||
|  |             <view class="comment-header"> | ||
|  |               <text class="comment-user">{{ comment.user.name }}</text> | ||
|  |               <text class="comment-time">{{ | ||
|  |                 formatTime(comment.createTime) | ||
|  |               }}</text> | ||
|  |             </view> | ||
|  |             <text class="comment-text">{{ comment.content }}</text> | ||
|  |             <view class="comment-actions"> | ||
|  |               <view class="comment-like" @click="toggleCommentLike(comment)"> | ||
|  |                 <text class="like-icon" :class="{ liked: comment.isLiked }" | ||
|  |                   >♥</text | ||
|  |                 > | ||
|  |                 <text class="like-count">{{ comment.likes }}</text> | ||
|  |               </view> | ||
|  |             </view> | ||
|  |           </view> | ||
|  |         </view> | ||
|  |       </view> | ||
|  | 
 | ||
|  |       <!-- 底部占位 --> | ||
|  |       <view class="bottom-placeholder"></view> | ||
|  |     </view> | ||
|  | 
 | ||
|  |     <!-- 底部评论输入框 --> | ||
|  |     <view class="comment-input-section"> | ||
|  |       <input | ||
|  |         class="comment-input" | ||
|  |         v-model="commentText" | ||
|  |         placeholder="写下你的想法..." | ||
|  |         @confirm="submitComment" | ||
|  |       /> | ||
|  |       <button | ||
|  |         class="send-btn" | ||
|  |         @click="submitComment" | ||
|  |         :disabled="!commentText.trim()" | ||
|  |       > | ||
|  |         发送 | ||
|  |       </button> | ||
|  |     </view> | ||
|  |   </view> | ||
|  | </template> | ||
|  | 
 | ||
|  | <script> | ||
|  | import headerVue from "@/components/header.vue"; | ||
|  | 
 | ||
|  | export default { | ||
|  |   name: "NoteDetail", | ||
|  |   components: { | ||
|  |     headerVue, | ||
|  |   }, | ||
|  |   data() { | ||
|  |     return { | ||
|  |       noteId: "", | ||
|  |       commentText: "", | ||
|  |       noteDetail: { | ||
|  |         id: "", | ||
|  |         title: "", | ||
|  |         content: "", | ||
|  |         image: "", | ||
|  |         tags: [], | ||
|  |         likes: 0, | ||
|  |         collects: 0, | ||
|  |         isLiked: false, | ||
|  |         isCollected: false, | ||
|  |         createTime: "", | ||
|  |         user: { | ||
|  |           id: "", | ||
|  |           name: "", | ||
|  |           avatar: "", | ||
|  |           isFollowed: false, | ||
|  |         }, | ||
|  |         comments: [], | ||
|  |       }, | ||
|  |     }; | ||
|  |   }, | ||
|  |   onLoad(options) { | ||
|  |     if (options.id) { | ||
|  |       this.noteId = options.id; | ||
|  |       this.loadNoteDetail(); | ||
|  |     } else { | ||
|  |       this.loadMockData(); | ||
|  |     } | ||
|  |   }, | ||
|  |   methods: { | ||
|  |     // 加载笔记详情
 | ||
|  |     async loadNoteDetail() { | ||
|  |       try { | ||
|  |         uni.showLoading({ title: "加载中..." }); | ||
|  |         // 模拟API调用
 | ||
|  |         const res = await this.getNoteDetail(this.noteId); | ||
|  |         this.noteDetail = res.data; | ||
|  |       } catch (error) { | ||
|  |         console.error("加载笔记详情失败:", error); | ||
|  |         uni.showToast({ | ||
|  |           title: "加载失败", | ||
|  |           icon: "none", | ||
|  |         }); | ||
|  |       } finally { | ||
|  |         uni.hideLoading(); | ||
|  |       } | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 加载模拟数据
 | ||
|  |     loadMockData() { | ||
|  |       this.noteDetail = { | ||
|  |         id: "mock001", | ||
|  |         title: "这里是用户发布内容的标题", | ||
|  |         content: | ||
|  |           "这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容这里是用户发布的内容", | ||
|  |         image: "https://picsum.photos/800/600", | ||
|  |         tags: ["时间力", "阅读体验"], | ||
|  |         likes: 128, | ||
|  |         collects: 64, | ||
|  |         isLiked: false, | ||
|  |         isCollected: false, | ||
|  |         createTime: "2024-01-15 14:30:00", | ||
|  |         user: { | ||
|  |           id: "user001", | ||
|  |           name: "杨璐摄影", | ||
|  |           avatar: | ||
|  |             "https://images.unsplash.com/photo-1535713875002-d1d0cf377fde?auto=format&fit=crop&w=100", | ||
|  |           isFollowed: false, | ||
|  |         }, | ||
|  |         comments: [ | ||
|  |           { | ||
|  |             id: "comment001", | ||
|  |             content: "很棒的分享,学到了很多!", | ||
|  |             likes: 5, | ||
|  |             isLiked: false, | ||
|  |             createTime: "2024-01-15 15:00:00", | ||
|  |             user: { | ||
|  |               id: "user002", | ||
|  |               name: "读书爱好者", | ||
|  |               avatar: | ||
|  |                 "https://images.unsplash.com/photo-1438761681033-6461ffad8d80?auto=format&fit=crop&w=100", | ||
|  |             }, | ||
|  |           }, | ||
|  |           { | ||
|  |             id: "comment002", | ||
|  |             content: "感谢分享,很有启发性的内容", | ||
|  |             likes: 3, | ||
|  |             isLiked: false, | ||
|  |             createTime: "2024-01-15 16:20:00", | ||
|  |             user: { | ||
|  |               id: "user003", | ||
|  |               name: "时间管理达人", | ||
|  |               avatar: | ||
|  |                 "https://images.unsplash.com/photo-1472099645785-5658abf4ff4e?auto=format&fit=crop&w=100", | ||
|  |             }, | ||
|  |           }, | ||
|  |         ], | ||
|  |       }; | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 预览图片
 | ||
|  |     previewImage(imageUrl) { | ||
|  |       uni.previewImage({ | ||
|  |         urls: [imageUrl], | ||
|  |         current: imageUrl, | ||
|  |       }); | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 切换关注状态
 | ||
|  |     toggleFollow() { | ||
|  |       this.noteDetail.user.isFollowed = !this.noteDetail.user.isFollowed; | ||
|  |       uni.showToast({ | ||
|  |         title: this.noteDetail.user.isFollowed ? "已关注" : "取消关注", | ||
|  |         icon: "none", | ||
|  |       }); | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 切换点赞状态
 | ||
|  |     toggleLike() { | ||
|  |       this.noteDetail.isLiked = !this.noteDetail.isLiked; | ||
|  |       this.noteDetail.likes += this.noteDetail.isLiked ? 1 : -1; | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 切换收藏状态
 | ||
|  |     toggleCollect() { | ||
|  |       this.noteDetail.isCollected = !this.noteDetail.isCollected; | ||
|  |       this.noteDetail.collects += this.noteDetail.isCollected ? 1 : -1; | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 显示分享菜单
 | ||
|  |     showShareMenu() { | ||
|  |       uni.share({ | ||
|  |         provider: "weixin", | ||
|  |         scene: "WXSceneSession", | ||
|  |         type: 0, | ||
|  |         href: `https://example.com/notes/${this.noteDetail.id}`, | ||
|  |         title: this.noteDetail.title, | ||
|  |         summary: this.noteDetail.content.substring(0, 100), | ||
|  |         imageUrl: this.noteDetail.image, | ||
|  |       }); | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 切换评论点赞
 | ||
|  |     toggleCommentLike(comment) { | ||
|  |       comment.isLiked = !comment.isLiked; | ||
|  |       comment.likes += comment.isLiked ? 1 : -1; | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 提交评论
 | ||
|  |     async submitComment() { | ||
|  |       if (!this.commentText.trim()) { | ||
|  |         return; | ||
|  |       } | ||
|  | 
 | ||
|  |       const newComment = { | ||
|  |         id: "comment" + Date.now(), | ||
|  |         content: this.commentText, | ||
|  |         likes: 0, | ||
|  |         isLiked: false, | ||
|  |         createTime: new Date().toISOString(), | ||
|  |         user: { | ||
|  |           id: "current_user", | ||
|  |           name: "当前用户", | ||
|  |           avatar: | ||
|  |             "https://images.unsplash.com/photo-1535713875002-d1d0cf377fde?auto=format&fit=crop&w=100", | ||
|  |         }, | ||
|  |       }; | ||
|  | 
 | ||
|  |       this.noteDetail.comments.unshift(newComment); | ||
|  |       this.commentText = ""; | ||
|  | 
 | ||
|  |       uni.showToast({ | ||
|  |         title: "评论成功", | ||
|  |         icon: "success", | ||
|  |       }); | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 格式化时间
 | ||
|  |     formatTime(timeString) { | ||
|  |       const time = new Date(timeString); | ||
|  |       const now = new Date(); | ||
|  |       const diff = now.getTime() - time.getTime(); | ||
|  | 
 | ||
|  |       if (diff < 60 * 1000) { | ||
|  |         return "刚刚"; | ||
|  |       } else if (diff < 60 * 60 * 1000) { | ||
|  |         return Math.floor(diff / (60 * 1000)) + "分钟前"; | ||
|  |       } else if (diff < 24 * 60 * 60 * 1000) { | ||
|  |         return Math.floor(diff / (60 * 60 * 1000)) + "小时前"; | ||
|  |       } else { | ||
|  |         return time.toLocaleDateString(); | ||
|  |       } | ||
|  |     }, | ||
|  | 
 | ||
|  |     // 模拟API - 获取笔记详情
 | ||
|  |     async getNoteDetail(noteId) { | ||
|  |       return new Promise((resolve) => { | ||
|  |         setTimeout(() => { | ||
|  |           this.loadMockData(); | ||
|  |           resolve({ | ||
|  |             code: 200, | ||
|  |             data: this.noteDetail, | ||
|  |           }); | ||
|  |         }, 500); | ||
|  |       }); | ||
|  |     }, | ||
|  |   }, | ||
|  | }; | ||
|  | </script> | ||
|  | 
 | ||
|  | <style lang="scss" scoped> | ||
|  | .note-detail-container { | ||
|  |   min-height: 100vh; | ||
|  |   background: #fff; | ||
|  |   display: flex; | ||
|  |   flex-direction: column; | ||
|  | } | ||
|  | 
 | ||
|  | .content-scroll { | ||
|  |   flex: 1; | ||
|  |   padding: 0 32rpx; | ||
|  | } | ||
|  | 
 | ||
|  | // 作者信息
 | ||
|  | .author-section { | ||
|  |   display: flex; | ||
|  |   align-items: center; | ||
|  |   padding: 32rpx 0; | ||
|  |   border-bottom: 1rpx solid #f0f0f0; | ||
|  | 
 | ||
|  |   .author-avatar { | ||
|  |     width: 80rpx; | ||
|  |     height: 80rpx; | ||
|  |     border-radius: 40rpx; | ||
|  |     margin-right: 24rpx; | ||
|  |   } | ||
|  | 
 | ||
|  |   .author-info { | ||
|  |     flex: 1; | ||
|  | 
 | ||
|  |     .author-name { | ||
|  |       display: block; | ||
|  |       font-size: 32rpx; | ||
|  |       font-weight: 600; | ||
|  |       color: #333; | ||
|  |       margin-bottom: 8rpx; | ||
|  |     } | ||
|  | 
 | ||
|  |     .publish-time { | ||
|  |       font-size: 24rpx; | ||
|  |       color: #999; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   .follow-btn { | ||
|  |     border-radius: 30rpx; | ||
|  |     font-size: 24rpx; | ||
|  |     border: 2rpx solid #ff4757; | ||
|  |     background: transparent; | ||
|  |     color: #ff4757; | ||
|  |     font-weight: 600; | ||
|  | 
 | ||
|  |     &.followed { | ||
|  |       background: #ff4757; | ||
|  |       color: #fff; | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | // 笔记标题
 | ||
|  | .note-title { | ||
|  |   font-size: 40rpx; | ||
|  |   font-weight: 600; | ||
|  |   color: #333; | ||
|  |   line-height: 1.4; | ||
|  |   margin: 32rpx 0; | ||
|  | } | ||
|  | 
 | ||
|  | // 笔记图片
 | ||
|  | .note-image-container { | ||
|  |   margin: 32rpx 0; | ||
|  | 
 | ||
|  |   .note-image { | ||
|  |     width: 100%; | ||
|  |     max-height: 800rpx; | ||
|  |     border-radius: 16rpx; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | // 笔记内容
 | ||
|  | .note-content { | ||
|  |   margin: 32rpx 0; | ||
|  | 
 | ||
|  |   .content-text { | ||
|  |     font-size: 32rpx; | ||
|  |     line-height: 1.6; | ||
|  |     color: #333; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | // 标签
 | ||
|  | .tags-section { | ||
|  |   display: flex; | ||
|  |   flex-wrap: wrap; | ||
|  |   gap: 16rpx; | ||
|  |   margin: 32rpx 0; | ||
|  | 
 | ||
|  |   .tag-item { | ||
|  |     padding: 12rpx 24rpx; | ||
|  |     background: #f8f9fa; | ||
|  |     border-radius: 32rpx; | ||
|  |     font-size: 24rpx; | ||
|  |     color: #666; | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | // 互动区域
 | ||
|  | .interaction-section { | ||
|  |   display: flex; | ||
|  |   align-items: center; | ||
|  |   gap: 48rpx; | ||
|  |   padding: 32rpx 0; | ||
|  |   border-bottom: 1rpx solid #f0f0f0; | ||
|  | 
 | ||
|  |   .interaction-item { | ||
|  |     display: flex; | ||
|  |     align-items: center; | ||
|  |     gap: 8rpx; | ||
|  |     cursor: pointer; | ||
|  | 
 | ||
|  |     .interaction-icon { | ||
|  |       font-size: 32rpx; | ||
|  |       color: #999; | ||
|  |       transition: color 0.3s; | ||
|  | 
 | ||
|  |       &.liked { | ||
|  |         color: #ff4757; | ||
|  |       } | ||
|  | 
 | ||
|  |       &.collected { | ||
|  |         color: #ffd700; | ||
|  |       } | ||
|  |     } | ||
|  | 
 | ||
|  |     .interaction-text { | ||
|  |       font-size: 28rpx; | ||
|  |       color: #666; | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | // 评论区域
 | ||
|  | .comments-section { | ||
|  |   margin: 32rpx 0; | ||
|  | 
 | ||
|  |   .comments-header { | ||
|  |     margin-bottom: 32rpx; | ||
|  | 
 | ||
|  |     .comments-title { | ||
|  |       font-size: 32rpx; | ||
|  |       font-weight: 600; | ||
|  |       color: #333; | ||
|  |     } | ||
|  |   } | ||
|  | 
 | ||
|  |   .comment-item { | ||
|  |     display: flex; | ||
|  |     margin-bottom: 32rpx; | ||
|  | 
 | ||
|  |     .comment-avatar { | ||
|  |       width: 64rpx; | ||
|  |       height: 64rpx; | ||
|  |       border-radius: 32rpx; | ||
|  |       margin-right: 24rpx; | ||
|  |     } | ||
|  | 
 | ||
|  |     .comment-content { | ||
|  |       flex: 1; | ||
|  | 
 | ||
|  |       .comment-header { | ||
|  |         display: flex; | ||
|  |         justify-content: space-between; | ||
|  |         align-items: center; | ||
|  |         margin-bottom: 12rpx; | ||
|  | 
 | ||
|  |         .comment-user { | ||
|  |           font-size: 28rpx; | ||
|  |           font-weight: 600; | ||
|  |           color: #333; | ||
|  |         } | ||
|  | 
 | ||
|  |         .comment-time { | ||
|  |           font-size: 24rpx; | ||
|  |           color: #999; | ||
|  |         } | ||
|  |       } | ||
|  | 
 | ||
|  |       .comment-text { | ||
|  |         font-size: 30rpx; | ||
|  |         line-height: 1.5; | ||
|  |         color: #333; | ||
|  |         margin-bottom: 16rpx; | ||
|  |       } | ||
|  | 
 | ||
|  |       .comment-actions { | ||
|  |         display: flex; | ||
|  |         align-items: center; | ||
|  | 
 | ||
|  |         .comment-like { | ||
|  |           display: flex; | ||
|  |           align-items: center; | ||
|  |           gap: 8rpx; | ||
|  |           cursor: pointer; | ||
|  | 
 | ||
|  |           .like-icon { | ||
|  |             font-size: 24rpx; | ||
|  |             color: #999; | ||
|  | 
 | ||
|  |             &.liked { | ||
|  |               color: #ff4757; | ||
|  |             } | ||
|  |           } | ||
|  | 
 | ||
|  |           .like-count { | ||
|  |             font-size: 24rpx; | ||
|  |             color: #999; | ||
|  |           } | ||
|  |         } | ||
|  |       } | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | // 底部评论输入
 | ||
|  | .comment-input-section { | ||
|  |   display: flex; | ||
|  |   align-items: center; | ||
|  |   padding: 24rpx 32rpx; | ||
|  |   background: #fff; | ||
|  |   border-top: 1rpx solid #f0f0f0; | ||
|  |   padding-bottom: calc(24rpx + env(safe-area-inset-bottom)); | ||
|  | position: fixed; | ||
|  | bottom: 0; | ||
|  | left: 0; | ||
|  | right: 0; | ||
|  | z-index: 999; | ||
|  |   .comment-input { | ||
|  |     flex: 1; | ||
|  |     height: 80rpx; | ||
|  |     background: #f8f9fa; | ||
|  |     border-radius: 40rpx; | ||
|  |     padding: 0 32rpx; | ||
|  |     font-size: 28rpx; | ||
|  |     border: none; | ||
|  |     margin-right: 16rpx; | ||
|  |   } | ||
|  | 
 | ||
|  |   .send-btn { | ||
|  |     width: 120rpx; | ||
|  |     height: 60rpx; | ||
|  | 	line-height: 60rpx; | ||
|  |     background: #ff4757; | ||
|  |     color: #fff; | ||
|  |     border-radius: 40rpx; | ||
|  |     font-size: 28rpx; | ||
|  |     border: none; | ||
|  |     font-weight: 600; | ||
|  | 
 | ||
|  |     &:disabled { | ||
|  |       background: #ccc; | ||
|  |     } | ||
|  |   } | ||
|  | } | ||
|  | 
 | ||
|  | .bottom-placeholder { | ||
|  |   height: 130rpx; | ||
|  |   padding-bottom: calc(24rpx + env(safe-area-inset-bottom)); | ||
|  |   padding-bottom: calc(24rpx + constant(safe-area-inset-bottom)); | ||
|  | } | ||
|  | </style> |