Compare commits
	
		
			No commits in common. "1850ffb7278d16f9e3a22460587f8d1fdfcf6f7c" and "0111453f06c9ba42630033d7d5dd40b6313266ef" have entirely different histories.
		
	
	
		
			1850ffb727
			...
			0111453f06
		
	
		
| @ -26,7 +26,6 @@ | |||||||
|     "@vueuse/core": "^10.7.0", |     "@vueuse/core": "^10.7.0", | ||||||
|     "ant-design-vue": "^4.2.6", |     "ant-design-vue": "^4.2.6", | ||||||
|     "axios": "^1.6.2", |     "axios": "^1.6.2", | ||||||
|     "dayjs": "^1.11.13", |  | ||||||
|     "highlight.js": "^11.5.0", |     "highlight.js": "^11.5.0", | ||||||
|     "js-audio-recorder": "^1.0.7", |     "js-audio-recorder": "^1.0.7", | ||||||
|     "lodash-es": "^4.17.21", |     "lodash-es": "^4.17.21", | ||||||
|  | |||||||
| @ -44,9 +44,6 @@ importers: | |||||||
|       axios: |       axios: | ||||||
|         specifier: ^1.6.2 |         specifier: ^1.6.2 | ||||||
|         version: 1.9.0 |         version: 1.9.0 | ||||||
|       dayjs: |  | ||||||
|         specifier: ^1.11.13 |  | ||||||
|         version: 1.11.13 |  | ||||||
|       highlight.js: |       highlight.js: | ||||||
|         specifier: ^11.5.0 |         specifier: ^11.5.0 | ||||||
|         version: 11.11.1 |         version: 11.11.1 | ||||||
|  | |||||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -62,15 +62,6 @@ const fileInfo = computed(() => { | |||||||
|   return fileTypes[extension] || fileTypes.DEFAULT |   return fileTypes[extension] || fileTypes.DEFAULT | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| // 判断文件是否可以预览 |  | ||||||
| const canPreview = computed(() => { |  | ||||||
|   const extension = getFileExtension(props.extra.path) |  | ||||||
|   return extension === 'PDF' ||  |  | ||||||
|          EXCEL_EXTENSIONS.includes(extension) ||  |  | ||||||
|          WORD_EXTENSIONS.includes(extension) ||  |  | ||||||
|          PPT_EXTENSIONS.includes(extension) |  | ||||||
| }) |  | ||||||
| 
 |  | ||||||
| // 获取文件扩展名 | // 获取文件扩展名 | ||||||
| function getFileExtension(filepath) { | function getFileExtension(filepath) { | ||||||
|   const parts = filepath?.split('.') |   const parts = filepath?.split('.') | ||||||
| @ -95,20 +86,15 @@ const strokeDashoffset = computed(() => | |||||||
| 
 | 
 | ||||||
| // 处理文件点击事件 | // 处理文件点击事件 | ||||||
| const handleClick = () => { | const handleClick = () => { | ||||||
|   // 只有在不上传中且文件类型支持预览时才打开预览窗口 |  | ||||||
|   if(!props.extra.is_uploading){ |   if(!props.extra.is_uploading){ | ||||||
|     if(canPreview.value){ |  | ||||||
|     window.open( |     window.open( | ||||||
|     `${import.meta.env.VITE_PAGE_URL}/office?url=${props.extra.path}`, |     `${import.meta.env.VITE_PAGE_URL}/office?url=${props.extra.path}`, | ||||||
|     '_blank', |     '_blank', | ||||||
|     'width=1200,height=900,left=200,top=200,toolbar=no,menubar=no,scrollbars=yes,resizable=yes,location=no,status=no' |     'width=1200,height=900,left=200,top=200,toolbar=no,menubar=no,scrollbars=yes,resizable=yes,location=no,status=no' | ||||||
|   ); |   ); | ||||||
|     }else{ |  | ||||||
|       window['$message'].warning('暂不支持在线预览该类型文件') |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
|  function downloadFileWithProgress(resourceUrl, filename) { |  function downloadFileWithProgress(resourceUrl, filename) { | ||||||
|   const iframe = document.createElement('iframe'); |   const iframe = document.createElement('iframe'); | ||||||
| @ -128,7 +114,7 @@ const handleDownload = () => { | |||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <div class="file-message flex flex-col can-preview"  @click="handleClick"> |   <div class="file-message flex flex-col" @click="handleClick"> | ||||||
|     <!-- 文件头部信息 --> |     <!-- 文件头部信息 --> | ||||||
|     <div class="file-header"> |     <div class="file-header"> | ||||||
|       <!-- 文件名 --> |       <!-- 文件名 --> | ||||||
| @ -198,14 +184,7 @@ const handleDownload = () => { | |||||||
|   border-radius: 8px; |   border-radius: 8px; | ||||||
|   box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); |   box-shadow: 0 2px 8px rgba(0, 0, 0, 0.1); | ||||||
|   padding: 0 14px; |   padding: 0 14px; | ||||||
| 
 |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| .can-preview { |  | ||||||
|   cursor: pointer; |   cursor: pointer; | ||||||
|   &:hover { |  | ||||||
|     background-color: #f9f9f9; |  | ||||||
|   } |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| .file-header { | .file-header { | ||||||
|  | |||||||
| @ -38,7 +38,7 @@ const img = (src: string, width = 200) => { | |||||||
|     <div class="image-container"> |     <div class="image-container"> | ||||||
|       <n-image class="h-149px" :src="extra.url" /> |       <n-image class="h-149px" :src="extra.url" /> | ||||||
|       <!-- 上传中的loading蒙版 --> |       <!-- 上传中的loading蒙版 --> | ||||||
|       <div v-if="extra.is_uploading" class="loading-overlay"> |       <div v-if="props.extra.is_uploading" class="loading-overlay"> | ||||||
|         <n-spin size="large" /> |         <n-spin size="large" /> | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
| @ -53,7 +53,7 @@ const img = (src: string, width = 200) => { | |||||||
|   height:149px; |   height:149px; | ||||||
|    |    | ||||||
|   &.left { |   &.left { | ||||||
|     background: #F4F4FC; |     background: var(--im-message-right-bg-color); | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .image-container { |   .image-container { | ||||||
|  | |||||||
| @ -115,7 +115,13 @@ const onRevoke = () => { | |||||||
|       </span> |       </span> | ||||||
| 
 | 
 | ||||||
|       <div style="display: inline-block;" v-if="login_uid === user_id"> |       <div style="display: inline-block;" v-if="login_uid === user_id"> | ||||||
|         <n-button @click="onRevoke" v-if="data.msg_type === 1&&data.extra?.content&&data.is_self_action" text class="text-#46299D text-11px">重新编辑</n-button> |         <n-button | ||||||
|  |           @click="onRevoke" | ||||||
|  |           v-if="data.msg_type === 1 && data.extra?.content" | ||||||
|  |           text | ||||||
|  |           class="text-#46299D text-11px" | ||||||
|  |           >重新编辑</n-button | ||||||
|  |         > | ||||||
|       </div> |       </div> | ||||||
|       <!-- <span v-if="login_uid == user_idA"> 你撤回B了一条消息 | {{ formatTime(datetime) }} </span> |       <!-- <span v-if="login_uid == user_idA"> 你撤回B了一条消息 | {{ formatTime(datetime) }} </span> | ||||||
|       <span v-else-if="login_uid == user_idB"> A撤回你了一条消息 | {{ formatTime(datetime) }} </span> |       <span v-else-if="login_uid == user_idB"> A撤回你了一条消息 | {{ formatTime(datetime) }} </span> | ||||||
| @ -136,7 +142,13 @@ const onRevoke = () => { | |||||||
|       </span> |       </span> | ||||||
|       <span v-if="talk_type === 2 && extra"> {{ extra }} | {{ formatTime(datetime) }} </span> |       <span v-if="talk_type === 2 && extra"> {{ extra }} | {{ formatTime(datetime) }} </span> | ||||||
|       <div style="display: inline-block;" v-if="login_uid === user_id"> |       <div style="display: inline-block;" v-if="login_uid === user_id"> | ||||||
|         <n-button @click="onRevoke" v-if="data.msg_type === 1&&data.extra?.content&&data.is_self_action" text class="text-#46299D text-11px">重新编辑</n-button> |         <n-button | ||||||
|  |           @click="onRevoke" | ||||||
|  |           v-if="data.msg_type === 1 && data.extra?.content" | ||||||
|  |           text | ||||||
|  |           class="text-#46299D text-11px" | ||||||
|  |           >重新编辑</n-button | ||||||
|  |         > | ||||||
|       </div> |       </div> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
|  | |||||||
| @ -1,6 +1,6 @@ | |||||||
| <script lang="ts" setup> | <script lang="ts" setup> | ||||||
| import 'xgplayer/dist/index.min.css' | import 'xgplayer/dist/index.min.css' | ||||||
| import { ref, nextTick, watch, computed } from 'vue' | import { ref, nextTick, watch } from 'vue' | ||||||
| import { NImage, NModal, NCard, NProgress, NPopconfirm } from 'naive-ui' | import { NImage, NModal, NCard, NProgress, NPopconfirm } from 'naive-ui' | ||||||
| import { Play, Close, Pause, Right, Attention } from '@icon-park/vue-next' | import { Play, Close, Pause, Right, Attention } from '@icon-park/vue-next' | ||||||
| import { getImageInfo } from '@/utils/functions' | import { getImageInfo } from '@/utils/functions' | ||||||
| @ -64,11 +64,6 @@ const updatePauseStatus = () => { | |||||||
| // 初始化时检查状态 | // 初始化时检查状态 | ||||||
| updatePauseStatus() | updatePauseStatus() | ||||||
| 
 | 
 | ||||||
| // 创建视频封面的URL |  | ||||||
| const videoSrc = computed(() => { |  | ||||||
|   // 即使在上传过程中也返回视频URL,这样可以显示视频封面 |  | ||||||
|   return props.extra.url || '' |  | ||||||
| }) |  | ||||||
| // // 监听关键道具变化 | // // 监听关键道具变化 | ||||||
| // watch(() => props.extra.percentage, (newVal: number | undefined) => { | // watch(() => props.extra.percentage, (newVal: number | undefined) => { | ||||||
| //   // 确保进度更新时 UI 也实时更新   | //   // 确保进度更新时 UI 也实时更新   | ||||||
| @ -141,7 +136,7 @@ function resumeUpload(e) { | |||||||
|   > |   > | ||||||
|    |    | ||||||
|     <!-- <n-image :src="extra.cover" preview-disabled /> --> |     <!-- <n-image :src="extra.cover" preview-disabled /> --> | ||||||
|     <video :src="videoSrc" :controls="false"></video> |     <video :src="props.extra.url" :controls="false"></video> | ||||||
|     <!-- 上传进度时的黑色半透明蒙层 --> |     <!-- 上传进度时的黑色半透明蒙层 --> | ||||||
|     <div v-if="extra.is_uploading && !uploadFailed" class="upload-mask"></div> |     <div v-if="extra.is_uploading && !uploadFailed" class="upload-mask"></div> | ||||||
|     <!-- 上传进度显示 --> |     <!-- 上传进度显示 --> | ||||||
| @ -257,7 +252,7 @@ function resumeUpload(e) { | |||||||
|   top: 0; |   top: 0; | ||||||
|   width: 100%; |   width: 100%; | ||||||
|   height: 100%; |   height: 100%; | ||||||
|   background: rgba(0, 0, 0, 0.3); /* 降低不透明度,从0.45改为0.3,让视频封面能够显示 */ |   background: rgba(0, 0, 0, 0.45); | ||||||
|   z-index: 1; |   z-index: 1; | ||||||
|   border-radius: 5px; |   border-radius: 5px; | ||||||
| } | } | ||||||
|  | |||||||
| @ -15,7 +15,7 @@ const { showUserInfoModal } = useInject() | |||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
| 
 | 
 | ||||||
|       <template v-for="(user, index) in extra.members" :key="index"> |       <template v-for="(user, index) in extra.members" :key="index"> | ||||||
|         <a>{{ user.nickname }}</a> |         <a @click="showUserInfoModal(user.erp_user_id,user.user_id)">{{ user.nickname }}</a> | ||||||
|         <em v-show="index < extra.members.length - 1">、</em> |         <em v-show="index < extra.members.length - 1">、</em> | ||||||
|       </template> |       </template> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ defineProps({ | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a > |       <a @click="showUserInfoModal(extra.owner_id)"> | ||||||
|         {{ extra.owner_name }} |         {{ extra.owner_name }} | ||||||
|       </a> |       </a> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -14,14 +14,14 @@ const { showUserInfoModal } = useInject() | |||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|        |        | ||||||
|       <a > |       <a @click="showUserInfoModal(extra.owner_id)"> | ||||||
|         {{ extra.owner_name }} |         {{ extra.owner_name }} | ||||||
|       </a> |       </a> | ||||||
| 
 | 
 | ||||||
|       <span>创建了群聊,并邀请了</span> |       <span>创建了群聊,并邀请了</span> | ||||||
| 
 | 
 | ||||||
|       <template v-for="(user, index) in extra.members" :key="index"> |       <template v-for="(user, index) in extra.members" :key="index"> | ||||||
|         <a >{{ user.nickname }}</a> |         <a @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> | ||||||
|         <em v-show="index < extra.members.length - 1">、</em> |         <em v-show="index < extra.members.length - 1">、</em> | ||||||
|       </template> |       </template> | ||||||
|     </div> |     </div> | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ defineProps({ | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a> |       <a @click="showUserInfoModal(data.user_id)"> | ||||||
|         <!-- {{ data.nickname }} --> |         <!-- {{ data.nickname }} --> | ||||||
|           管理员 |           管理员 | ||||||
|       </a> |       </a> | ||||||
|  | |||||||
| @ -13,14 +13,14 @@ const { showUserInfoModal } = useInject() | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a > |       <a @click="showUserInfoModal(extra.owner_id)"> | ||||||
|         {{ extra.owner_name }} |         {{ extra.owner_name }} | ||||||
|       </a> |       </a> | ||||||
| 
 | 
 | ||||||
|       <span>邀请了</span> |       <span>邀请了</span> | ||||||
| 
 | 
 | ||||||
|       <template v-for="(user, index) in extra.members" :key="index"> |       <template v-for="(user, index) in extra.members" :key="index"> | ||||||
|         <a>{{ user.nickname }}</a> |         <a @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> | ||||||
|         <em v-show="index < extra.members.length - 1">、</em> |         <em v-show="index < extra.members.length - 1">、</em> | ||||||
|       </template> |       </template> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,14 +13,14 @@ const { showUserInfoModal } = useInject() | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a > |       <a @click="showUserInfoModal(extra.owner_id)"> | ||||||
|         {{ extra.owner_name }} |         {{ extra.owner_name }} | ||||||
|       </a> |       </a> | ||||||
| 
 | 
 | ||||||
|       <span>解除了</span> |       <span>解除了</span> | ||||||
| 
 | 
 | ||||||
|       <template v-for="(user, index) in extra.members" :key="index"> |       <template v-for="(user, index) in extra.members" :key="index"> | ||||||
|         <a >{{ user.nickname }}</a> |         <a @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> | ||||||
|         <em v-show="index < extra.members.length - 1">、</em> |         <em v-show="index < extra.members.length - 1">、</em> | ||||||
|       </template> |       </template> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,14 +13,14 @@ const { showUserInfoModal } = useInject() | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a> |       <a @click="showUserInfoModal(extra.owner_id)"> | ||||||
|         {{ extra.owner_name }} |         {{ extra.owner_name }} | ||||||
|       </a> |       </a> | ||||||
| 
 | 
 | ||||||
|       <span>将</span> |       <span>将</span> | ||||||
| 
 | 
 | ||||||
|       <template v-for="(user, index) in extra.members" :key="index"> |       <template v-for="(user, index) in extra.members" :key="index"> | ||||||
|         <a>{{ user.nickname }}</a> |         <a @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> | ||||||
|         <em v-show="index < extra.members.length - 1">、</em> |         <em v-show="index < extra.members.length - 1">、</em> | ||||||
|       </template> |       </template> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,14 +13,14 @@ const { showUserInfoModal } = useInject() | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a> |       <a @click="showUserInfoModal(extra.owner_id)"> | ||||||
|         {{ extra.owner_name }} |         {{ extra.owner_name }} | ||||||
|       </a> |       </a> | ||||||
| 
 | 
 | ||||||
|       <span>设置了</span> |       <span>设置了</span> | ||||||
| 
 | 
 | ||||||
|       <template v-for="(user, index) in extra.members" :key="index"> |       <template v-for="(user, index) in extra.members" :key="index"> | ||||||
|         <a>{{ user.nickname }}</a> |         <a @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> | ||||||
|         <em v-show="index < extra.members.length - 1">、</em> |         <em v-show="index < extra.members.length - 1">、</em> | ||||||
|       </template> |       </template> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ const { showUserInfoModal } = useInject() | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a > |       <a @click="showUserInfoModal(extra.owner_id)"> | ||||||
|         {{ extra.owner_name }} |         {{ extra.owner_name }} | ||||||
|       </a> |       </a> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -14,7 +14,7 @@ defineProps({ | |||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <template v-for="(user, index) in extra?.members" :key="index"> |       <template v-for="(user, index) in extra?.members" :key="index"> | ||||||
|         <a >{{ user.nickname }}</a> |         <a @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> | ||||||
|         <em v-show="index < extra.members.length - 1">、</em> |         <em v-show="index < extra.members.length - 1">、</em> | ||||||
|       </template> |       </template> | ||||||
|       <span>已离开此群</span> |       <span>已离开此群</span> | ||||||
|  | |||||||
| @ -13,7 +13,7 @@ const { showUserInfoModal } = useInject() | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a > |       <a @click="showUserInfoModal(extra.owner_id)"> | ||||||
|         {{ extra.owner_name }} |         {{ extra.owner_name }} | ||||||
|       </a> |       </a> | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -13,9 +13,9 @@ const { showUserInfoModal } = useInject() | |||||||
| <template> | <template> | ||||||
|   <div class="im-message-sys-text"> |   <div class="im-message-sys-text"> | ||||||
|     <div class="sys-text"> |     <div class="sys-text"> | ||||||
|       <a >{{ extra.old_owner_name }}</a> |       <a @click="showUserInfoModal(extra.old_owner_id)">{{ extra.old_owner_name }}</a> | ||||||
|       <span>将群主转让给</span> |       <span>将群主转让给</span> | ||||||
|       <a >{{ extra.new_owner_name }}</a> |       <a @click="showUserInfoModal(extra.new_owner_id)">{{ extra.new_owner_name }}</a> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
| </template> | </template> | ||||||
|  | |||||||
| @ -10,6 +10,7 @@ | |||||||
|     padding: 0 8px; |     padding: 0 8px; | ||||||
|     word-wrap: break-word; |     word-wrap: break-word; | ||||||
|     color: #979191; |     color: #979191; | ||||||
|  |     user-select: none; | ||||||
|     font-weight: 300; |     font-weight: 300; | ||||||
|     display: inline-block; |     display: inline-block; | ||||||
|     border-radius: 3px; |     border-radius: 3px; | ||||||
| @ -22,11 +23,13 @@ | |||||||
| 
 | 
 | ||||||
|     a { |     a { | ||||||
|       color: #939596; |       color: #939596; | ||||||
|      |       cursor: pointer; | ||||||
|       font-size: 12px; |       font-size: 12px; | ||||||
|       font-weight: 400; |       font-weight: 400; | ||||||
| 
 | 
 | ||||||
|        |       &:hover { | ||||||
|  |         color: #462AA0; | ||||||
|  |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -131,6 +131,8 @@ const onSubmit = () => { | |||||||
|       talk_type: item.talk_type |       talk_type: item.talk_type | ||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
|  |   console.log('data', data); | ||||||
|  |   console.log('checkedFilter.value', checkedFilter.value); | ||||||
|   emit('on-submit', data) |   emit('on-submit', data) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -96,7 +96,6 @@ export const MessageComponents = { | |||||||
| 
 | 
 | ||||||
| // 可转发的消息类型
 | // 可转发的消息类型
 | ||||||
| export const ForwardableMessageType = [ | export const ForwardableMessageType = [ | ||||||
|   ChatMsgTypeForward, |  | ||||||
|   ChatMsgTypeText, |   ChatMsgTypeText, | ||||||
|   ChatMsgTypeCode, |   ChatMsgTypeCode, | ||||||
|   ChatMsgTypeImage, |   ChatMsgTypeImage, | ||||||
|  | |||||||
| @ -38,23 +38,23 @@ export function useSessionMenu() { | |||||||
| 
 | 
 | ||||||
|     const options: any[] = [] |     const options: any[] = [] | ||||||
| 
 | 
 | ||||||
|     // if (item.talk_type == 1) {
 |     if (item.talk_type == 1) { | ||||||
|     //   options.push({
 |       options.push({ | ||||||
|         |         | ||||||
|     //     label: '好友信息',
 |         label: '好友信息', | ||||||
|     //     key: 'info'
 |         key: 'info' | ||||||
|     //   })
 |       }) | ||||||
| 
 |  | ||||||
|     //   options.push({
 |  | ||||||
|       |  | ||||||
|     //     label: '修改备注',
 |  | ||||||
|     //     key: 'remark'
 |  | ||||||
|     //   })
 |  | ||||||
|     // }
 |  | ||||||
| 
 | 
 | ||||||
|       options.push({ |       options.push({ | ||||||
|       |       | ||||||
|       label: item.is_top ? '取消置顶' : '置顶', |         label: '修改备注', | ||||||
|  |         key: 'remark' | ||||||
|  |       }) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     options.push({ | ||||||
|  |    | ||||||
|  |       label: item.is_top ? '取消置顶' : '会话置顶', | ||||||
|       key: 'top' |       key: 'top' | ||||||
|     }) |     }) | ||||||
| 
 | 
 | ||||||
| @ -66,7 +66,7 @@ export function useSessionMenu() { | |||||||
| 
 | 
 | ||||||
|     options.push({ |     options.push({ | ||||||
|      |      | ||||||
|       label: '删除聊天', |       label: '移除会话', | ||||||
|       key: 'remove' |       key: 'remove' | ||||||
|     }) |     }) | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -248,6 +248,8 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
|       }).then((res) => { |       }).then((res) => { | ||||||
|         if (res.code == 200) { |         if (res.code == 200) { | ||||||
|           this.batchDelDialogueRecord(msgIds) |           this.batchDelDialogueRecord(msgIds) | ||||||
|  |         } else { | ||||||
|  |           window['$message'].warning(res.message) | ||||||
|         } |         } | ||||||
|       }) |       }) | ||||||
|     }, |     }, | ||||||
|  | |||||||
| @ -170,14 +170,14 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|        |        | ||||||
|       // 更新状态为上传中
 |       // 更新状态为上传中
 | ||||||
|       currentItem.status = 1 |       currentItem.status = 1 | ||||||
|       const updatedItem:any = this.findItem(uploadId) |        | ||||||
|       // 上传当前分片
 |       // 上传当前分片
 | ||||||
|       try { |       try { | ||||||
| 
 | 
 | ||||||
|         const res = await ServeFileSubareaUpload(form) |         const res = await ServeFileSubareaUpload(form) | ||||||
|    |    | ||||||
|         // 获取最新的项目状态,确保仍然存在且没有被暂停
 |         // 获取最新的项目状态,确保仍然存在且没有被暂停
 | ||||||
|         |         const updatedItem:any = this.findItem(uploadId) | ||||||
|         if (res.code == 200) { |         if (res.code == 200) { | ||||||
|           // 当前分片上传成功,增加索引
 |           // 当前分片上传成功,增加索引
 | ||||||
|           updatedItem.uploadIndex++ |           updatedItem.uploadIndex++ | ||||||
| @ -209,12 +209,10 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|          |          | ||||||
|         } |         } | ||||||
|       } catch (error) { |       } catch (error) { | ||||||
|         updatedItem.onProgress(-1) |  | ||||||
|         console.error("分片上传错误:", error); |         console.error("分片上传错误:", error); | ||||||
|          |          | ||||||
|         // 获取最新的项目状态
 |         // 获取最新的项目状态
 | ||||||
|         // 这里不应该重新定义变量,而是使用已有的updatedItem
 |         const updatedItem = this.findItem(uploadId) | ||||||
|         // const updatedItem = this.findItem(uploadId)
 |  | ||||||
|         if (!updatedItem) return |         if (!updatedItem) return | ||||||
|          |          | ||||||
|         // 如果是暂停导致的错误,不改变状态
 |         // 如果是暂停导致的错误,不改变状态
 | ||||||
|  | |||||||
| @ -18,7 +18,7 @@ export function isLoggedIn() { | |||||||
|  */ |  */ | ||||||
| export function getAccessToken() { | export function getAccessToken() { | ||||||
|   // return storage.get(AccessToken) || ''
 |   // return storage.get(AccessToken) || ''
 | ||||||
|   return JSON.parse(localStorage.getItem('token'))||'46d71a72d8d845ad7ed23eba9bdde260e635407190c2ce1bf7fd22088e41682ea07773ec65cae8946d2003f264d55961f96e0fc5da10eb96d3a348c1664e9644ce2108c311309f398ae8ea1b8200bfd490e5cb6e8c52c9e5d493cbabb163368f8351420451a631dbfa749829ee4cda49b77b5ed2d3dced5d0f2b7dd9ee76ba5465c84a17c23af040cd92b6b2a4ea48befbb5c729dcdad0a9c9668befe84074cc24f78899c1d947f8e7f94c7eda5325b8ed698df729e76febb98549ef3482ae942fb4f4a1c92d21836fa784728f0c5483aab2760a991b6b36e6b10c84f840a6433a6ecc31dee36e8f1c6158818bc89d2222dd6882c9bea84c2a8463737b8c2ebff9ca7c6d060fb963530bd14a2520dd9f2da63d38ec62765f787ba4ecad169369e97555f4f32b390f4cff376e30e2cb64f992f0f42a75cf0a559462e6c4e9cffe' |   return JSON.parse(localStorage.getItem('token'))||'46d71a72d8d845ad7ed23eba9bdde260e635407190c2ce1bf7fd22088e41682ea07773ec65cae8946d2003f264d55961f96e0fc5da10eb96d3a348c1664e9644ce2108c311309f398ae8ea1b8200bfd490e5cb6e8c52c9e5d493cbabb163368f8351420451a631dbfa749829ee4cda49b77b5ed2d3dced5d0f2b7dd9ee76ba5465c84a17c23af040cd92b6b2a4ea48befbb5c729dcdad0a9c9668befe84074cc24f78899c1d947f8e7f94c7eda5325b8ed698df729e76febb98549ef3482ae942fb4f4a1c92d21836fa784728f0c5483aab2760a991b6b36e6b10c84f840a6433a6ecc31dee36e8f1c6158818bc89d22c9c2f9b60a57573e8b08cdf47105e1ba85550c21fa55526e8a00bf316c623eb67abf749622c48beab908d61d3db7b22ed3eb6aa8a08c77680ad4d8a3458c1e72f97ba2b8480674df77f0501a34e82b58' | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  | |||||||
| @ -68,11 +68,6 @@ export function clipboard(text, callback) { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| export async function clipboardImage(src, callback) { | export async function clipboardImage(src, callback) { | ||||||
|   // 在wujie环境下使用主应用的clipboard
 |  | ||||||
|   const clipboardObj = window.__POWERED_BY_WUJIE__  |  | ||||||
|     ? window.parent.navigator.clipboard  |  | ||||||
|     : navigator.clipboard |  | ||||||
| 
 |  | ||||||
|   const { state } = await navigator.permissions.query({ |   const { state } = await navigator.permissions.query({ | ||||||
|     name: 'clipboard-write' |     name: 'clipboard-write' | ||||||
|   }) |   }) | ||||||
| @ -85,7 +80,7 @@ export async function clipboardImage(src, callback) { | |||||||
| 
 | 
 | ||||||
|     // navigator.clipboard.write 仅支持 png 图片
 |     // navigator.clipboard.write 仅支持 png 图片
 | ||||||
|     if (blob.type == 'image/png') { |     if (blob.type == 'image/png') { | ||||||
|       await clipboardObj.write([ |       await navigator.clipboard.write([ | ||||||
|         new ClipboardItem({ |         new ClipboardItem({ | ||||||
|           [blob.type]: blob |           [blob.type]: blob | ||||||
|         }) |         }) | ||||||
| @ -104,13 +99,13 @@ export async function clipboardImage(src, callback) { | |||||||
| 
 | 
 | ||||||
|       canvas.width = img.width |       canvas.width = img.width | ||||||
|       canvas.height = img.height |       canvas.height = img.height | ||||||
|       ctx.drawImage(img, 0, 0, canvas.width, canvas.height) |       ctx.drawImage(img, 0, 0) | ||||||
| 
 | 
 | ||||||
|       canvas.toBlob( |       canvas.toBlob( | ||||||
|         (blob) => { |         (blob) => { | ||||||
|           const data = [new ClipboardItem({ [blob.type]: blob })] |           const data = [new ClipboardItem({ [blob.type]: blob })] | ||||||
| 
 | 
 | ||||||
|           clipboardObj |           navigator.clipboard | ||||||
|             .write(data) |             .write(data) | ||||||
|             .then(() => { |             .then(() => { | ||||||
|               callback() |               callback() | ||||||
|  | |||||||
| @ -53,13 +53,7 @@ request.interceptors.request.use((config) => { | |||||||
| }, errorHandler) | }, errorHandler) | ||||||
| 
 | 
 | ||||||
| // 响应拦截器
 | // 响应拦截器
 | ||||||
| request.interceptors.response.use((response) => { | request.interceptors.response.use((response) => response.data, errorHandler) | ||||||
|   console.log('response.data.status',response.data.status) |  | ||||||
|   if(response.data.code !==200&&response.data.status!==0){ |  | ||||||
|     window['$message'].warning(response.data.msg) |  | ||||||
|   } |  | ||||||
|   return response.data |  | ||||||
| }, errorHandler) |  | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  * GET 请求 |  * GET 请求 | ||||||
|  | |||||||
| @ -20,6 +20,8 @@ const { useMessage } = useUtil() | |||||||
| const userStore = useUserStore() | const userStore = useUserStore() | ||||||
| const dialogueStore = useDialogueStore() | const dialogueStore = useDialogueStore() | ||||||
| const uploadsStore = useUploadsStore() | const uploadsStore = useUploadsStore() | ||||||
|  | console.log('dialogueStore', dialogueStore) | ||||||
|  | 
 | ||||||
| const members = computed(() => dialogueStore.members) | const members = computed(() => dialogueStore.members) | ||||||
| const membersByAlphabet = computed(() => { | const membersByAlphabet = computed(() => { | ||||||
|   if (state.searchMemberByAlphabet) { |   if (state.searchMemberByAlphabet) { | ||||||
|  | |||||||
| @ -31,7 +31,6 @@ const onSingleForward = () => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const onMultiDelete = () => { | const onMultiDelete = () => { | ||||||
|   if(dialogueStore.selectItems.length>0){ |  | ||||||
|   confirmBox({ |   confirmBox({ | ||||||
|     content:'确定删除聊天记录', |     content:'确定删除聊天记录', | ||||||
|     confirmText:'删除' |     confirmText:'删除' | ||||||
| @ -43,10 +42,6 @@ if (!msgIds.length) return | |||||||
| dialogueStore.ApiDeleteRecord(msgIds) | dialogueStore.ApiDeleteRecord(msgIds) | ||||||
|      |      | ||||||
|   }) |   }) | ||||||
|   }else{ |  | ||||||
|     window['$message'].warning('请选择聊天记录') |  | ||||||
|   } |  | ||||||
|   |  | ||||||
|   // 批量删除 |   // 批量删除 | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| @ -64,6 +59,7 @@ const onContactModal = (data: { receiver_id: number; talk_type: number }[]) => { | |||||||
|       group_ids.push(o.receiver_id) |       group_ids.push(o.receiver_id) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  |   console.log('user_ids',user_ids) | ||||||
|   dialogueStore.ApiForwardRecord({ |   dialogueStore.ApiForwardRecord({ | ||||||
|     mode: forwardMode.value, |     mode: forwardMode.value, | ||||||
|     message_ids: msg_ids, |     message_ids: msg_ids, | ||||||
|  | |||||||
| @ -82,7 +82,7 @@ const { loadConfig, records, onLoad, onRefreshLoad, onJumpMessage, onLoadMoreDow | |||||||
| ) | ) | ||||||
| const uploadsStore = useUploadsStore() | const uploadsStore = useUploadsStore() | ||||||
| const { useMessage } = useUtil() | const { useMessage } = useUtil() | ||||||
| const { dropdown, showDropdownMenu, closeDropdownMenu, isOneMonthBefore } = useMenu() | const { dropdown, showDropdownMenu, closeDropdownMenu } = useMenu() | ||||||
| const { showUserInfoModal } = useInject() | const { showUserInfoModal } = useInject() | ||||||
| const dialogueStore = useDialogueStore() | const dialogueStore = useDialogueStore() | ||||||
| const userStore = useUserStore() | const userStore = useUserStore() | ||||||
| @ -184,7 +184,7 @@ const onCopyText = (data: ITalkRecord) => { | |||||||
|       return clipboard(htmlDecode(data.extra.content), () => useMessage.success('复制成功')) |       return clipboard(htmlDecode(data.extra.content), () => useMessage.success('复制成功')) | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   console.log('data.extra?.url', data.extra?.url) | 
 | ||||||
|   if (data.extra?.url) { |   if (data.extra?.url) { | ||||||
|     return clipboardImage(data.extra.url, () => { |     return clipboardImage(data.extra.url, () => { | ||||||
|       useMessage.success('复制成功') |       useMessage.success('复制成功') | ||||||
| @ -330,15 +330,12 @@ const onContextMenuHandle = (key: string) => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const onRowClick = (item: ITalkRecord) => { | const onRowClick = (item: ITalkRecord) => { | ||||||
|   if (dialogueStore.isOpenMultiSelect && isOneMonthBefore(item.created_at.split(' ')[0])) { |   if (dialogueStore.isOpenMultiSelect) { | ||||||
|     console.log('item.msg_type', item.msg_type) |  | ||||||
|     if (ForwardableMessageType.includes(item.msg_type)) { |     if (ForwardableMessageType.includes(item.msg_type)) { | ||||||
|       item.isCheck = !item.isCheck |       item.isCheck = !item.isCheck | ||||||
|     } else { |     } else { | ||||||
|       useMessage.info('此类消息不支持转发') |       useMessage.info('此类消息不支持转发') | ||||||
|     } |     } | ||||||
|   } else { |  | ||||||
|     useMessage.info('只支持转发近一个月内的消息') |  | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -818,10 +815,8 @@ const onCustomSkipBottomEvent = () => { | |||||||
|         > |         > | ||||||
|           <!-- 多选按钮 --> |           <!-- 多选按钮 --> | ||||||
|           <aside v-if="dialogueStore.isOpenMultiSelect" class="checkbox-column shrink-0"> |           <aside v-if="dialogueStore.isOpenMultiSelect" class="checkbox-column shrink-0"> | ||||||
|             <!-- 近一个月外的消息多选框禁用 {{ item }} --> |  | ||||||
|             <n-checkbox |             <n-checkbox | ||||||
|               size="small" |               size="small" | ||||||
|               :disabled="!isOneMonthBefore(item.created_at.split(' ')[0])" |  | ||||||
|               :checked="item.isCheck" |               :checked="item.isCheck" | ||||||
|               @update:checked="item.isCheck = !item.isCheck" |               @update:checked="item.isCheck = !item.isCheck" | ||||||
|             /> |             /> | ||||||
| @ -859,14 +854,7 @@ const onCustomSkipBottomEvent = () => { | |||||||
|             </div> |             </div> | ||||||
|             <div |             <div | ||||||
|               class="talk-content" |               class="talk-content" | ||||||
|               :class="{ |               :class="{ pointer: dialogueStore.isOpenMultiSelect }" | ||||||
|                 pointer: |  | ||||||
|                   dialogueStore.isOpenMultiSelect && |  | ||||||
|                   isOneMonthBefore(item.created_at.split(' ')[0]), |  | ||||||
|                 'cursor-not-allowed': |  | ||||||
|                   dialogueStore.isOpenMultiSelect && |  | ||||||
|                   !isOneMonthBefore(item.created_at.split(' ')[0]) |  | ||||||
|               }" |  | ||||||
|               @click="onRowClick(item)" |               @click="onRowClick(item)" | ||||||
|             > |             > | ||||||
|               <component |               <component | ||||||
| @ -883,7 +871,7 @@ const onCustomSkipBottomEvent = () => { | |||||||
|                 " |                 " | ||||||
|                 class="mr-10px" |                 class="mr-10px" | ||||||
|               > |               > | ||||||
|                 <n-button text style="font-size: 20px" @click="retry(item)"> |                 <n-button text style="font-size: 20px;" @click="retry(item)"> | ||||||
|                   <n-icon color="#CF3050"> |                   <n-icon color="#CF3050"> | ||||||
|                     <ExclamationCircleFilled /> |                     <ExclamationCircleFilled /> | ||||||
|                   </n-icon> |                   </n-icon> | ||||||
| @ -928,14 +916,14 @@ const onCustomSkipBottomEvent = () => { | |||||||
|               <n-popover |               <n-popover | ||||||
|                 trigger="click" |                 trigger="click" | ||||||
|                 placement="bottom-end" |                 placement="bottom-end" | ||||||
|                 style="height: 382px; padding: 0" |                 style="height: 382px; padding: 0;" | ||||||
|                 v-if="props.talk_type === 2" |                 v-if="props.talk_type === 2" | ||||||
|               > |               > | ||||||
|                 <template #trigger> |                 <template #trigger> | ||||||
|                   <span |                   <span | ||||||
|                     v-if="props.talk_type === 2" |                     v-if="props.talk_type === 2" | ||||||
|                     @click="toShowMessageReadDetail(item)" |                     @click="toShowMessageReadDetail(item)" | ||||||
|                     style="cursor: pointer" |                     style="cursor: pointer;" | ||||||
|                   > |                   > | ||||||
|                     已读 ({{ item?.read_total_num || 0 }}/{{ |                     已读 ({{ item?.read_total_num || 0 }}/{{ | ||||||
|                       props.num - 1 > 0 ? props.num - 1 : 0 |                       props.num - 1 > 0 ? props.num - 1 : 0 | ||||||
| @ -957,12 +945,11 @@ const onCustomSkipBottomEvent = () => { | |||||||
|                     </n-tab> |                     </n-tab> | ||||||
|                   </n-tabs> |                   </n-tabs> | ||||||
|                   <div class="talk-read-list"> |                   <div class="talk-read-list"> | ||||||
|                     <n-infinite-scroll style="height: 340px" @load="loadMoreReadListDetail"> |                     <n-infinite-scroll style="height: 340px;" @load="loadMoreReadListDetail"> | ||||||
|                       <div |                       <div | ||||||
|                         class="talk-read-list-item" |                         class="talk-read-list-item" | ||||||
|                         v-for="( |                         v-for="(talkReadDetailItem, | ||||||
|                           talkReadDetailItem, talkReadDetailIndex |                         talkReadDetailIndex) in state.talkReadListDetail" | ||||||
|                         ) in state.talkReadListDetail" |  | ||||||
|                         :key="talkReadDetailIndex" |                         :key="talkReadDetailIndex" | ||||||
|                       > |                       > | ||||||
|                         <avatarModule |                         <avatarModule | ||||||
| @ -982,10 +969,10 @@ const onCustomSkipBottomEvent = () => { | |||||||
|                           }" |                           }" | ||||||
|                         ></avatarModule> |                         ></avatarModule> | ||||||
|                         <div class="talk-read-list-item-info"> |                         <div class="talk-read-list-item-info"> | ||||||
|                           <span style="font-size: 12px; font-weight: 600; line-height: 17px">{{ |                           <span style="font-size: 12px; font-weight: 600; line-height: 17px;">{{ | ||||||
|                             talkReadDetailItem.nickName |                             talkReadDetailItem.nickName | ||||||
|                           }}</span> |                           }}</span> | ||||||
|                           <span style="font-size: 12px; color: #999; line-height: 14px">{{ |                           <span style="font-size: 12px; color: #999; line-height: 14px;">{{ | ||||||
|                             talkReadDetailItem.jobNum |                             talkReadDetailItem.jobNum | ||||||
|                           }}</span> |                           }}</span> | ||||||
|                         </div> |                         </div> | ||||||
| @ -1017,7 +1004,7 @@ const onCustomSkipBottomEvent = () => { | |||||||
|     :show="dropdown.show" |     :show="dropdown.show" | ||||||
|     :x="dropdown.x" |     :x="dropdown.x" | ||||||
|     :y="dropdown.y" |     :y="dropdown.y" | ||||||
|     style="width: 142px" |     style="width: 142px;" | ||||||
|     :options="dropdown.options" |     :options="dropdown.options" | ||||||
|     @select="onContextMenuHandle" |     @select="onContextMenuHandle" | ||||||
|     @clickoutside="closeDropdownMenu" |     @clickoutside="closeDropdownMenu" | ||||||
|  | |||||||
| @ -101,13 +101,10 @@ const onSendImageEvent = ({ data, callBack }) => { | |||||||
| 
 | 
 | ||||||
| // 发送视频消息 | // 发送视频消息 | ||||||
| const onSendVideoEvent = async ({ data }) => { | const onSendVideoEvent = async ({ data }) => { | ||||||
|  | 
 | ||||||
|  |    | ||||||
|   // 获取视频首帧作为封面图 |   // 获取视频首帧作为封面图 | ||||||
|   let videoPreview = null |   // let resp = await getVideoImage(data) | ||||||
|   try { |  | ||||||
|     videoPreview = await getVideoImage(data) |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('获取视频封面失败:', error) |  | ||||||
|   } |  | ||||||
|    |    | ||||||
|   // 先创建一个带有上传ID的临时消息对象,用于显示进度 |   // 先创建一个带有上传ID的临时消息对象,用于显示进度 | ||||||
|   const uploadId = `video-${Date.now()}-${Math.floor(Math.random() * 1000)}` |   const uploadId = `video-${Date.now()}-${Math.floor(Math.random() * 1000)}` | ||||||
| @ -126,7 +123,7 @@ const onSendVideoEvent = async ({ data }) => { | |||||||
|     content: '', |     content: '', | ||||||
|     created_at: parseTime(new Date(), '{y}-{m}-{d} {h}:{i}'), |     created_at: parseTime(new Date(), '{y}-{m}-{d} {h}:{i}'), | ||||||
|     extra: { |     extra: { | ||||||
|       url: videoPreview ? URL.createObjectURL(data) : '', // 使用本地视频URL作为预览 |       url: '',  | ||||||
|       size: data.size, |       size: data.size, | ||||||
|       is_uploading: true, |       is_uploading: true, | ||||||
|       upload_id: uploadId, |       upload_id: uploadId, | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| import { reactive } from 'vue' | import { reactive } from 'vue' | ||||||
| import dayjs from 'dayjs' |  | ||||||
| import { useDialogueStore } from '@/store/modules/dialogue.js' | import { useDialogueStore } from '@/store/modules/dialogue.js' | ||||||
| 
 | 
 | ||||||
| interface IDropdown { | interface IDropdown { | ||||||
| @ -10,34 +9,16 @@ interface IDropdown { | |||||||
|   item: any |   item: any | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const isRevoke = (uid: number, item: any): boolean => { | const isRevoke = (uid: any, item: any): boolean => { | ||||||
|   // 不是自己发的消息不能撤回
 |   if (uid != item.user_id) { | ||||||
|   if (uid !== item.user_id) { |     return false | ||||||
|     return false; |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // 检查消息是否在撤回时间限制内(5分钟)
 |   const datetime = item.created_at.replace(/-/g, '/') | ||||||
|   const messageTime = dayjs(item.created_at); | 
 | ||||||
|   const now = dayjs(); |   const time = new Date().getTime() - Date.parse(datetime) | ||||||
|   const diffInMinutes = now.diff(messageTime, 'minute'); | 
 | ||||||
|   return diffInMinutes <= 5; |   return Math.floor(time / 1000 / 60) <= 2 | ||||||
| } |  | ||||||
| // 判断是否可以添加撤回选项的函数
 |  | ||||||
| const canAddRevokeOption = (uid: number, item: any, isManager: boolean): boolean => { |  | ||||||
|   // 单聊情况:自己发的且在时间限制内
 |  | ||||||
|   if (item.talk_type === 1) { |  | ||||||
|     return isRevoke(uid, item) && item.float === 'right'; |  | ||||||
|   } |  | ||||||
|   // 群聊情况
 |  | ||||||
|   else if (item.talk_type === 2) { |  | ||||||
|     // 管理员可以撤回任何消息
 |  | ||||||
|     if (isManager) { |  | ||||||
|       return true; |  | ||||||
|     } |  | ||||||
|     // 普通成员只能撤回自己的且在时间限制内的消息
 |  | ||||||
|     return isRevoke(uid, item) && item.float === 'right'; |  | ||||||
|   } |  | ||||||
|   return false; |  | ||||||
| } | } | ||||||
| const dialogueStore = useDialogueStore() | const dialogueStore = useDialogueStore() | ||||||
| export function useMenu() { | export function useMenu() { | ||||||
| @ -48,17 +29,10 @@ export function useMenu() { | |||||||
|     y: 0, |     y: 0, | ||||||
|     item: {} |     item: {} | ||||||
|   }) |   }) | ||||||
|   // 判断时间是否超过一个月
 | 
 | ||||||
|   function isOneMonthBefore(date) { |  | ||||||
|     const oneMonthAgo = new Date() |  | ||||||
|     oneMonthAgo.setMonth(oneMonthAgo.getMonth() - 1) |  | ||||||
|     const inputDate = new Date(date) |  | ||||||
|     return !(inputDate <= oneMonthAgo) |  | ||||||
|   } |  | ||||||
|   const showDropdownMenu = (e: any, uid: number, item: any) => { |   const showDropdownMenu = (e: any, uid: number, item: any) => { | ||||||
|   //  dropdown.item = Object.assign({}, item)
 |   //  dropdown.item = Object.assign({}, item)
 | ||||||
|   dropdown.item = item |   dropdown.item = item | ||||||
|   dropdown.item.is_self_action = true |  | ||||||
|     dropdown.options = [] |     dropdown.options = [] | ||||||
|     if ([4].includes(item.msg_type)) { |     if ([4].includes(item.msg_type)) { | ||||||
|       if(item.is_convert_text === 1){ |       if(item.is_convert_text === 1){ | ||||||
| @ -66,22 +40,21 @@ export function useMenu() { | |||||||
|       }else{ |       }else{ | ||||||
|         dropdown.options.push({ label: '转文字', key: 'convertText' }) |         dropdown.options.push({ label: '转文字', key: 'convertText' }) | ||||||
|       } |       } | ||||||
|  |      | ||||||
|     } |     } | ||||||
|     if ([1, 3].includes(item.msg_type)) { |     if ([1, 3].includes(item.msg_type)) { | ||||||
|       dropdown.options.push({ label: '复制', key: 'copy' }) |       dropdown.options.push({ label: '复制', key: 'copy' }) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (isOneMonthBefore(new Date(item.created_at.split(' ')[0]))) { |  | ||||||
|       // 根据时间判断只有近一个月内的消息才能支持多选
 |  | ||||||
|     dropdown.options.push({ label: '多选', key: 'multiSelect' }) |     dropdown.options.push({ label: '多选', key: 'multiSelect' }) | ||||||
|     } |  | ||||||
|     dropdown.options.push({ label: '引用', key: 'quote' }) |     dropdown.options.push({ label: '引用', key: 'quote' }) | ||||||
|     if (canAddRevokeOption(uid, item, (dialogueStore.groupInfo as any).is_manager)) { |     if (isRevoke(uid, item)|| (dialogueStore.groupInfo as any).is_manager) { | ||||||
|       dropdown.options.push({ label: '撤回', key: 'revoke' }); |       dropdown.options.push({ label: `撤回`, key: 'revoke' }) | ||||||
|     } |     } | ||||||
|     |  | ||||||
|     dropdown.options.push({ label: '删除', key: 'delete' }) |     dropdown.options.push({ label: '删除', key: 'delete' }) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|     // if ([3, 4, 5].includes(item.msg_type)) {
 |     // if ([3, 4, 5].includes(item.msg_type)) {
 | ||||||
|     //   dropdown.options.push({ label: '下载', key: 'download' })
 |     //   dropdown.options.push({ label: '下载', key: 'download' })
 | ||||||
|     // }
 |     // }
 | ||||||
| @ -90,6 +63,7 @@ export function useMenu() { | |||||||
|     //   dropdown.options.push({ label: '收藏', key: 'collect' })
 |     //   dropdown.options.push({ label: '收藏', key: 'collect' })
 | ||||||
|     // }
 |     // }
 | ||||||
|     |     | ||||||
|  | 
 | ||||||
|     dropdown.x = e.clientX |     dropdown.x = e.clientX | ||||||
|     dropdown.y = e.clientY |     dropdown.y = e.clientY | ||||||
|     dropdown.show = true |     dropdown.show = true | ||||||
| @ -100,5 +74,5 @@ export function useMenu() { | |||||||
|     dropdown.item = {} |     dropdown.item = {} | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return { dropdown, showDropdownMenu, closeDropdownMenu, isOneMonthBefore } |   return { dropdown, showDropdownMenu, closeDropdownMenu } | ||||||
| } | } | ||||||
|  | |||||||
| @ -57,6 +57,7 @@ const config = { | |||||||
|   }, |   }, | ||||||
|   documentType, |   documentType, | ||||||
|   editorConfig: { |   editorConfig: { | ||||||
|  |      | ||||||
|     mode: 'view', |     mode: 'view', | ||||||
|     lang: 'zh-CN', |     lang: 'zh-CN', | ||||||
|     user: { |     user: { | ||||||
|  | |||||||
| @ -46,9 +46,9 @@ export default defineConfig(({ mode }) => { | |||||||
|       vueJsx({}),  |       vueJsx({}),  | ||||||
|       compressPlugin(),  |       compressPlugin(),  | ||||||
|       UnoCSS(), |       UnoCSS(), | ||||||
|       // vueDevTools({
 |       vueDevTools({ | ||||||
|       //   launchEditor: 'trae',
 |         launchEditor: 'trae', | ||||||
|       // })
 |       }) | ||||||
|     ], |     ], | ||||||
|     define: { |     define: { | ||||||
|       __APP_ENV__: env.APP_ENV |       __APP_ENV__: env.APP_ENV | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user