Compare commits
	
		
			43 Commits
		
	
	
		
			d0dd83451c
			...
			f876ee7bbe
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | f876ee7bbe | ||
|  | db8621ec5c | ||
| 07c3808122 | |||
|  | b28c288665 | ||
|  | ca958bb2cb | ||
|  | fd9a5555dc | ||
|  | 7733f88dae | ||
|  | 1a85e9d13e | ||
|  | bab907a1e2 | ||
|  | a506b4dcc1 | ||
|  | 45e4415cec | ||
|  | 57e4ba69d9 | ||
|  | 88bbf16699 | ||
|  | d46ced7614 | ||
|  | 044617580c | ||
|  | 54a46e2fb4 | ||
|  | 28938aba66 | ||
|  | 8e645226b8 | ||
| 97f05d2c5c | |||
| 8d73e0d48b | |||
|  | 4b5c160e94 | ||
|  | ebd567a757 | ||
| 18871db6b6 | |||
|  | 1ae317dbb3 | ||
|  | e4354d42cd | ||
|  | 8bba2d64af | ||
|  | d4e52152ef | ||
|  | bdf07155c8 | ||
|  | b905db0cfa | ||
|  | 3b6d998ce1 | ||
|  | 5340461a7e | ||
|  | 45eec2ff22 | ||
|  | 9c34066128 | ||
|  | 628894a254 | ||
| 92fce58429 | |||
|  | 2e998a1174 | ||
|  | 60a2fb996b | ||
| b282562cdd | |||
| d0abf7d8ab | |||
|  | 409af72039 | ||
|  | 799599bd83 | ||
| ec18d85546 | |||
|  | a97f293a6c | 
| @ -26,6 +26,7 @@ | |||||||
|     "@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,6 +44,9 @@ 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,6 +62,15 @@ 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('.') | ||||||
| @ -86,15 +95,20 @@ 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'); | ||||||
| @ -114,7 +128,7 @@ const handleDownload = () => { | |||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <div class="file-message flex flex-col" @click="handleClick"> |   <div class="file-message flex flex-col can-preview"  @click="handleClick"> | ||||||
|     <!-- 文件头部信息 --> |     <!-- 文件头部信息 --> | ||||||
|     <div class="file-header"> |     <div class="file-header"> | ||||||
|       <!-- 文件名 --> |       <!-- 文件名 --> | ||||||
| @ -184,7 +198,14 @@ 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="props.extra.is_uploading" class="loading-overlay"> |       <div v-if="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: var(--im-message-right-bg-color); |     background: #F4F4FC; | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   .image-container { |   .image-container { | ||||||
|  | |||||||
| @ -115,13 +115,7 @@ 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 |         <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> | ||||||
|           @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> | ||||||
| @ -142,13 +136,7 @@ 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 |         <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> | ||||||
|           @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 } from 'vue' | import { ref, nextTick, watch, computed } 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,6 +64,11 @@ 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 也实时更新   | ||||||
| @ -136,7 +141,7 @@ function resumeUpload(e) { | |||||||
|   > |   > | ||||||
|    |    | ||||||
|     <!-- <n-image :src="extra.cover" preview-disabled /> --> |     <!-- <n-image :src="extra.cover" preview-disabled /> --> | ||||||
|     <video :src="props.extra.url" :controls="false"></video> |     <video :src="videoSrc" :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> | ||||||
|     <!-- 上传进度显示 --> |     <!-- 上传进度显示 --> | ||||||
| @ -252,7 +257,7 @@ function resumeUpload(e) { | |||||||
|   top: 0; |   top: 0; | ||||||
|   width: 100%; |   width: 100%; | ||||||
|   height: 100%; |   height: 100%; | ||||||
|   background: rgba(0, 0, 0, 0.45); |   background: rgba(0, 0, 0, 0.3); /* 降低不透明度,从0.45改为0.3,让视频封面能够显示 */ | ||||||
|   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 @click="showUserInfoModal(user.erp_user_id,user.user_id)">{{ user.nickname }}</a> |         <a>{{ 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 @click="showUserInfoModal(extra.owner_id)"> |       <a > | ||||||
|         {{ 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 @click="showUserInfoModal(extra.owner_id)"> |       <a > | ||||||
|         {{ 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 @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> |         <a >{{ 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 @click="showUserInfoModal(data.user_id)"> |       <a> | ||||||
|         <!-- {{ 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 @click="showUserInfoModal(extra.owner_id)"> |       <a > | ||||||
|         {{ 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 @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> |         <a>{{ 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 @click="showUserInfoModal(extra.owner_id)"> |       <a > | ||||||
|         {{ 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 @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> |         <a >{{ 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 @click="showUserInfoModal(extra.owner_id)"> |       <a> | ||||||
|         {{ 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 @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> |         <a>{{ 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 @click="showUserInfoModal(extra.owner_id)"> |       <a> | ||||||
|         {{ 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 @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> |         <a>{{ 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 @click="showUserInfoModal(extra.owner_id)"> |       <a > | ||||||
|         {{ 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 @click="showUserInfoModal(user.user_id)">{{ user.nickname }}</a> |         <a >{{ 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 @click="showUserInfoModal(extra.owner_id)"> |       <a > | ||||||
|         {{ 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 @click="showUserInfoModal(extra.old_owner_id)">{{ extra.old_owner_name }}</a> |       <a >{{ extra.old_owner_name }}</a> | ||||||
|       <span>将群主转让给</span> |       <span>将群主转让给</span> | ||||||
|       <a @click="showUserInfoModal(extra.new_owner_id)">{{ extra.new_owner_name }}</a> |       <a >{{ extra.new_owner_name }}</a> | ||||||
|     </div> |     </div> | ||||||
|   </div> |   </div> | ||||||
| </template> | </template> | ||||||
|  | |||||||
| @ -10,7 +10,6 @@ | |||||||
|     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; | ||||||
| @ -23,13 +22,11 @@ | |||||||
| 
 | 
 | ||||||
|     a { |     a { | ||||||
|       color: #939596; |       color: #939596; | ||||||
|       cursor: pointer; |      | ||||||
|       font-size: 12px; |       font-size: 12px; | ||||||
|       font-weight: 400; |       font-weight: 400; | ||||||
| 
 | 
 | ||||||
|       &:hover { |        | ||||||
|         color: #462AA0; |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | |||||||
| @ -131,8 +131,6 @@ 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,6 +96,7 @@ 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: '修改备注', |       label: item.is_top ? '取消置顶' : '置顶', | ||||||
|         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,8 +248,6 @@ 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,10 +209,12 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|          |          | ||||||
|         } |         } | ||||||
|       } catch (error) { |       } catch (error) { | ||||||
|  |         updatedItem.onProgress(-1) | ||||||
|         console.error("分片上传错误:", error); |         console.error("分片上传错误:", error); | ||||||
|          |          | ||||||
|         // 获取最新的项目状态
 |         // 获取最新的项目状态
 | ||||||
|         const updatedItem = this.findItem(uploadId) |         // 这里不应该重新定义变量,而是使用已有的updatedItem
 | ||||||
|  |         // 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'))||'46d71a72d8d845ad7ed23eba9bdde260e635407190c2ce1bf7fd22088e41682ea07773ec65cae8946d2003f264d55961f96e0fc5da10eb96d3a348c1664e9644ce2108c311309f398ae8ea1b8200bfd490e5cb6e8c52c9e5d493cbabb163368f8351420451a631dbfa749829ee4cda49b77b5ed2d3dced5d0f2b7dd9ee76ba5465c84a17c23af040cd92b6b2a4ea48befbb5c729dcdad0a9c9668befe84074cc24f78899c1d947f8e7f94c7eda5325b8ed698df729e76febb98549ef3482ae942fb4f4a1c92d21836fa784728f0c5483aab2760a991b6b36e6b10c84f840a6433a6ecc31dee36e8f1c6158818bc89d22c9c2f9b60a57573e8b08cdf47105e1ba85550c21fa55526e8a00bf316c623eb67abf749622c48beab908d61d3db7b22ed3eb6aa8a08c77680ad4d8a3458c1e72f97ba2b8480674df77f0501a34e82b58' |   return JSON.parse(localStorage.getItem('token'))||'46d71a72d8d845ad7ed23eba9bdde260e635407190c2ce1bf7fd22088e41682ea07773ec65cae8946d2003f264d55961f96e0fc5da10eb96d3a348c1664e9644ce2108c311309f398ae8ea1b8200bfd490e5cb6e8c52c9e5d493cbabb163368f8351420451a631dbfa749829ee4cda49b77b5ed2d3dced5d0f2b7dd9ee76ba5465c84a17c23af040cd92b6b2a4ea48befbb5c729dcdad0a9c9668befe84074cc24f78899c1d947f8e7f94c7eda5325b8ed698df729e76febb98549ef3482ae942fb4f4a1c92d21836fa784728f0c5483aab2760a991b6b36e6b10c84f840a6433a6ecc31dee36e8f1c6158818bc89d2222dd6882c9bea84c2a8463737b8c2ebff9ca7c6d060fb963530bd14a2520dd9f2da63d38ec62765f787ba4ecad169369e97555f4f32b390f4cff376e30e2cb64f992f0f42a75cf0a559462e6c4e9cffe' | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /** | /** | ||||||
|  | |||||||
| @ -68,6 +68,11 @@ 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' | ||||||
|   }) |   }) | ||||||
| @ -80,7 +85,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 navigator.clipboard.write([ |       await clipboardObj.write([ | ||||||
|         new ClipboardItem({ |         new ClipboardItem({ | ||||||
|           [blob.type]: blob |           [blob.type]: blob | ||||||
|         }) |         }) | ||||||
| @ -99,13 +104,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) |       ctx.drawImage(img, 0, 0, canvas.width, canvas.height) | ||||||
| 
 | 
 | ||||||
|       canvas.toBlob( |       canvas.toBlob( | ||||||
|         (blob) => { |         (blob) => { | ||||||
|           const data = [new ClipboardItem({ [blob.type]: blob })] |           const data = [new ClipboardItem({ [blob.type]: blob })] | ||||||
| 
 | 
 | ||||||
|           navigator.clipboard |           clipboardObj | ||||||
|             .write(data) |             .write(data) | ||||||
|             .then(() => { |             .then(() => { | ||||||
|               callback() |               callback() | ||||||
|  | |||||||
| @ -53,7 +53,13 @@ request.interceptors.request.use((config) => { | |||||||
| }, errorHandler) | }, errorHandler) | ||||||
| 
 | 
 | ||||||
| // 响应拦截器
 | // 响应拦截器
 | ||||||
| request.interceptors.response.use((response) => response.data, errorHandler) | request.interceptors.response.use((response) => { | ||||||
|  |   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 请求 | ||||||
|  | |||||||
| @ -16,8 +16,6 @@ import avatarModule from '@/components/avatar-module/index.vue' | |||||||
| 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,6 +31,7 @@ const onSingleForward = () => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const onMultiDelete = () => { | const onMultiDelete = () => { | ||||||
|  |   if(dialogueStore.selectItems.length>0){ | ||||||
|     confirmBox({ |     confirmBox({ | ||||||
|     content:'确定删除聊天记录', |     content:'确定删除聊天记录', | ||||||
|     confirmText:'删除' |     confirmText:'删除' | ||||||
| @ -42,6 +43,10 @@ if (!msgIds.length) return | |||||||
| dialogueStore.ApiDeleteRecord(msgIds) | dialogueStore.ApiDeleteRecord(msgIds) | ||||||
|      |      | ||||||
|   }) |   }) | ||||||
|  |   }else{ | ||||||
|  |     window['$message'].warning('请选择聊天记录') | ||||||
|  |   } | ||||||
|  |   | ||||||
|   // 批量删除 |   // 批量删除 | ||||||
| 
 | 
 | ||||||
| } | } | ||||||
| @ -59,7 +64,6 @@ 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, | ||||||
|  | |||||||
| @ -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('复制成功') | ||||||
| @ -331,6 +331,7 @@ const onContextMenuHandle = (key: string) => { | |||||||
| 
 | 
 | ||||||
| const onRowClick = (item: ITalkRecord) => { | const onRowClick = (item: ITalkRecord) => { | ||||||
|   if (dialogueStore.isOpenMultiSelect) { |   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 { | ||||||
|  | |||||||
| @ -101,10 +101,13 @@ const onSendImageEvent = ({ data, callBack }) => { | |||||||
| 
 | 
 | ||||||
| // 发送视频消息 | // 发送视频消息 | ||||||
| const onSendVideoEvent = async ({ data }) => { | const onSendVideoEvent = async ({ data }) => { | ||||||
| 
 |  | ||||||
|    |  | ||||||
|   // 获取视频首帧作为封面图 |   // 获取视频首帧作为封面图 | ||||||
|   // let resp = await getVideoImage(data) |   let videoPreview = null | ||||||
|  |   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)}` | ||||||
| @ -123,7 +126,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: '',  |       url: videoPreview ? URL.createObjectURL(data) : '', // 使用本地视频URL作为预览 | ||||||
|       size: data.size, |       size: data.size, | ||||||
|       is_uploading: true, |       is_uploading: true, | ||||||
|       upload_id: uploadId, |       upload_id: uploadId, | ||||||
|  | |||||||
| @ -1,4 +1,5 @@ | |||||||
| 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 { | ||||||
| @ -9,16 +10,34 @@ interface IDropdown { | |||||||
|   item: any |   item: any | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const isRevoke = (uid: any, item: any): boolean => { | const isRevoke = (uid: number, item: any): boolean => { | ||||||
|   if (uid != item.user_id) { |   // 不是自己发的消息不能撤回
 | ||||||
|     return false |   if (uid !== item.user_id) { | ||||||
|  |     return false; | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   const datetime = item.created_at.replace(/-/g, '/') |   // 检查消息是否在撤回时间限制内(5分钟)
 | ||||||
| 
 |   const messageTime = dayjs(item.created_at); | ||||||
|   const time = new Date().getTime() - Date.parse(datetime) |   const now = dayjs(); | ||||||
| 
 |   const diffInMinutes = now.diff(messageTime, 'minute'); | ||||||
|   return Math.floor(time / 1000 / 60) <= 2 |   return diffInMinutes <= 5; | ||||||
|  | } | ||||||
|  | // 判断是否可以添加撤回选项的函数
 | ||||||
|  | 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() { | ||||||
| @ -33,6 +52,7 @@ export function useMenu() { | |||||||
|   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){ | ||||||
| @ -48,9 +68,10 @@ export function useMenu() { | |||||||
| 
 | 
 | ||||||
|     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 (isRevoke(uid, item)|| (dialogueStore.groupInfo as any).is_manager) { |     if (canAddRevokeOption(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' }) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -57,7 +57,6 @@ 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