Compare commits
	
		
			17 Commits
		
	
	
		
			acc8aeed2c
			...
			efd61b30f4
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | efd61b30f4 | ||
|  | 84096be043 | ||
|  | 4b7c69ea36 | ||
|  | f5ca14f746 | ||
| 576e950650 | |||
| aa3c7e1350 | |||
| 7a269b0215 | |||
| 999df303ea | |||
| 85de430b09 | |||
| 1850ffb727 | |||
|  | 6a94750c05 | ||
| f808c018fd | |||
|  | b101831c53 | ||
|  | 6791da7d8e | ||
|  | 4cf5e8ce18 | ||
| ace9b39fe3 | |||
| 0111453f06 | 
							
								
								
									
										9309
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										9309
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							
							
								
								
									
										7356
									
								
								pnpm-lock.yaml
									
									
									
									
									
								
							
							
						
						
									
										7356
									
								
								pnpm-lock.yaml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -99,3 +99,13 @@ export const ServeEmptyMessage = (data) => { | |||||||
| export const ServeMessageReadDetail = (data) => { | export const ServeMessageReadDetail = (data) => { | ||||||
|   return post('/api/v1/talk/my-records/read/condition', data) |   return post('/api/v1/talk/my-records/read/condition', data) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // 主动添加好友(单向好友)
 | ||||||
|  | export const ServeAddFriend = (data) => { | ||||||
|  |   return post('/api/v1/contact/friend/add', data) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 检测是否需要加好友
 | ||||||
|  | export const ServeCheckFriend = (data) => { | ||||||
|  |   return post('/api/v1/contact/friend/check', data) | ||||||
|  | } | ||||||
|  | |||||||
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/bofang.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/assets/image/bofang.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 2.0 KiB | 
							
								
								
									
										
											BIN
										
									
								
								src/assets/image/yuyin.png
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								src/assets/image/yuyin.png
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							| After Width: | Height: | Size: 3.3 KiB | 
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -72,8 +72,9 @@ const formatTime = (value: number = 0) => { | |||||||
| } | } | ||||||
| </script> | </script> | ||||||
| <template> | <template> | ||||||
|   <div class="pointer w-200px bg-#f5f5f5 rounded-10px px-11px"> |   <div class="pointer w-200px bg-#F3F4FD rounded-10px px-11px"> | ||||||
|     <div class="im-message-audio h-44px"> |     <div class="im-message-audio h-44px"> | ||||||
|  |       <aTrumpet :isPlay="false" color="black" :size="30"></aTrumpet> | ||||||
|       <audio |       <audio | ||||||
|         ref="audioRef" |         ref="audioRef" | ||||||
|         preload="auto" |         preload="auto" | ||||||
| @ -87,23 +88,37 @@ const formatTime = (value: number = 0) => { | |||||||
| 
 | 
 | ||||||
|       <div class="play"> |       <div class="play"> | ||||||
|         <div class="btn pointer" @click.stop="onPlay"> |         <div class="btn pointer" @click.stop="onPlay"> | ||||||
|         <n-icon :size="18" :component="state.isAudioPlay ? PauseOne : PlayOne" /> |           <!-- <n-icon :size="18" :component="state.isAudioPlay ? PauseOne : PlayOne" /> --> | ||||||
|  |           <img | ||||||
|  |             v-if="!state.isAudioPlay" | ||||||
|  |             src="@/assets/image/yuyin.png" | ||||||
|  |             class="w-[16px] h-[16px]" | ||||||
|  |             alt="" | ||||||
|  |           /> | ||||||
|  |           <img v-else src="@/assets/image/bofang.png" class="w-[16px] h-[16px]" alt="" /> | ||||||
|         </div> |         </div> | ||||||
|       </div> |       </div> | ||||||
|     <div class="desc"> |       <!-- <div class="desc"> | ||||||
|       <span class="line" v-for="i in 23" :key="i"></span> |       <span class="line" v-for="i in 23" :key="i"></span> | ||||||
|       <span |       <span | ||||||
|         class="indicator" |         class="indicator" | ||||||
|         :style="{ left: state.progress + '%' }" |         :style="{ left: state.progress + '%' }" | ||||||
|         v-show="state.progress > 0" |         v-show="state.progress > 0" | ||||||
|       ></span> |       ></span> | ||||||
|     </div> |     </div> --> | ||||||
|     <div class="time">{{ durationDesc }}</div> |       <!-- <div class="time">{{ durationDesc }}</div> --> | ||||||
|  |       <div>{{ durationDesc.split('"')[0] }}s</div> | ||||||
|     </div> |     </div> | ||||||
| 
 | 
 | ||||||
|     <transition name="expand"> |     <transition name="expand"> | ||||||
|     <div class="text-container py-12px border-t-2px border-t-solid border-t-#E0E0E4" v-if="data.is_convert_text===1"> |       <div | ||||||
|       <div class="flex justify-center items-center" v-if="data.is_convert_text===1&&!data.extra.content"> |         class="text-container py-12px border-t-1px border-t-solid border-t-#E2E2EB" | ||||||
|  |         v-if="data.is_convert_text === 1" | ||||||
|  |       > | ||||||
|  |         <div | ||||||
|  |           class="flex justify-center items-center" | ||||||
|  |           v-if="data.is_convert_text === 1 && !data.extra.content" | ||||||
|  |         > | ||||||
|           <n-spin :stroke-width="3" size="small" /> |           <n-spin :stroke-width="3" size="small" /> | ||||||
|         </div> |         </div> | ||||||
|         <transition name="fade"> |         <transition name="fade"> | ||||||
| @ -112,7 +127,6 @@ const formatTime = (value: number = 0) => { | |||||||
|       </div> |       </div> | ||||||
|     </transition> |     </transition> | ||||||
|   </div> |   </div> | ||||||
| 
 |  | ||||||
| </template> | </template> | ||||||
| <style lang="less" scoped> | <style lang="less" scoped> | ||||||
| .im-message-audio { | .im-message-audio { | ||||||
| @ -135,7 +149,7 @@ const formatTime = (value: number = 0) => { | |||||||
|     .btn { |     .btn { | ||||||
|       width: 26px; |       width: 26px; | ||||||
|       height: 26px; |       height: 26px; | ||||||
|       background-color: var(--audio-btn-bg-color); |       // background-color: var(--audio-btn-bg-color); | ||||||
|       border-radius: 50%; |       border-radius: 50%; | ||||||
|       color: rgb(24, 24, 24); |       color: rgb(24, 24, 24); | ||||||
|       display: flex; |       display: flex; | ||||||
|  | |||||||
| @ -8,9 +8,17 @@ import { useTalkStore } from '@/store' | |||||||
| import { useRouter } from 'vue-router' | import { useRouter } from 'vue-router' | ||||||
| import xNModal from '@/components/x-naive-ui/x-n-modal/index.vue' | import xNModal from '@/components/x-naive-ui/x-n-modal/index.vue' | ||||||
| import { NSkeleton } from 'naive-ui' | import { NSkeleton } from 'naive-ui' | ||||||
|  | import { ServeCheckFriend, ServeAddFriend } from '@/api/chat' | ||||||
|  | import { useUtil } from '@/hooks/useUtil' | ||||||
|  | 
 | ||||||
|  | const { useMessage } = useUtil() | ||||||
|  | 
 | ||||||
|  | const isFriend = ref(false) // 是否是我的好友 | ||||||
|  | const showBtn = ref(false) | ||||||
|  | 
 | ||||||
| const router = useRouter() | const router = useRouter() | ||||||
| const talkStore = useTalkStore() | const talkStore = useTalkStore() | ||||||
| const emit = defineEmits(['update:show', 'update:uid', 'updateRemark']) | const emit = defineEmits(['update:show', 'update:uid', 'updateRemark', 'update:send']) | ||||||
| const props = defineProps({ | const props = defineProps({ | ||||||
|   show: { |   show: { | ||||||
|     type: Boolean, |     type: Boolean, | ||||||
| @ -86,6 +94,7 @@ const onLoadData = () => { | |||||||
| const onToTalk = () => { | const onToTalk = () => { | ||||||
|   talkStore.toTalk(1, props.uid, router) |   talkStore.toTalk(1, props.uid, router) | ||||||
|   emit('update:show', false) |   emit('update:show', false) | ||||||
|  |   emit('update:send') | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // const onJoinContact = () => { | // const onJoinContact = () => { | ||||||
| @ -167,8 +176,27 @@ const onToTalk = () => { | |||||||
| //   emit('update:show', value) | //   emit('update:show', value) | ||||||
| // } | // } | ||||||
| 
 | 
 | ||||||
|  | // 添加好友 | ||||||
|  | const addFriend = () => { | ||||||
|  |   let params = { | ||||||
|  |     receiver_id: props.uid, //聊天的用户id | ||||||
|  |     talk_type: 1 | ||||||
|  |   } | ||||||
|  |   ServeAddFriend(params).then((res) => { | ||||||
|  |     if (res?.code === 200) { | ||||||
|  |       useMessage.success('添加成功') | ||||||
|  |       isFriend.value = !isFriend.value | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  | } | ||||||
| const onAfterEnter = () => { | const onAfterEnter = () => { | ||||||
|   onLoadData() |   onLoadData() | ||||||
|  |   ServeCheckFriend({ receiver_id: props.uid, talk_type: 1 }).then((res) => { | ||||||
|  |     if (res?.code === 200) { | ||||||
|  |       showBtn.value = true | ||||||
|  |       isFriend.value = res.data?.is_friend || false | ||||||
|  |     } | ||||||
|  |   }) | ||||||
| } | } | ||||||
| const onAfterLeave = () => { | const onAfterLeave = () => { | ||||||
|   // loading.value = true |   // loading.value = true | ||||||
| @ -188,10 +216,18 @@ const onAfterLeave = () => { | |||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <template> | <template> | ||||||
|   <x-n-modal content-style="padding:0;" :closable="false" class="w-311px min-h-445px" style="border-radius: 10px;overflow:hidden;" :show="show" :on-after-leave="onAfterLeave" :on-after-enter="onAfterEnter"> |   <x-n-modal | ||||||
|  |     content-style="padding:0;" | ||||||
|  |     :closable="false" | ||||||
|  |     class="w-311px min-h-445px" | ||||||
|  |     style="border-radius: 10px; overflow: hidden" | ||||||
|  |     :show="show" | ||||||
|  |     :on-after-leave="onAfterLeave" | ||||||
|  |     :on-after-enter="onAfterEnter" | ||||||
|  |   > | ||||||
|     <div class="section relative px-7px pt-82px pb-20px"> |     <div class="section relative px-7px pt-82px pb-20px"> | ||||||
|       <div class="absolute top-9px right-7px pointer z-10" @click="emit('update:show', false)"> |       <div class="absolute top-9px right-7px pointer z-10" @click="emit('update:show', false)"> | ||||||
|         <img class="w-20px h-20px" src="@/assets/image/close.png" alt=""> |         <img class="w-20px h-20px" src="@/assets/image/close.png" alt="" /> | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       <template v-if="loading"> |       <template v-if="loading"> | ||||||
| @ -200,27 +236,25 @@ const onAfterLeave = () => { | |||||||
|             <n-skeleton height="59px" width="59px" /> |             <n-skeleton height="59px" width="59px" /> | ||||||
|           </div> |           </div> | ||||||
|           <div class="w-full"> |           <div class="w-full"> | ||||||
|             <n-skeleton text style="width: 80%; margin-bottom: 5px;" /> |             <n-skeleton text style="width: 80%; margin-bottom: 5px" /> | ||||||
|             <n-skeleton text style="width: 60%;" /> |             <n-skeleton text style="width: 60%" /> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div class="bg-#fff rounded-4px mb-20px"> |         <div class="bg-#fff rounded-4px mb-20px"> | ||||||
|           <div class="flex px-15px py-9px" v-for="i in 6" :key="i"> |           <div class="flex px-15px py-9px" v-for="i in 6" :key="i"> | ||||||
|             <n-skeleton text style="width: 30%; margin-right: 10px;" /> |             <n-skeleton text style="width: 30%; margin-right: 10px" /> | ||||||
|             <n-skeleton text style="width: 60%;" /> |             <n-skeleton text style="width: 60%" /> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div> |         <div> | ||||||
|           <n-skeleton text style="width: 100%; height: 42px; border-radius: 4px;" /> |           <n-skeleton text style="width: 100%; height: 42px; border-radius: 4px" /> | ||||||
|         </div> |         </div> | ||||||
|       </template> |       </template> | ||||||
| 
 | 
 | ||||||
|       <template v-else> |       <template v-else> | ||||||
|         <div class="flex py-10px bg-#fff px-16px rounded-4px items-center mb-10px"> |         <div class="flex py-10px bg-#fff px-16px rounded-4px items-center mb-10px"> | ||||||
|           <div class="w-59px h-59px rounded-8px mr-12px overflow-hidden"> |           <div class="w-59px h-59px rounded-8px mr-12px overflow-hidden"> | ||||||
|            <n-image width="59" :src="userInfo.avatar" > |             <n-image width="59" :src="userInfo.avatar"> </n-image> | ||||||
| 
 |  | ||||||
|            </n-image> |  | ||||||
|           </div> |           </div> | ||||||
|           <div> |           <div> | ||||||
|             <div class="text-#000 text-16px mb-5px">{{ userInfo.nickname }}</div> |             <div class="text-#000 text-16px mb-5px">{{ userInfo.nickname }}</div> | ||||||
| @ -234,11 +268,15 @@ const onAfterLeave = () => { | |||||||
|           </div> |           </div> | ||||||
|           <div class="flex px-15px py-9px"> |           <div class="flex px-15px py-9px"> | ||||||
|             <div class="text-#000 text-12px w-84px">主管</div> |             <div class="text-#000 text-12px w-84px">主管</div> | ||||||
|             <div class="text-#747474 text-12px">{{ userInfo.leaders?.map(x=>x.user_name)?.join(',') }}</div> |             <div class="text-#747474 text-12px"> | ||||||
|  |               {{ userInfo.leaders?.map((x) => x.user_name)?.join(',') }} | ||||||
|  |             </div> | ||||||
|           </div> |           </div> | ||||||
|           <div class="flex px-15px py-9px"> |           <div class="flex px-15px py-9px"> | ||||||
|             <div class="text-#000 text-12px w-84px">部门</div> |             <div class="text-#000 text-12px w-84px">部门</div> | ||||||
|             <div class="text-#747474 text-12px">{{ userInfo.erp_dept_position?.map(x=>x.department_name)?.join(',') }}</div> |             <div class="text-#747474 text-12px"> | ||||||
|  |               {{ userInfo.erp_dept_position?.map((x) => x.department_name)?.join(',') }} | ||||||
|  |             </div> | ||||||
|           </div> |           </div> | ||||||
|           <div class="flex px-15px py-9px"> |           <div class="flex px-15px py-9px"> | ||||||
|             <div class="text-#000 text-12px w-84px">手机号</div> |             <div class="text-#000 text-12px w-84px">手机号</div> | ||||||
| @ -246,20 +284,40 @@ const onAfterLeave = () => { | |||||||
|           </div> |           </div> | ||||||
|           <div class="flex px-15px py-9px"> |           <div class="flex px-15px py-9px"> | ||||||
|             <div class="text-#000 text-12px w-84px">岗位</div> |             <div class="text-#000 text-12px w-84px">岗位</div> | ||||||
|             <div class="text-#747474 text-12px">{{ userInfo.erp_dept_position?.map(x=>x.position_name)?.join(',') }}</div> |             <div class="text-#747474 text-12px"> | ||||||
|  |               {{ userInfo.erp_dept_position?.map((x) => x.position_name)?.join(',') }} | ||||||
|  |             </div> | ||||||
|           </div> |           </div> | ||||||
|           <div class="flex px-15px py-9px"> |           <div class="flex px-15px py-9px"> | ||||||
|             <div class="text-#000 text-12px w-84px">入职日期</div> |             <div class="text-#000 text-12px w-84px">入职日期</div> | ||||||
|             <div class="text-#747474 text-12px">{{ userInfo.enter_date }}</div> |             <div class="text-#747474 text-12px">{{ userInfo.enter_date }}</div> | ||||||
|           </div> |           </div> | ||||||
|         </div> |         </div> | ||||||
|         <div> |         <div v-if="showBtn"> | ||||||
|           <n-button block color="#EEE9F8" text-color="#46299D"     @click="onToTalk"> |           <n-button block color="#EEE9F8" text-color="#46299D" @click="onToTalk" v-if="isFriend"> | ||||||
|             <div class="flex items-center justify-center py-11px"> |             <div class="flex items-center justify-center py-11px"> | ||||||
|                 <img class="w-19.8px h-20px mr-15px" src="@/assets/image/faxi@2x.png" alt=""> |               <img class="w-19.8px h-20px mr-15px" src="@/assets/image/faxi@2x.png" alt="" /> | ||||||
|               <span>发送消息</span> |               <span>发送消息</span> | ||||||
|             </div> |             </div> | ||||||
|           </n-button> |           </n-button> | ||||||
|  |           <n-button | ||||||
|  |             block | ||||||
|  |             type="success" | ||||||
|  |             color="#46299D" | ||||||
|  |             text-color="#ffffff" | ||||||
|  |             @click="addFriend" | ||||||
|  |             v-else | ||||||
|  |           > | ||||||
|  |             <div class="flex items-center justify-center py-11px"> | ||||||
|  |               <img | ||||||
|  |                 class="w-10px h-10px mr-15px" | ||||||
|  |                 src="@/assets/image/icon/close-btn-grey-line.png" | ||||||
|  |                 alt="" | ||||||
|  |                 style="transform: rotate(45deg)" | ||||||
|  |               /> | ||||||
|  |               <span>添加好友</span> | ||||||
|  |             </div> | ||||||
|  |           </n-button> | ||||||
|         </div> |         </div> | ||||||
|       </template> |       </template> | ||||||
|     </div> |     </div> | ||||||
|  | |||||||
| @ -12,6 +12,10 @@ import customModal from '@/components/common/customModal.vue' | |||||||
| import historyRecord from '@/components/search/searchByCondition.vue' | import historyRecord from '@/components/search/searchByCondition.vue' | ||||||
| import { ServeEditGroupNotice, ServeGetGroupNotices, ServeDeleteGroupNotice } from '@/api/group' | import { ServeEditGroupNotice, ServeGetGroupNotices, ServeDeleteGroupNotice } from '@/api/group' | ||||||
| import avatarModule from '@/components/avatar-module/index.vue' | import avatarModule from '@/components/avatar-module/index.vue' | ||||||
|  | import { ServeCheckFriend, ServeAddFriend } from '@/api/chat' | ||||||
|  | import { useUtil } from '@/hooks/useUtil' | ||||||
|  | 
 | ||||||
|  | const { useMessage } = useUtil() | ||||||
| 
 | 
 | ||||||
| const userStore = useUserStore() | const userStore = useUserStore() | ||||||
| const dialogueStore = useDialogueStore() | const dialogueStore = useDialogueStore() | ||||||
| @ -120,10 +124,32 @@ const events = { | |||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | const isFriend = ref(true) // 是否为好友 | ||||||
|  | // 添加好友 | ||||||
|  | const AddFriends = () => { | ||||||
|  |   let params = { | ||||||
|  |     receiver_id: talkParams.receiver_id, //聊天的用户id | ||||||
|  |     talk_type: 1 | ||||||
|  |   } | ||||||
|  |   ServeAddFriend(params).then((res) => { | ||||||
|  |     if (res?.code === 200) { | ||||||
|  |       isFriend.value = !isFriend.value | ||||||
|  |       useMessage.success('添加成功') | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  | } | ||||||
| watch( | watch( | ||||||
|   () => talkParams, |   () => talkParams, | ||||||
|   (newValue, oldValue) => { |   (newValue, oldValue) => { | ||||||
|     console.log(newValue) |     // 判断是否为好友 | ||||||
|  |     if (talkParams.type !== 2) { | ||||||
|  |       ServeCheckFriend({ receiver_id: newValue.receiver_id, talk_type: 1 }).then((res) => { | ||||||
|  |         if (res?.code === 200) { | ||||||
|  |           console.log(res, 'ress') | ||||||
|  |           isFriend.value = !res.data.is_friend | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |     } | ||||||
|   }, |   }, | ||||||
|   { deep: true, immediate: true } |   { deep: true, immediate: true } | ||||||
| ) | ) | ||||||
| @ -508,7 +534,27 @@ const clearSelectedDateTime = () => { | |||||||
|     </header> |     </header> | ||||||
| 
 | 
 | ||||||
|     <!-- 聊天区域 --> |     <!-- 聊天区域 --> | ||||||
|     <main class="el-main"> |     <main class="el-main relative"> | ||||||
|  |       <div | ||||||
|  |         class="p-[15px] pt-[10px] w-[100%] z-99 absolute" | ||||||
|  |         v-if="isFriend && talkParams.type !== 2" | ||||||
|  |       > | ||||||
|  |         <div | ||||||
|  |           class="bg-[#FFFFFF] w-[100%] p-[10px] text-[14px] flex justify-between" | ||||||
|  |           style="box-shadow: 0 2px 6px 1px rgba(0, 0, 0, 0.2) !important; border-radius: 5px" | ||||||
|  |         > | ||||||
|  |           对方还不是您的好友,请添加到通讯录中吧! | ||||||
|  |           <n-button | ||||||
|  |             @click="AddFriends" | ||||||
|  |             size="tiny" | ||||||
|  |             type="success" | ||||||
|  |             color="#46299D" | ||||||
|  |             text-color="#ffffff" | ||||||
|  |           > | ||||||
|  |             <span>添加好友</span> | ||||||
|  |           </n-button> | ||||||
|  |         </div> | ||||||
|  |       </div> | ||||||
|       <PanelContent |       <PanelContent | ||||||
|         :uid="talkParams.uid" |         :uid="talkParams.uid" | ||||||
|         :talk_type="talkParams.type" |         :talk_type="talkParams.type" | ||||||
| @ -593,7 +639,7 @@ const clearSelectedDateTime = () => { | |||||||
|   > |   > | ||||||
|     <template #content> |     <template #content> | ||||||
|       <div class="search-record-modal-searchArea"> |       <div class="search-record-modal-searchArea"> | ||||||
|         <n-card style="padding: 0 12px;"> |         <n-card style="padding: 0 12px"> | ||||||
|           <div class="search-record-input"> |           <div class="search-record-input"> | ||||||
|             <span class="search-record-input-title">搜索</span> |             <span class="search-record-input-title">搜索</span> | ||||||
|             <n-input |             <n-input | ||||||
| @ -623,7 +669,7 @@ const clearSelectedDateTime = () => { | |||||||
|               v-model:show="state.showDateConditionPopover" |               v-model:show="state.showDateConditionPopover" | ||||||
|               trigger="click" |               trigger="click" | ||||||
|               placement="bottom-start" |               placement="bottom-start" | ||||||
|               style="height: 312px; padding: 0;" |               style="height: 312px; padding: 0" | ||||||
|               @update:show="onDatePickShow" |               @update:show="onDatePickShow" | ||||||
|             > |             > | ||||||
|               <template #trigger> |               <template #trigger> | ||||||
| @ -651,7 +697,7 @@ const clearSelectedDateTime = () => { | |||||||
|               v-model:show="state.showMemberListByAlphabetPopover" |               v-model:show="state.showMemberListByAlphabetPopover" | ||||||
|               trigger="click" |               trigger="click" | ||||||
|               placement="bottom-start" |               placement="bottom-start" | ||||||
|               style="width: 290px; height: 505px; padding: 0;" |               style="width: 290px; height: 505px; padding: 0" | ||||||
|               v-if="talkParams.type === 2" |               v-if="talkParams.type === 2" | ||||||
|             > |             > | ||||||
|               <template #trigger> |               <template #trigger> | ||||||
| @ -660,10 +706,10 @@ const clearSelectedDateTime = () => { | |||||||
|               <div class="member-list-by-alphabet-container"> |               <div class="member-list-by-alphabet-container"> | ||||||
|                 <n-input |                 <n-input | ||||||
|                   placeholder="请输入群成员" |                   placeholder="请输入群成员" | ||||||
|                   style="margin: 0 0 17px;" |                   style="margin: 0 0 17px" | ||||||
|                   v-model:value="state.searchMemberByAlphabet" |                   v-model:value="state.searchMemberByAlphabet" | ||||||
|                 /> |                 /> | ||||||
|                 <n-scrollbar style="height: 430px;"> |                 <n-scrollbar style="height: 430px"> | ||||||
|                   <div |                   <div | ||||||
|                     class="member-list-by-alphabet" |                     class="member-list-by-alphabet" | ||||||
|                     v-for="(alphabetMembersItem, alphabetMembersIndex) in membersByAlphabet" |                     v-for="(alphabetMembersItem, alphabetMembersIndex) in membersByAlphabet" | ||||||
| @ -675,7 +721,8 @@ const clearSelectedDateTime = () => { | |||||||
|                     <div class="member-list-each-alphabet"> |                     <div class="member-list-each-alphabet"> | ||||||
|                       <div |                       <div | ||||||
|                         class="member-item-each-alphabet" |                         class="member-item-each-alphabet" | ||||||
|                         v-for="(memberItem, memberItemIndex) in (alphabetMembersItem as any).members" |                         v-for="(memberItem, memberItemIndex) in (alphabetMembersItem as any) | ||||||
|  |                           .members" | ||||||
|                         :key="memberItemIndex" |                         :key="memberItemIndex" | ||||||
|                         @click="handleMemberItemClick(memberItem)" |                         @click="handleMemberItemClick(memberItem)" | ||||||
|                       > |                       > | ||||||
| @ -776,7 +823,7 @@ const clearSelectedDateTime = () => { | |||||||
|               }" |               }" | ||||||
|             ></avatarModule> |             ></avatarModule> | ||||||
|             <div class="group-notice-header-userInfo"> |             <div class="group-notice-header-userInfo"> | ||||||
|               <span style="color: #1b1b1b; font-weight: 600; line-height: 20px;">{{ |               <span style="color: #1b1b1b; font-weight: 600; line-height: 20px">{{ | ||||||
|                 state.groupNoticeInfo.updater_name |                 state.groupNoticeInfo.updater_name | ||||||
|               }}</span> |               }}</span> | ||||||
|               <span>{{ state.groupNoticeInfo.updated_at }}</span> |               <span>{{ state.groupNoticeInfo.updated_at }}</span> | ||||||
|  | |||||||
| @ -1,4 +1,4 @@ | |||||||
| <script lang="ts" setup> | <script lang="tsx" setup> | ||||||
| import { | import { | ||||||
|   computed, |   computed, | ||||||
|   ref, |   ref, | ||||||
| @ -23,10 +23,10 @@ import { | |||||||
|   NButton, |   NButton, | ||||||
|   NPagination |   NPagination | ||||||
| } from 'naive-ui' | } from 'naive-ui' | ||||||
| import { Search, Plus, Right } from '@icon-park/vue-next' | import { Search, Plus, Right, AddOne, PeoplePlusOne } from '@icon-park/vue-next' | ||||||
| import TalkItem from './TalkItem.vue' | import TalkItem from './TalkItem.vue' | ||||||
| import Skeleton from './Skeleton.vue' | import Skeleton from './Skeleton.vue' | ||||||
| import { ServeClearTalkUnreadNum } from '@/api/chat' | import { ServeClearTalkUnreadNum, ServeAddFriend } from '@/api/chat' | ||||||
| import GroupLaunch from '@/components/group/GroupLaunch.vue' | import GroupLaunch from '@/components/group/GroupLaunch.vue' | ||||||
| import { getCacheIndexName } from '@/utils/talk' | import { getCacheIndexName } from '@/utils/talk' | ||||||
| import { ISession } from '@/types/chat' | import { ISession } from '@/types/chat' | ||||||
| @ -41,6 +41,11 @@ import { ServeSeachQueryAll, ServeQueryTalkRecord, ServeUserGroupChatList } from | |||||||
| import { getUserInfoByERPUserId } from '@/api/user' | import { getUserInfoByERPUserId } from '@/api/user' | ||||||
| import HighlightText from '@/components/search/highLightText.vue' | import HighlightText from '@/components/search/highLightText.vue' | ||||||
| import { useRouter } from 'vue-router' | import { useRouter } from 'vue-router' | ||||||
|  | import icon from '@/assets/image/chatList/addressBook.png' | ||||||
|  | import { useUtil } from '@/hooks/useUtil' | ||||||
|  | import UserCardModal from '@/components/user/UserCardModal.vue' | ||||||
|  | 
 | ||||||
|  | const { useMessage } = useUtil() | ||||||
| const router = useRouter() | const router = useRouter() | ||||||
| 
 | 
 | ||||||
| const currentInstance = getCurrentInstance() | const currentInstance = getCurrentInstance() | ||||||
| @ -60,7 +65,43 @@ const isShowGroup = ref(false) | |||||||
| const searchKeyword = ref('') | const searchKeyword = ref('') | ||||||
| const topItems = computed((): ISession[] => talkStore.topItems) | const topItems = computed((): ISession[] => talkStore.topItems) | ||||||
| const unreadNum = computed(() => talkStore.talkUnreadNum) | const unreadNum = computed(() => talkStore.talkUnreadNum) | ||||||
|  | // 是否删除好友弹框 | ||||||
|  | const handleConfirmDel = (row) => { | ||||||
|  |   window['$dialog'].create({ | ||||||
|  |     title: '温馨提示', | ||||||
|  |     content: '是否删除该好友?', | ||||||
|  |     positiveText: '确定', | ||||||
|  |     negativeText: '取消', | ||||||
|  |     onPositiveClick: () => { | ||||||
|  |       console.log('确定') | ||||||
|  |       let params = { | ||||||
|  |         receiver_id: row.id, //聊天的用户id | ||||||
|  |         talk_type: 1 | ||||||
|  |       } | ||||||
|  |       let url = '/api/v1/contact/friend/delete' | ||||||
|  |       $request.HTTP.components.postDataByParams(url, params).then((res) => { | ||||||
|  |         // console.log(res) | ||||||
|  |         if (res?.code === 200) { | ||||||
|  |           useMessage.success('删除成功') | ||||||
|  |           getMyFriends() | ||||||
|  |         } | ||||||
|  |       }) | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
|  | const option = ref([ | ||||||
|  |   { | ||||||
|  |     label: '添加好友', | ||||||
|  |     key: 'addFriend', | ||||||
|  |     icon: () => <n-icon size="20" component={PeoplePlusOne} /> | ||||||
|  |   }, | ||||||
|  |   { | ||||||
|  |     label: '通讯录', | ||||||
|  |     key: 'addressBook', | ||||||
|  |     icon: () => <img style="width: 19px; height: 20px; cursor: pointer" src={icon} /> | ||||||
|  |   } | ||||||
|  | ]) | ||||||
| //自定义搜索 | //自定义搜索 | ||||||
| const renderChatAppSearch = () => { | const renderChatAppSearch = () => { | ||||||
|   return h( |   return h( | ||||||
| @ -145,7 +186,13 @@ const renderChatAppSearch = () => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const state = reactive({ | const state = reactive({ | ||||||
|  |   userInfo: { | ||||||
|  |     isShowUserCardModal: false, | ||||||
|  |     user_id: NaN, | ||||||
|  |     erp_user_id: NaN | ||||||
|  |   }, | ||||||
|   isShowAddressBookModal: false, // 是否显示通讯录模态框 |   isShowAddressBookModal: false, // 是否显示通讯录模态框 | ||||||
|  |   isShowAddFriendModal: false, // 是否显示添加好友模态框 | ||||||
|   customModalStyle: { |   customModalStyle: { | ||||||
|     width: '1288px', |     width: '1288px', | ||||||
|     height: '846px', |     height: '846px', | ||||||
| @ -166,7 +213,25 @@ const state = reactive({ | |||||||
|       type: 'input', |       type: 'input', | ||||||
|       valueType: 'string' |       valueType: 'string' | ||||||
|     } |     } | ||||||
|   ], // 群聊列表搜索配置 |   ], | ||||||
|  |   // 我的好友搜索配置 | ||||||
|  |   myFriendSearchConfig: [ | ||||||
|  |     { | ||||||
|  |       label: '好友名称', | ||||||
|  |       key: 'myFriendname', | ||||||
|  |       type: 'input', | ||||||
|  |       valueType: 'string' | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  |   addFriendSearchConfig: [ | ||||||
|  |     { | ||||||
|  |       label: '姓名', | ||||||
|  |       key: 'friendName', | ||||||
|  |       type: 'input', | ||||||
|  |       valueType: 'string' | ||||||
|  |     } | ||||||
|  |   ], // 添加好友搜索配置 | ||||||
|  |   // 群聊列表搜索配置 | ||||||
|   treeData: [], |   treeData: [], | ||||||
|   expandedKeys: [], |   expandedKeys: [], | ||||||
|   clickKey: 3, |   clickKey: 3, | ||||||
| @ -175,29 +240,29 @@ const state = reactive({ | |||||||
|   addressBookColumns: [ |   addressBookColumns: [ | ||||||
|     { |     { | ||||||
|       title: '姓名 【工号】', |       title: '姓名 【工号】', | ||||||
|       field: 'nickName', |       field: 'nickname', | ||||||
|       width: 200, |       width: 200, | ||||||
|       ellipsis: { |       ellipsis: { | ||||||
|         tooltip: true |         tooltip: true | ||||||
|       }, |       }, | ||||||
|       render(row, index) { |       render(row, index) { | ||||||
|         return row.nickName + '【' + row.jobNum + '】' |         return row.nickname + '【' + row.job_num + '】' | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
|       title: '岗位名称', |       title: '岗位名称', | ||||||
|       field: 'positionName', |       field: 'user_position', | ||||||
|       width: 400, |       width: 400, | ||||||
|       ellipsis: { |       ellipsis: { | ||||||
|         tooltip: true |         tooltip: true | ||||||
|       }, |       }, | ||||||
|       render(row, index) { |       render(row, index) { | ||||||
|         let positionNames = Array.isArray(row.depPositions) |         // let positionNames = Array.isArray(row.user_position) | ||||||
|           ? row.depPositions.flatMap((dep) => |         //   ? row.depPositions.flatMap((dep) => | ||||||
|               Array.isArray(dep.positions) ? dep.positions.map((pos) => pos.name) : [] |         //       Array.isArray(dep.positions) ? dep.positions.map((pos) => pos.name) : [] | ||||||
|             ) |         //     ) | ||||||
|           : [] |         //   : [] | ||||||
|         return positionNames.join(' , ') |         return row.user_position.map((item) => item.position_name).join(' , ') | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     { |     { | ||||||
| @ -270,10 +335,126 @@ const state = reactive({ | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   ], // 群聊列表表格列 |   ], // 群聊列表表格列 | ||||||
|  |   myFriendListColumns: [ | ||||||
|  |     { | ||||||
|  |       title: '姓名 【工号】', | ||||||
|  |       field: 'nickname', | ||||||
|  |       width: 200, | ||||||
|  |       ellipsis: { | ||||||
|  |         tooltip: true | ||||||
|  |       }, | ||||||
|  |       render(row, index) { | ||||||
|  |         return row.nickname + '【' + row.job_num + '】' | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '岗位名称', | ||||||
|  |       field: 'user_position', | ||||||
|  |       width: 400, | ||||||
|  |       ellipsis: { | ||||||
|  |         tooltip: true | ||||||
|  |       }, | ||||||
|  |       render(row, index) { | ||||||
|  |         // let positionNames = Array.isArray(row.user_position) | ||||||
|  |         //   ? row.depPositions.flatMap((dep) => | ||||||
|  |         //       Array.isArray(dep.positions) ? dep.positions.map((pos) => pos.name) : [] | ||||||
|  |         //     ) | ||||||
|  |         //   : [] | ||||||
|  |         return row.user_position.map((item) => item.position_name).join(' , ') | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '操作', | ||||||
|  |       field: 'action', | ||||||
|  |       width: 180, | ||||||
|  |       align: 'center', | ||||||
|  |       fixed: 'right', | ||||||
|  |       render(row, index) { | ||||||
|  |         return [ | ||||||
|  |           h( | ||||||
|  |             NButton, | ||||||
|  |             { | ||||||
|  |               size: 'small', | ||||||
|  |               text: true, | ||||||
|  |               color: '#46299d', | ||||||
|  |               onClick: () => handleEnterChat(row) | ||||||
|  |             }, | ||||||
|  |             { default: () => '进入聊天' } | ||||||
|  |           ), | ||||||
|  |           h( | ||||||
|  |             NButton, | ||||||
|  |             { | ||||||
|  |               size: 'small', | ||||||
|  |               text: true, | ||||||
|  |               color: '#46299d', | ||||||
|  |               class: 'pl-[10px]', | ||||||
|  |               onClick: () => handleConfirmDel(row) | ||||||
|  |             }, | ||||||
|  |             { default: () => '删除好友' } | ||||||
|  |           ) | ||||||
|  |         ] | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   ], // 我的好友表格列 | ||||||
|  |   addFriendListColumns: [ | ||||||
|  |     { | ||||||
|  |       title: '姓名 【工号】', | ||||||
|  |       field: 'nickname', | ||||||
|  |       width: 200, | ||||||
|  |       ellipsis: { | ||||||
|  |         tooltip: true | ||||||
|  |       }, | ||||||
|  |       render(row, index) { | ||||||
|  |         return row.nickname + '【' + row.job_num + '】' | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '岗位名称', | ||||||
|  |       field: 'user_position', | ||||||
|  |       width: 400, | ||||||
|  |       ellipsis: { | ||||||
|  |         tooltip: true | ||||||
|  |       }, | ||||||
|  |       render(row, index) { | ||||||
|  |         // let positionNames = Array.isArray(row.user_position) | ||||||
|  |         //   ? row.depPositions.flatMap((dep) => | ||||||
|  |         //       Array.isArray(dep.positions) ? dep.positions.map((pos) => pos.name) : [] | ||||||
|  |         //     ) | ||||||
|  |         //   : [] | ||||||
|  |         return row.user_position.map((item) => item.position_name).join(' , ') | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     { | ||||||
|  |       title: '操作', | ||||||
|  |       field: 'action', | ||||||
|  |       width: 180, | ||||||
|  |       align: 'center', | ||||||
|  |       fixed: 'right', | ||||||
|  |       render(row, index) { | ||||||
|  |         return h( | ||||||
|  |           NButton, | ||||||
|  |           { | ||||||
|  |             size: 'small', | ||||||
|  |             text: true, | ||||||
|  |             color: '#46299d', | ||||||
|  |             onClick: () => { | ||||||
|  |               state.userInfo.user_id = row.id | ||||||
|  |               state.userInfo.erp_user_id = row.erp_user_id | ||||||
|  |               state.userInfo.isShowUserCardModal = true | ||||||
|  |             } | ||||||
|  |           }, | ||||||
|  |           { default: () => '查看' } | ||||||
|  |         ) | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  |   ], // 添加表格列 | ||||||
|   addressBookData: [], // 通讯录表格数据 |   addressBookData: [], // 通讯录表格数据 | ||||||
|  |   company_name: '', // 当前公司别 | ||||||
|   groupChatListData: [], // 群聊列表表格数据 |   groupChatListData: [], // 群聊列表表格数据 | ||||||
|  |   myFriendListData: [], // 我的好友表格数据 | ||||||
|  |   addFriendList: [], // 搜索出来的可添加好友 | ||||||
|   addressBookTableHeight: 524, // 通讯录表格高度 |   addressBookTableHeight: 524, // 通讯录表格高度 | ||||||
|   addressBookTableWidth: 800, // 通讯录表格宽度 |   addressBookTableWidth: 1142, // 通讯录表格宽度 | ||||||
|   addressBookPage: 1, // 通讯录表格页码 |   addressBookPage: 1, // 通讯录表格页码 | ||||||
|   addressBookPageSize: 10, // 通讯录表格每页条数 |   addressBookPageSize: 10, // 通讯录表格每页条数 | ||||||
|   addressBookTotal: 0, // 通讯录表格总条数 |   addressBookTotal: 0, // 通讯录表格总条数 | ||||||
| @ -283,6 +464,10 @@ const state = reactive({ | |||||||
|   groupChatListPageSize: 10, // 群聊列表表格每页条数 |   groupChatListPageSize: 10, // 群聊列表表格每页条数 | ||||||
|   groupChatListTotal: 0, // 群聊列表表格总条数 |   groupChatListTotal: 0, // 群聊列表表格总条数 | ||||||
|   groupChatListSearchGroupName: '', // 群聊列表搜索条件-群聊名称 |   groupChatListSearchGroupName: '', // 群聊列表搜索条件-群聊名称 | ||||||
|  |   myFriendListPage: 1, // 我的好友表格页码 | ||||||
|  |   myFriendListPageSize: 10, // 我的好友表格每页条数 | ||||||
|  |   myFriendListTotal: 0, // 我的好友表格总条数 | ||||||
|  |   myFriendListSearchName: '', // 我的好友搜索条件-好友名称 | ||||||
|   chatSearchOptions: [ |   chatSearchOptions: [ | ||||||
|     { |     { | ||||||
|       key: 'chatSearch', |       key: 'chatSearch', | ||||||
| @ -342,7 +527,7 @@ watch( | |||||||
|       state.addressBookTableWidth = 1142 |       state.addressBookTableWidth = 1142 | ||||||
|       state.addressBookPage = 1 |       state.addressBookPage = 1 | ||||||
|     } else { |     } else { | ||||||
|       state.addressBookTableWidth = 800 |       // state.addressBookTableWidth = 800 | ||||||
|       state.clickKey = 3 |       state.clickKey = 3 | ||||||
|       state.treeRefreshCount++ |       state.treeRefreshCount++ | ||||||
|       state.addressBookPage = 1 |       state.addressBookPage = 1 | ||||||
| @ -350,6 +535,17 @@ watch( | |||||||
|     getDepPoisUser() |     getDepPoisUser() | ||||||
|   } |   } | ||||||
| ) | ) | ||||||
|  | watch( | ||||||
|  |   () => state.myFriendListSearchName, | ||||||
|  |   (newValue, oldValue) => { | ||||||
|  |     if (newValue) { | ||||||
|  |       state.myFriendListPage = 1 | ||||||
|  |     } else { | ||||||
|  |       state.myFriendListPage = 1 | ||||||
|  |     } | ||||||
|  |     getMyFriends() | ||||||
|  |   } | ||||||
|  | ) | ||||||
| watch( | watch( | ||||||
|   () => state.groupChatListSearchGroupName, |   () => state.groupChatListSearchGroupName, | ||||||
|   (newValue, oldValue) => { |   (newValue, oldValue) => { | ||||||
| @ -450,7 +646,9 @@ const onInitialize = () => { | |||||||
| onBeforeRouteUpdate(onInitialize) | onBeforeRouteUpdate(onInitialize) | ||||||
| 
 | 
 | ||||||
| onBeforeMount(() => { | onBeforeMount(() => { | ||||||
|   getTreeData() |   // getTreeData() | ||||||
|  |   getDepPoisUser() | ||||||
|  |   getMyFriends() | ||||||
|   getUserGroupChatList() |   getUserGroupChatList() | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| @ -462,11 +660,24 @@ onMounted(() => { | |||||||
| const showAddressBookModal = () => { | const showAddressBookModal = () => { | ||||||
|   state.isShowAddressBookModal = true |   state.isShowAddressBookModal = true | ||||||
| } | } | ||||||
|  | // 点击显示添加好友模态框 | ||||||
|  | const showAddFriendModal = () => { | ||||||
|  |   state.isShowAddFriendModal = true | ||||||
|  | } | ||||||
|  | const handleSelect = (key: string | number) => { | ||||||
|  |   if (key === 'addressBook') return showAddressBookModal() | ||||||
|  |   showAddFriendModal() | ||||||
|  | } | ||||||
| // 点击关闭通讯录模态框 | // 点击关闭通讯录模态框 | ||||||
| const closeAddressBookModal = () => { | const closeAddressBookModal = () => { | ||||||
|   state.isShowAddressBookModal = false |   state.isShowAddressBookModal = false | ||||||
|   resetAddressBookModal() |   resetAddressBookModal() | ||||||
| } | } | ||||||
|  | // 点击关闭添加好友模态框 | ||||||
|  | const closeAddFriendModal = () => { | ||||||
|  |   state.isShowAddFriendModal = false | ||||||
|  |   resetAddressBookModal() | ||||||
|  | } | ||||||
| const handleTreeClick = ({ selectedKey, tree }) => { | const handleTreeClick = ({ selectedKey, tree }) => { | ||||||
|   // console.log(tree) |   // console.log(tree) | ||||||
|   state.clickKey = tree.key |   state.clickKey = tree.key | ||||||
| @ -488,7 +699,7 @@ const calcTreeData = (data) => { | |||||||
|     delete item.sons |     delete item.sons | ||||||
|   } |   } | ||||||
| } | } | ||||||
| // 获取组织树数据 | // 获取组织树数据-已隐藏 | ||||||
| const getTreeData = () => { | const getTreeData = () => { | ||||||
|   let url = '/department/v2/tree/filter' |   let url = '/department/v2/tree/filter' | ||||||
|   let params = {} |   let params = {} | ||||||
| @ -512,30 +723,84 @@ const getTreeData = () => { | |||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  | // 获取我的好友 | ||||||
|  | const getMyFriends = () => { | ||||||
|  |   // myFriendListPage: 1, // 我的好友表格页码 | ||||||
|  |   // myFriendListPageSize: 10, // 我的好友表格每页条数 | ||||||
|  |   // myFriendListTotal: 0, // 我的好友表格总条数 | ||||||
|  |   // myFriendListSearchName: '', // 我的好友搜索条件-好友名称 | ||||||
|  |   let params = { | ||||||
|  |     type: 'myFriends', //查我得好友的时候写死myFriends | ||||||
|  |     page: state.myFriendListPage, | ||||||
|  |     page_size: state.myFriendListPageSize, | ||||||
|  |     name: state.myFriendListSearchName | ||||||
|  |   } | ||||||
|  |   let url = '/api/v1/contact/friend/list' | ||||||
|  |   $request.HTTP.components.postDataByParams(url, params).then((res) => { | ||||||
|  |     // console.log(res) | ||||||
|  |     if (res.code === 200 && Array.isArray(res.data.user_list)) { | ||||||
|  |       state.myFriendListData = res.data.user_list || [] | ||||||
|  |       state.company_name = res.data.company_name || '' | ||||||
|  |       state.myFriendListTotal = res.data.count | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  | } | ||||||
| // 获取部门下的人员 | // 获取部门下的人员 | ||||||
| const getDepPoisUser = () => { | const getDepPoisUser = () => { | ||||||
|   let url = '/user/v2/list' |   // let url = '/user/v2/list' | ||||||
|  |   let url = '/api/v1/contact/friend/list' | ||||||
|  |   // let params = { | ||||||
|  |   //   departmentId: state.addressBookSearchNickName ? undefined : state.clickKey, | ||||||
|  |   //   page: state.addressBookPage, | ||||||
|  |   //   pageSize: state.addressBookPageSize, | ||||||
|  |   //   status: 'notactive', | ||||||
|  |   //   nickName: state.addressBookSearchNickName | ||||||
|  |   // } | ||||||
|  |   // $request.HTTP.components.postDataByParams(url, params).then((res) => { | ||||||
|  |   //   // console.log(res) | ||||||
|  |   //   if (res.status === 0 && Array.isArray(res.data.data)) { | ||||||
|  |   //     state.addressBookData = res.data.data || [] | ||||||
|  |   //     state.addressBookTotal = res.data.count | ||||||
|  |   //   } | ||||||
|  |   // }) | ||||||
|   let params = { |   let params = { | ||||||
|     departmentId: state.addressBookSearchNickName ? undefined : state.clickKey, |     type: 'addressBook', //查我的通讯录的时候写死addressBook | ||||||
|     page: state.addressBookPage, |     page: state.addressBookPage, | ||||||
|     pageSize: state.addressBookPageSize, |     page_size: state.addressBookPageSize, | ||||||
|     status: 'notactive', |     name: state.addressBookSearchNickName | ||||||
|     nickName: state.addressBookSearchNickName |  | ||||||
|   } |   } | ||||||
|   $request.HTTP.components.postDataByParams(url, params).then((res) => { |   $request.HTTP.components.postDataByParams(url, params).then((res) => { | ||||||
|     // console.log(res) |     // console.log(res) | ||||||
|     if (res.status === 0 && Array.isArray(res.data.data)) { |     if (res.code === 200 && Array.isArray(res.data.user_list)) { | ||||||
|       state.addressBookData = res.data.data || [] |       state.addressBookData = res.data.user_list || [] | ||||||
|  |       state.company_name = res.data.company_name || '' | ||||||
|       state.addressBookTotal = res.data.count |       state.addressBookTotal = res.data.count | ||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // 搜索可添加好友 | ||||||
|  | const AddFriends = (row) => { | ||||||
|  |   let params = { | ||||||
|  |     receiver_id: row.erp_user_id, //聊天的用户id | ||||||
|  |     talk_type: 1 | ||||||
|  |   } | ||||||
|  |   ServeAddFriend(params).then((res) => { | ||||||
|  |     if (res?.code === 200) { | ||||||
|  |       useMessage.success('添加成功') | ||||||
|  |     } | ||||||
|  |   }) | ||||||
|  | } | ||||||
|  | 
 | ||||||
| //点击进入对应的聊天 | //点击进入对应的聊天 | ||||||
| const handleEnterChat = async (row) => { | const handleEnterChat = async (row) => { | ||||||
|   console.log(row) |   console.log(row) | ||||||
|   if (state.addressBookCurrentTab === 'employeeAddressBook') { |   if ( | ||||||
|  |     state.addressBookCurrentTab === 'employeeAddressBook' || | ||||||
|  |     state.addressBookCurrentTab === 'myFriend' | ||||||
|  |   ) { | ||||||
|     //员工通讯录,聊天类型一定为单聊 |     //员工通讯录,聊天类型一定为单聊 | ||||||
|     await getUserInfoByERPUserId({ erp_user_id: row.ID }).then((res) => { |     await getUserInfoByERPUserId({ erp_user_id: row.erp_user_id }).then((res) => { | ||||||
|       // console.log(res) |       // console.log(res) | ||||||
|       if (res.code === 200) { |       if (res.code === 200) { | ||||||
|         let sysUserInfo = res.data |         let sysUserInfo = res.data | ||||||
| @ -555,7 +820,7 @@ const resetAddressBookModal = () => { | |||||||
|     state.addressBookCurrentTab = 'employeeAddressBook' |     state.addressBookCurrentTab = 'employeeAddressBook' | ||||||
|     state.addressBookSearchNickName = '' |     state.addressBookSearchNickName = '' | ||||||
|     state.groupChatListSearchGroupName = '' |     state.groupChatListSearchGroupName = '' | ||||||
|     state.addressBookTableWidth = 800 |     state.addressBookTableWidth = 1142 | ||||||
|     state.clickKey = 3 |     state.clickKey = 3 | ||||||
|     state.treeRefreshCount++ |     state.treeRefreshCount++ | ||||||
|     state.addressBookPage = 1 |     state.addressBookPage = 1 | ||||||
| @ -563,7 +828,9 @@ const resetAddressBookModal = () => { | |||||||
|     state.groupChatListPage = 1 |     state.groupChatListPage = 1 | ||||||
|     state.groupChatListPageSize = 10 |     state.groupChatListPageSize = 10 | ||||||
|     getDepPoisUser() |     getDepPoisUser() | ||||||
|  |     getMyFriends() | ||||||
|     getUserGroupChatList() |     getUserGroupChatList() | ||||||
|  |     state.addFriendList = [] | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| //处理页数变化 | //处理页数变化 | ||||||
| @ -599,6 +866,32 @@ const changeGroupChatListSearch = (value) => { | |||||||
|     state.groupChatListSearchGroupName = value.groupName |     state.groupChatListSearchGroupName = value.groupName | ||||||
|   } |   } | ||||||
| } | } | ||||||
|  | //处理我的好友搜索 | ||||||
|  | const changeMyFriendListSearch = (value) => { | ||||||
|  |   console.log(value, 'value') | ||||||
|  |   if (!value.myFriendname?.trim()) { | ||||||
|  |     state.myFriendListSearchName = '' | ||||||
|  |   } else { | ||||||
|  |     state.myFriendListSearchName = value.myFriendname | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | const changeAddFriendSearch = (value) => { | ||||||
|  |   console.log(11) | ||||||
|  |   if (value.friendName?.trim()) { | ||||||
|  |     state.myFriendListSearchName = '' | ||||||
|  |     // 搜索好友 | ||||||
|  |     let params = { | ||||||
|  |       name: value.friendName | ||||||
|  |     } | ||||||
|  |     let url = '/api/v1/contact/friend/search' | ||||||
|  |     $request.HTTP.components.postDataByParams(url, params).then((res) => { | ||||||
|  |       // console.log(res) | ||||||
|  |       if (res.code === 200) { | ||||||
|  |         state.addFriendList = res.data?.user_list || [] | ||||||
|  |       } | ||||||
|  |     }) | ||||||
|  |   } | ||||||
|  | } | ||||||
| //获取用户所在群聊列表 | //获取用户所在群聊列表 | ||||||
| const getUserGroupChatList = () => { | const getUserGroupChatList = () => { | ||||||
|   let params = { |   let params = { | ||||||
| @ -628,6 +921,19 @@ const handleGroupChatListPaginationSize = (value) => { | |||||||
|   state.groupChatListPage = 1 |   state.groupChatListPage = 1 | ||||||
|   getUserGroupChatList() |   getUserGroupChatList() | ||||||
| } | } | ||||||
|  | //处理我的好友页数变化 | ||||||
|  | const handleMyFriendListPagination = (value) => { | ||||||
|  |   console.log(value, 'value') | ||||||
|  |   state.myFriendListPage = value | ||||||
|  |   getMyFriends() | ||||||
|  | } | ||||||
|  | //处理我的好友每页条数变化 | ||||||
|  | const handleMyFriendListPaginationSize = (value) => { | ||||||
|  |   console.log(value, 'value') | ||||||
|  |   state.myFriendListPageSize = value | ||||||
|  |   state.myFriendListPage = 1 | ||||||
|  |   getMyFriends() | ||||||
|  | } | ||||||
| //处理搜索聊天记录点击 | //处理搜索聊天记录点击 | ||||||
| const handleClickSearchItem = (searchText, searchResultKey, talk_type, receiver_id, res) => { | const handleClickSearchItem = (searchText, searchResultKey, talk_type, receiver_id, res) => { | ||||||
|   console.log(searchText, searchResultKey, talk_type, receiver_id) |   console.log(searchText, searchResultKey, talk_type, receiver_id) | ||||||
| @ -766,7 +1072,7 @@ const handleEnterSearchResultChat = () => { | |||||||
|       <n-dropdown |       <n-dropdown | ||||||
|         trigger="click" |         trigger="click" | ||||||
|         :options="state.chatSearchOptions" |         :options="state.chatSearchOptions" | ||||||
|         style="width: 248px; height: 677px;" |         style="width: 248px; height: 677px" | ||||||
|         :show="state.showSearchDropdown" |         :show="state.showSearchDropdown" | ||||||
|         @clickoutside="state.showSearchDropdown = false" |         @clickoutside="state.showSearchDropdown = false" | ||||||
|       > |       > | ||||||
| @ -774,7 +1080,7 @@ const handleEnterSearchResultChat = () => { | |||||||
|           placeholder="搜索好友 / 群聊" |           placeholder="搜索好友 / 群聊" | ||||||
|           v-model:value.trim="searchKeyword" |           v-model:value.trim="searchKeyword" | ||||||
|           clearable |           clearable | ||||||
|           style="width: 78%;" |           style="width: 78%" | ||||||
|           @click="state.showSearchDropdown = true" |           @click="state.showSearchDropdown = true" | ||||||
|         > |         > | ||||||
|           <!-- <template #prefix> |           <!-- <template #prefix> | ||||||
| @ -787,12 +1093,15 @@ const handleEnterSearchResultChat = () => { | |||||||
|           <n-icon :component="Plus" /> |           <n-icon :component="Plus" /> | ||||||
|         </template> |         </template> | ||||||
|       </n-button> --> |       </n-button> --> | ||||||
|       <img |       <!-- <img | ||||||
|         style="width: 19px; height: 20px; cursor: pointer;" |         style="width: 19px; height: 20px; cursor: pointer" | ||||||
|         src="@/assets/image/chatList/addressBook.png" |         src="@/assets/image/chatList/addressBook.png" | ||||||
|         alt="" |         alt="" | ||||||
|         @click="showAddressBookModal" |         @click="showAddressBookModal" | ||||||
|       /> |       /> --> | ||||||
|  |       <n-dropdown :options="option" @select="handleSelect"> | ||||||
|  |         <n-button> <n-icon :component="AddOne" /></n-button> | ||||||
|  |       </n-dropdown> | ||||||
|     </header> |     </header> | ||||||
| 
 | 
 | ||||||
|     <!-- 置顶栏目 --> |     <!-- 置顶栏目 --> | ||||||
| @ -838,7 +1147,7 @@ const handleEnterSearchResultChat = () => { | |||||||
|       <template v-else> |       <template v-else> | ||||||
|         <TalkItem |         <TalkItem | ||||||
|           v-for="item in items" |           v-for="item in items" | ||||||
|           :key="item.index_name" |           :key="item.index_name + item.unread_num" | ||||||
|           :data="item" |           :data="item" | ||||||
|           :avatar="item.avatar" |           :avatar="item.avatar" | ||||||
|           :username="item.remark || item.name" |           :username="item.remark || item.name" | ||||||
| @ -853,6 +1162,12 @@ const handleEnterSearchResultChat = () => { | |||||||
| 
 | 
 | ||||||
|   <GroupLaunch v-if="isShowGroup" @close="isShowGroup = false" @on-submit="onReload" /> |   <GroupLaunch v-if="isShowGroup" @close="isShowGroup = false" @on-submit="onReload" /> | ||||||
| 
 | 
 | ||||||
|  |   <UserCardModal | ||||||
|  |     v-model:show="state.userInfo.isShowUserCardModal" | ||||||
|  |     v-model:uid="(state.userInfo as any).user_id" | ||||||
|  |     :euid="(state.userInfo as any).erp_user_id" | ||||||
|  |     @update:send="closeAddFriendModal" | ||||||
|  |   /> | ||||||
|   <customModal |   <customModal | ||||||
|     v-model:show="state.isShowAddressBookModal" |     v-model:show="state.isShowAddressBookModal" | ||||||
|     title="通讯录" |     title="通讯录" | ||||||
| @ -864,13 +1179,16 @@ const handleEnterSearchResultChat = () => { | |||||||
|   > |   > | ||||||
|     <template #content> |     <template #content> | ||||||
|       <div class="custom-modal-content"> |       <div class="custom-modal-content"> | ||||||
|         <n-card style="padding: 0 12px;"> |         <n-card style="padding: 0 12px"> | ||||||
|           <n-tabs |           <n-tabs | ||||||
|             type="line" |             type="line" | ||||||
|             @update:value="handleAddressBookTabChange" |             @update:value="handleAddressBookTabChange" | ||||||
|             tab-style="font-size: 16px; font-weight: 600;color: #8B8B8B;" |             tab-style="font-size: 16px; font-weight: 600;color: #8B8B8B;" | ||||||
|           > |           > | ||||||
|             <n-tab name="employeeAddressBook">员工通讯录</n-tab> |             <!-- <n-tab name="employeeAddressBook">组织架构</n-tab> | ||||||
|  |             <n-tab name="employeeAddressBook">我的好友</n-tab> --> | ||||||
|  |             <n-tab name="employeeAddressBook">组织架构</n-tab> | ||||||
|  |             <n-tab name="myFriend">我的好友</n-tab> | ||||||
|             <n-tab name="groupChatList">群聊列表</n-tab> |             <n-tab name="groupChatList">群聊列表</n-tab> | ||||||
|           </n-tabs> |           </n-tabs> | ||||||
|           <xSearchForm |           <xSearchForm | ||||||
| @ -887,11 +1205,25 @@ const handleEnterSearchResultChat = () => { | |||||||
|             @change="changeGroupChatListSearch" |             @change="changeGroupChatListSearch" | ||||||
|             :cols="3" |             :cols="3" | ||||||
|           ></xSearchForm> |           ></xSearchForm> | ||||||
|  |           <xSearchForm | ||||||
|  |             v-if="state.addressBookCurrentTab == 'myFriend'" | ||||||
|  |             :search-config="state.myFriendSearchConfig" | ||||||
|  |             customInputPlaceholder="请输入好友名称" | ||||||
|  |             @change="changeMyFriendListSearch" | ||||||
|  |             :cols="3" | ||||||
|  |           ></xSearchForm> | ||||||
|  |           <p | ||||||
|  |             v-if="state.addressBookCurrentTab === 'employeeAddressBook'" | ||||||
|  |             style="transform: translateY(-10px)" | ||||||
|  |           > | ||||||
|  |             {{ state.company_name }} | ||||||
|  |           </p> | ||||||
|           <div |           <div | ||||||
|             class="addressBook-content" |             class="addressBook-content" | ||||||
|             v-if="state.addressBookCurrentTab == 'employeeAddressBook'" |             v-if="state.addressBookCurrentTab == 'employeeAddressBook'" | ||||||
|           > |           > | ||||||
|             <div class="addressBook-tree" v-if="!state.addressBookSearchNickName"> |             <!-- 隐藏组织架构树 --> | ||||||
|  |             <div class="addressBook-tree" v-if="!state.addressBookSearchNickName && 0"> | ||||||
|               <fl-tree |               <fl-tree | ||||||
|                 :data="state.treeData" |                 :data="state.treeData" | ||||||
|                 :expandedKeys="state.expandedKeys" |                 :expandedKeys="state.expandedKeys" | ||||||
| @ -926,6 +1258,36 @@ const handleEnterSearchResultChat = () => { | |||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|           </div> |           </div> | ||||||
|  | 
 | ||||||
|  |           <!-- 我的好友 --> | ||||||
|  |           <div class="groupChatList-content" v-if="state.addressBookCurrentTab == 'myFriend'"> | ||||||
|  |             <div class="groupChatList-table"> | ||||||
|  |               <xNDataTable | ||||||
|  |                 :columns="state.myFriendListColumns" | ||||||
|  |                 :data="state.myFriendListData" | ||||||
|  |                 :style="{ | ||||||
|  |                   height: '523px', | ||||||
|  |                   width: '1148px' | ||||||
|  |                 }" | ||||||
|  |                 flex-height | ||||||
|  |               ></xNDataTable> | ||||||
|  |               <div class="groupChatList-pagination"> | ||||||
|  |                 <n-pagination | ||||||
|  |                   v-model:page="state.myFriendListPage" | ||||||
|  |                   v-model:page-size="state.myFriendListPageSize" | ||||||
|  |                   :item-count="state.myFriendListTotal" | ||||||
|  |                   show-quick-jumper | ||||||
|  |                   show-size-picker | ||||||
|  |                   :page-sizes="[10, 20, 50]" | ||||||
|  |                   :on-update:page="handleMyFriendListPagination" | ||||||
|  |                   :on-update:page-size="handleMyFriendListPaginationSize" | ||||||
|  |                 > | ||||||
|  |                   <template #prefix="{ itemCount }"> 共 {{ itemCount }} 条记录 </template> | ||||||
|  |                 </n-pagination> | ||||||
|  |               </div> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  | 
 | ||||||
|           <div class="groupChatList-content" v-if="state.addressBookCurrentTab == 'groupChatList'"> |           <div class="groupChatList-content" v-if="state.addressBookCurrentTab == 'groupChatList'"> | ||||||
|             <div class="groupChatList-table"> |             <div class="groupChatList-table"> | ||||||
|               <xNDataTable |               <xNDataTable | ||||||
| @ -957,6 +1319,41 @@ const handleEnterSearchResultChat = () => { | |||||||
|       </div> |       </div> | ||||||
|     </template> |     </template> | ||||||
|   </customModal> |   </customModal> | ||||||
|  |   <customModal | ||||||
|  |     v-model:show="state.isShowAddFriendModal" | ||||||
|  |     title="添加好友" | ||||||
|  |     :style="state.customModalStyle" | ||||||
|  |     :customCloseBtn="true" | ||||||
|  |     :closable="false" | ||||||
|  |     :customCloseEvent="true" | ||||||
|  |     @customCloseModal="closeAddFriendModal" | ||||||
|  |   > | ||||||
|  |     <template #content> | ||||||
|  |       <div class="custom-modal-content"> | ||||||
|  |         <n-card style="padding: 0 12px"> | ||||||
|  |           <xSearchForm | ||||||
|  |             :search-config="state.addFriendSearchConfig" | ||||||
|  |             customInputPlaceholder="请输入姓名" | ||||||
|  |             @change="changeAddFriendSearch" | ||||||
|  |             :cols="3" | ||||||
|  |           ></xSearchForm> | ||||||
|  |           <div class="groupChatList-content"> | ||||||
|  |             <div class="groupChatList-table"> | ||||||
|  |               <xNDataTable | ||||||
|  |                 :columns="state.addFriendListColumns" | ||||||
|  |                 :data="state.addFriendList" | ||||||
|  |                 :style="{ | ||||||
|  |                   height: '523px', | ||||||
|  |                   width: '1148px' | ||||||
|  |                 }" | ||||||
|  |                 flex-height | ||||||
|  |               ></xNDataTable> | ||||||
|  |             </div> | ||||||
|  |           </div> | ||||||
|  |         </n-card> | ||||||
|  |       </div> | ||||||
|  |     </template> | ||||||
|  |   </customModal> | ||||||
| 
 | 
 | ||||||
|   <customModal |   <customModal | ||||||
|     v-model:show="state.isShowSearchRecordModal" |     v-model:show="state.isShowSearchRecordModal" | ||||||
| @ -969,7 +1366,7 @@ const handleEnterSearchResultChat = () => { | |||||||
|   > |   > | ||||||
|     <template #content> |     <template #content> | ||||||
|       <div class="search-record-modal-content"> |       <div class="search-record-modal-content"> | ||||||
|         <n-card style="padding: 0 12px;"> |         <n-card style="padding: 0 12px"> | ||||||
|           <div class="search-record-input"> |           <div class="search-record-input"> | ||||||
|             <span class="search-record-input-title">搜索</span> |             <span class="search-record-input-title">搜索</span> | ||||||
|             <n-input |             <n-input | ||||||
|  | |||||||
| @ -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 } = useMenu() | const { dropdown, showDropdownMenu, closeDropdownMenu, isOneMonthBefore } = useMenu() | ||||||
| const { showUserInfoModal } = useInject() | const { showUserInfoModal } = useInject() | ||||||
| const dialogueStore = useDialogueStore() | const dialogueStore = useDialogueStore() | ||||||
| const userStore = useUserStore() | const userStore = useUserStore() | ||||||
| @ -330,13 +330,15 @@ const onContextMenuHandle = (key: string) => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const onRowClick = (item: ITalkRecord) => { | const onRowClick = (item: ITalkRecord) => { | ||||||
|   if (dialogueStore.isOpenMultiSelect) { |   if (dialogueStore.isOpenMultiSelect && isOneMonthBefore(item.created_at.split(' ')[0])) { | ||||||
|     console.log('item.msg_type', item.msg_type) |     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('只支持转发近一个月内的消息') | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -816,8 +818,10 @@ 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" | ||||||
|             /> |             /> | ||||||
| @ -855,7 +859,14 @@ const onCustomSkipBottomEvent = () => { | |||||||
|             </div> |             </div> | ||||||
|             <div |             <div | ||||||
|               class="talk-content" |               class="talk-content" | ||||||
|               :class="{ pointer: dialogueStore.isOpenMultiSelect }" |               :class="{ | ||||||
|  |                 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 | ||||||
| @ -872,7 +883,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> | ||||||
| @ -917,14 +928,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 | ||||||
| @ -946,11 +957,12 @@ 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="(talkReadDetailItem, |                         v-for="( | ||||||
|                         talkReadDetailIndex) in state.talkReadListDetail" |                           talkReadDetailItem, talkReadDetailIndex | ||||||
|  |                         ) in state.talkReadListDetail" | ||||||
|                         :key="talkReadDetailIndex" |                         :key="talkReadDetailIndex" | ||||||
|                       > |                       > | ||||||
|                         <avatarModule |                         <avatarModule | ||||||
| @ -970,10 +982,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> | ||||||
| @ -1005,7 +1017,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" | ||||||
|  | |||||||
| @ -48,7 +48,13 @@ 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 | ||||||
| @ -60,13 +66,15 @@ 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 (canAddRevokeOption(uid, item, (dialogueStore.groupInfo as any).is_manager)) { | ||||||
|       dropdown.options.push({ label: '撤回', key: 'revoke' }); |       dropdown.options.push({ label: '撤回', key: 'revoke' }); | ||||||
| @ -74,8 +82,6 @@ export function useMenu() { | |||||||
|     |     | ||||||
|     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' })
 | ||||||
|     // }
 |     // }
 | ||||||
| @ -84,7 +90,6 @@ 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 | ||||||
| @ -95,5 +100,5 @@ export function useMenu() { | |||||||
|     dropdown.item = {} |     dropdown.item = {} | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   return { dropdown, showDropdownMenu, closeDropdownMenu } |   return { dropdown, showDropdownMenu, closeDropdownMenu, isOneMonthBefore } | ||||||
| } | } | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user