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) => { | ||||
|   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
											
										
									
								
							| @ -41,7 +41,7 @@ const onCanplay = () => { | ||||
|   state.duration = audioRef.value.duration | ||||
|   durationDesc.value = formatTime(parseInt(audioRef.value.duration)) | ||||
|   state.loading = false | ||||
|    | ||||
| 
 | ||||
|   if (props.data.is_convert_text === 1 && props.data.extra.content) { | ||||
|     setTimeout(() => { | ||||
|       state.showText = true | ||||
| @ -72,47 +72,61 @@ const formatTime = (value: number = 0) => { | ||||
| } | ||||
| </script> | ||||
| <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"> | ||||
|     <audio | ||||
|       ref="audioRef" | ||||
|       preload="auto" | ||||
|       type="audio/mp3,audio/wav" | ||||
|       :src="extra.url" | ||||
|       @timeupdate="onTimeUpdate" | ||||
|       @ended="onPlayEnd" | ||||
|       @canplay="onCanplay" | ||||
|       @error="onError" | ||||
|     /> | ||||
|       <aTrumpet :isPlay="false" color="black" :size="30"></aTrumpet> | ||||
|       <audio | ||||
|         ref="audioRef" | ||||
|         preload="auto" | ||||
|         type="audio/mp3,audio/wav" | ||||
|         :src="extra.url" | ||||
|         @timeupdate="onTimeUpdate" | ||||
|         @ended="onPlayEnd" | ||||
|         @canplay="onCanplay" | ||||
|         @error="onError" | ||||
|       /> | ||||
| 
 | ||||
|     <div class="play"> | ||||
|       <div class="btn pointer" @click.stop="onPlay"> | ||||
|         <n-icon :size="18" :component="state.isAudioPlay ? PauseOne : PlayOne" /> | ||||
|       <div class="play"> | ||||
|         <div class="btn pointer" @click.stop="onPlay"> | ||||
|           <!-- <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 class="desc"> | ||||
|       <!-- <div class="desc"> | ||||
|       <span class="line" v-for="i in 23" :key="i"></span> | ||||
|       <span | ||||
|         class="indicator" | ||||
|         :style="{ left: state.progress + '%' }" | ||||
|         v-show="state.progress > 0" | ||||
|       ></span> | ||||
|     </div> --> | ||||
|       <!-- <div class="time">{{ durationDesc }}</div> --> | ||||
|       <div>{{ durationDesc.split('"')[0] }}s</div> | ||||
|     </div> | ||||
|     <div class="time">{{ durationDesc }}</div> | ||||
|   </div> | ||||
|    | ||||
|   <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 class="flex justify-center items-center" v-if="data.is_convert_text===1&&!data.extra.content"> | ||||
|         <n-spin :stroke-width="3" size="small" /> | ||||
|       </div> | ||||
|       <transition name="fade"> | ||||
|         <div class="text-content" v-if="data.extra.content">{{ data.extra.content }}</div> | ||||
|       </transition> | ||||
|     </div> | ||||
|   </transition> | ||||
|   </div> | ||||
| 
 | ||||
|     <transition name="expand"> | ||||
|       <div | ||||
|         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" /> | ||||
|         </div> | ||||
|         <transition name="fade"> | ||||
|           <div class="text-content" v-if="data.extra.content">{{ data.extra.content }}</div> | ||||
|         </transition> | ||||
|       </div> | ||||
|     </transition> | ||||
|   </div> | ||||
| </template> | ||||
| <style lang="less" scoped> | ||||
| .im-message-audio { | ||||
| @ -135,7 +149,7 @@ const formatTime = (value: number = 0) => { | ||||
|     .btn { | ||||
|       width: 26px; | ||||
|       height: 26px; | ||||
|       background-color: var(--audio-btn-bg-color); | ||||
|       // background-color: var(--audio-btn-bg-color); | ||||
|       border-radius: 50%; | ||||
|       color: rgb(24, 24, 24); | ||||
|       display: flex; | ||||
|  | ||||
| @ -8,9 +8,17 @@ import { useTalkStore } from '@/store' | ||||
| import { useRouter } from 'vue-router' | ||||
| import xNModal from '@/components/x-naive-ui/x-n-modal/index.vue' | ||||
| 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 talkStore = useTalkStore() | ||||
| const emit = defineEmits(['update:show', 'update:uid', 'updateRemark']) | ||||
| const emit = defineEmits(['update:show', 'update:uid', 'updateRemark', 'update:send']) | ||||
| const props = defineProps({ | ||||
|   show: { | ||||
|     type: Boolean, | ||||
| @ -86,6 +94,7 @@ const onLoadData = () => { | ||||
| const onToTalk = () => { | ||||
|   talkStore.toTalk(1, props.uid, router) | ||||
|   emit('update:show', false) | ||||
|   emit('update:send') | ||||
| } | ||||
| 
 | ||||
| // const onJoinContact = () => { | ||||
| @ -167,60 +176,85 @@ const onToTalk = () => { | ||||
| //   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 = () => { | ||||
|   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 = () => { | ||||
|   // loading.value = true | ||||
|   userInfo.value = { | ||||
|   id: 0, | ||||
|   avatar: '', | ||||
|   gender: 0, | ||||
|   mobile: '', | ||||
|   motto: '', | ||||
|   nickname: '', | ||||
|   remark: '', | ||||
|   email: '', | ||||
|   status: 1, | ||||
|   text: '' | ||||
| } | ||||
|     id: 0, | ||||
|     avatar: '', | ||||
|     gender: 0, | ||||
|     mobile: '', | ||||
|     motto: '', | ||||
|     nickname: '', | ||||
|     remark: '', | ||||
|     email: '', | ||||
|     status: 1, | ||||
|     text: '' | ||||
|   } | ||||
| } | ||||
| </script> | ||||
| 
 | ||||
| <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="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> | ||||
|        | ||||
| 
 | ||||
|       <template v-if="loading"> | ||||
|         <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"> | ||||
|             <n-skeleton  height="59px" width="59px" /> | ||||
|             <n-skeleton height="59px" width="59px" /> | ||||
|           </div> | ||||
|           <div class="w-full"> | ||||
|             <n-skeleton text style="width: 80%; margin-bottom: 5px;" /> | ||||
|             <n-skeleton text style="width: 60%;" /> | ||||
|             <n-skeleton text style="width: 80%; margin-bottom: 5px" /> | ||||
|             <n-skeleton text style="width: 60%" /> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div class="bg-#fff rounded-4px mb-20px"> | ||||
|           <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: 60%;" /> | ||||
|             <n-skeleton text style="width: 30%; margin-right: 10px" /> | ||||
|             <n-skeleton text style="width: 60%" /> | ||||
|           </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> | ||||
|       </template> | ||||
|        | ||||
| 
 | ||||
|       <template v-else> | ||||
|         <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"> | ||||
|            <n-image width="59" :src="userInfo.avatar" > | ||||
| 
 | ||||
|            </n-image> | ||||
|           <div class="w-59px h-59px rounded-8px mr-12px overflow-hidden"> | ||||
|             <n-image width="59" :src="userInfo.avatar"> </n-image> | ||||
|           </div> | ||||
|           <div> | ||||
|             <div class="text-#000 text-16px mb-5px">{{ userInfo.nickname }}</div> | ||||
| @ -234,11 +268,15 @@ const onAfterLeave = () => { | ||||
|           </div> | ||||
|           <div class="flex px-15px py-9px"> | ||||
|             <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 class="flex px-15px py-9px"> | ||||
|             <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 class="flex px-15px py-9px"> | ||||
|             <div class="text-#000 text-12px w-84px">手机号</div> | ||||
| @ -246,19 +284,39 @@ const onAfterLeave = () => { | ||||
|           </div> | ||||
|           <div class="flex px-15px py-9px"> | ||||
|             <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 class="flex px-15px py-9px"> | ||||
|             <div class="text-#000 text-12px w-84px">入职日期</div> | ||||
|             <div class="text-#747474 text-12px">{{ userInfo.enter_date }}</div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <div> | ||||
|           <n-button block color="#EEE9F8" text-color="#46299D"     @click="onToTalk"> | ||||
|               <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=""> | ||||
|                 <span>发送消息</span> | ||||
|               </div> | ||||
|         <div v-if="showBtn"> | ||||
|           <n-button block color="#EEE9F8" text-color="#46299D" @click="onToTalk" v-if="isFriend"> | ||||
|             <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="" /> | ||||
|               <span>发送消息</span> | ||||
|             </div> | ||||
|           </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> | ||||
|       </template> | ||||
|  | ||||
| @ -12,6 +12,10 @@ import customModal from '@/components/common/customModal.vue' | ||||
| import historyRecord from '@/components/search/searchByCondition.vue' | ||||
| import { ServeEditGroupNotice, ServeGetGroupNotices, ServeDeleteGroupNotice } from '@/api/group' | ||||
| 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 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( | ||||
|   () => talkParams, | ||||
|   (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 } | ||||
| ) | ||||
| @ -508,7 +534,27 @@ const clearSelectedDateTime = () => { | ||||
|     </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 | ||||
|         :uid="talkParams.uid" | ||||
|         :talk_type="talkParams.type" | ||||
| @ -593,13 +639,13 @@ const clearSelectedDateTime = () => { | ||||
|   > | ||||
|     <template #content> | ||||
|       <div class="search-record-modal-searchArea"> | ||||
|         <n-card style="padding: 0 12px;"> | ||||
|         <n-card style="padding: 0 12px"> | ||||
|           <div class="search-record-input"> | ||||
|             <span class="search-record-input-title">搜索</span> | ||||
|             <n-input | ||||
|               type="text" | ||||
|               v-model:value="state.searchRecordByConditionText" | ||||
|               :placeholder="state.conditionTag && state.conditionTag !== 'all'?'':'请输入'" | ||||
|               :placeholder="state.conditionTag && state.conditionTag !== 'all' ? '' : '请输入'" | ||||
|               clearable | ||||
|             > | ||||
|               <template #clear-icon> | ||||
| @ -623,7 +669,7 @@ const clearSelectedDateTime = () => { | ||||
|               v-model:show="state.showDateConditionPopover" | ||||
|               trigger="click" | ||||
|               placement="bottom-start" | ||||
|               style="height: 312px; padding: 0;" | ||||
|               style="height: 312px; padding: 0" | ||||
|               @update:show="onDatePickShow" | ||||
|             > | ||||
|               <template #trigger> | ||||
| @ -651,7 +697,7 @@ const clearSelectedDateTime = () => { | ||||
|               v-model:show="state.showMemberListByAlphabetPopover" | ||||
|               trigger="click" | ||||
|               placement="bottom-start" | ||||
|               style="width: 290px; height: 505px; padding: 0;" | ||||
|               style="width: 290px; height: 505px; padding: 0" | ||||
|               v-if="talkParams.type === 2" | ||||
|             > | ||||
|               <template #trigger> | ||||
| @ -660,10 +706,10 @@ const clearSelectedDateTime = () => { | ||||
|               <div class="member-list-by-alphabet-container"> | ||||
|                 <n-input | ||||
|                   placeholder="请输入群成员" | ||||
|                   style="margin: 0 0 17px;" | ||||
|                   style="margin: 0 0 17px" | ||||
|                   v-model:value="state.searchMemberByAlphabet" | ||||
|                 /> | ||||
|                 <n-scrollbar style="height: 430px;"> | ||||
|                 <n-scrollbar style="height: 430px"> | ||||
|                   <div | ||||
|                     class="member-list-by-alphabet" | ||||
|                     v-for="(alphabetMembersItem, alphabetMembersIndex) in membersByAlphabet" | ||||
| @ -675,7 +721,8 @@ const clearSelectedDateTime = () => { | ||||
|                     <div class="member-list-each-alphabet"> | ||||
|                       <div | ||||
|                         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" | ||||
|                         @click="handleMemberItemClick(memberItem)" | ||||
|                       > | ||||
| @ -776,7 +823,7 @@ const clearSelectedDateTime = () => { | ||||
|               }" | ||||
|             ></avatarModule> | ||||
|             <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 | ||||
|               }}</span> | ||||
|               <span>{{ state.groupNoticeInfo.updated_at }}</span> | ||||
|  | ||||
| @ -1,4 +1,4 @@ | ||||
| <script lang="ts" setup> | ||||
| <script lang="tsx" setup> | ||||
| import { | ||||
|   computed, | ||||
|   ref, | ||||
| @ -23,10 +23,10 @@ import { | ||||
|   NButton, | ||||
|   NPagination | ||||
| } 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 Skeleton from './Skeleton.vue' | ||||
| import { ServeClearTalkUnreadNum } from '@/api/chat' | ||||
| import { ServeClearTalkUnreadNum, ServeAddFriend } from '@/api/chat' | ||||
| import GroupLaunch from '@/components/group/GroupLaunch.vue' | ||||
| import { getCacheIndexName } from '@/utils/talk' | ||||
| import { ISession } from '@/types/chat' | ||||
| @ -41,6 +41,11 @@ import { ServeSeachQueryAll, ServeQueryTalkRecord, ServeUserGroupChatList } from | ||||
| import { getUserInfoByERPUserId } from '@/api/user' | ||||
| import HighlightText from '@/components/search/highLightText.vue' | ||||
| 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 currentInstance = getCurrentInstance() | ||||
| @ -60,7 +65,43 @@ const isShowGroup = ref(false) | ||||
| const searchKeyword = ref('') | ||||
| const topItems = computed((): ISession[] => talkStore.topItems) | ||||
| 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 = () => { | ||||
|   return h( | ||||
| @ -89,10 +130,10 @@ const renderChatAppSearch = () => { | ||||
|           state.searchRecordText = searchText | ||||
|           state.selectItemInList = res | ||||
|         } else { | ||||
|           if(searchResultKey === 'user_infos'){ | ||||
|           if (searchResultKey === 'user_infos') { | ||||
|             talk_type = 1 | ||||
|           } | ||||
|           if(searchResultKey === 'combinedGroup'){ | ||||
|           if (searchResultKey === 'combinedGroup') { | ||||
|             talk_type = 2 | ||||
|           } | ||||
|           talkStore.toTalk(talk_type, receiver_id, router) | ||||
| @ -145,7 +186,13 @@ const renderChatAppSearch = () => { | ||||
| } | ||||
| 
 | ||||
| const state = reactive({ | ||||
|   userInfo: { | ||||
|     isShowUserCardModal: false, | ||||
|     user_id: NaN, | ||||
|     erp_user_id: NaN | ||||
|   }, | ||||
|   isShowAddressBookModal: false, // 是否显示通讯录模态框 | ||||
|   isShowAddFriendModal: false, // 是否显示添加好友模态框 | ||||
|   customModalStyle: { | ||||
|     width: '1288px', | ||||
|     height: '846px', | ||||
| @ -166,7 +213,25 @@ const state = reactive({ | ||||
|       type: 'input', | ||||
|       valueType: 'string' | ||||
|     } | ||||
|   ], // 群聊列表搜索配置 | ||||
|   ], | ||||
|   // 我的好友搜索配置 | ||||
|   myFriendSearchConfig: [ | ||||
|     { | ||||
|       label: '好友名称', | ||||
|       key: 'myFriendname', | ||||
|       type: 'input', | ||||
|       valueType: 'string' | ||||
|     } | ||||
|   ], | ||||
|   addFriendSearchConfig: [ | ||||
|     { | ||||
|       label: '姓名', | ||||
|       key: 'friendName', | ||||
|       type: 'input', | ||||
|       valueType: 'string' | ||||
|     } | ||||
|   ], // 添加好友搜索配置 | ||||
|   // 群聊列表搜索配置 | ||||
|   treeData: [], | ||||
|   expandedKeys: [], | ||||
|   clickKey: 3, | ||||
| @ -175,29 +240,29 @@ const state = reactive({ | ||||
|   addressBookColumns: [ | ||||
|     { | ||||
|       title: '姓名 【工号】', | ||||
|       field: 'nickName', | ||||
|       field: 'nickname', | ||||
|       width: 200, | ||||
|       ellipsis: { | ||||
|         tooltip: true | ||||
|       }, | ||||
|       render(row, index) { | ||||
|         return row.nickName + '【' + row.jobNum + '】' | ||||
|         return row.nickname + '【' + row.job_num + '】' | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
|       title: '岗位名称', | ||||
|       field: 'positionName', | ||||
|       field: 'user_position', | ||||
|       width: 400, | ||||
|       ellipsis: { | ||||
|         tooltip: true | ||||
|       }, | ||||
|       render(row, index) { | ||||
|         let positionNames = Array.isArray(row.depPositions) | ||||
|           ? row.depPositions.flatMap((dep) => | ||||
|               Array.isArray(dep.positions) ? dep.positions.map((pos) => pos.name) : [] | ||||
|             ) | ||||
|           : [] | ||||
|         return positionNames.join(' , ') | ||||
|         // 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(' , ') | ||||
|       } | ||||
|     }, | ||||
|     { | ||||
| @ -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: [], // 通讯录表格数据 | ||||
|   company_name: '', // 当前公司别 | ||||
|   groupChatListData: [], // 群聊列表表格数据 | ||||
|   myFriendListData: [], // 我的好友表格数据 | ||||
|   addFriendList: [], // 搜索出来的可添加好友 | ||||
|   addressBookTableHeight: 524, // 通讯录表格高度 | ||||
|   addressBookTableWidth: 800, // 通讯录表格宽度 | ||||
|   addressBookTableWidth: 1142, // 通讯录表格宽度 | ||||
|   addressBookPage: 1, // 通讯录表格页码 | ||||
|   addressBookPageSize: 10, // 通讯录表格每页条数 | ||||
|   addressBookTotal: 0, // 通讯录表格总条数 | ||||
| @ -283,6 +464,10 @@ const state = reactive({ | ||||
|   groupChatListPageSize: 10, // 群聊列表表格每页条数 | ||||
|   groupChatListTotal: 0, // 群聊列表表格总条数 | ||||
|   groupChatListSearchGroupName: '', // 群聊列表搜索条件-群聊名称 | ||||
|   myFriendListPage: 1, // 我的好友表格页码 | ||||
|   myFriendListPageSize: 10, // 我的好友表格每页条数 | ||||
|   myFriendListTotal: 0, // 我的好友表格总条数 | ||||
|   myFriendListSearchName: '', // 我的好友搜索条件-好友名称 | ||||
|   chatSearchOptions: [ | ||||
|     { | ||||
|       key: 'chatSearch', | ||||
| @ -342,7 +527,7 @@ watch( | ||||
|       state.addressBookTableWidth = 1142 | ||||
|       state.addressBookPage = 1 | ||||
|     } else { | ||||
|       state.addressBookTableWidth = 800 | ||||
|       // state.addressBookTableWidth = 800 | ||||
|       state.clickKey = 3 | ||||
|       state.treeRefreshCount++ | ||||
|       state.addressBookPage = 1 | ||||
| @ -350,6 +535,17 @@ watch( | ||||
|     getDepPoisUser() | ||||
|   } | ||||
| ) | ||||
| watch( | ||||
|   () => state.myFriendListSearchName, | ||||
|   (newValue, oldValue) => { | ||||
|     if (newValue) { | ||||
|       state.myFriendListPage = 1 | ||||
|     } else { | ||||
|       state.myFriendListPage = 1 | ||||
|     } | ||||
|     getMyFriends() | ||||
|   } | ||||
| ) | ||||
| watch( | ||||
|   () => state.groupChatListSearchGroupName, | ||||
|   (newValue, oldValue) => { | ||||
| @ -450,7 +646,9 @@ const onInitialize = () => { | ||||
| onBeforeRouteUpdate(onInitialize) | ||||
| 
 | ||||
| onBeforeMount(() => { | ||||
|   getTreeData() | ||||
|   // getTreeData() | ||||
|   getDepPoisUser() | ||||
|   getMyFriends() | ||||
|   getUserGroupChatList() | ||||
| }) | ||||
| 
 | ||||
| @ -462,11 +660,24 @@ onMounted(() => { | ||||
| const showAddressBookModal = () => { | ||||
|   state.isShowAddressBookModal = true | ||||
| } | ||||
| // 点击显示添加好友模态框 | ||||
| const showAddFriendModal = () => { | ||||
|   state.isShowAddFriendModal = true | ||||
| } | ||||
| const handleSelect = (key: string | number) => { | ||||
|   if (key === 'addressBook') return showAddressBookModal() | ||||
|   showAddFriendModal() | ||||
| } | ||||
| // 点击关闭通讯录模态框 | ||||
| const closeAddressBookModal = () => { | ||||
|   state.isShowAddressBookModal = false | ||||
|   resetAddressBookModal() | ||||
| } | ||||
| // 点击关闭添加好友模态框 | ||||
| const closeAddFriendModal = () => { | ||||
|   state.isShowAddFriendModal = false | ||||
|   resetAddressBookModal() | ||||
| } | ||||
| const handleTreeClick = ({ selectedKey, tree }) => { | ||||
|   // console.log(tree) | ||||
|   state.clickKey = tree.key | ||||
| @ -488,7 +699,7 @@ const calcTreeData = (data) => { | ||||
|     delete item.sons | ||||
|   } | ||||
| } | ||||
| // 获取组织树数据 | ||||
| // 获取组织树数据-已隐藏 | ||||
| const getTreeData = () => { | ||||
|   let url = '/department/v2/tree/filter' | ||||
|   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 = () => { | ||||
|   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 = { | ||||
|     departmentId: state.addressBookSearchNickName ? undefined : state.clickKey, | ||||
|     type: 'addressBook', //查我的通讯录的时候写死addressBook | ||||
|     page: state.addressBookPage, | ||||
|     pageSize: state.addressBookPageSize, | ||||
|     status: 'notactive', | ||||
|     nickName: state.addressBookSearchNickName | ||||
|     page_size: state.addressBookPageSize, | ||||
|     name: 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 || [] | ||||
|     if (res.code === 200 && Array.isArray(res.data.user_list)) { | ||||
|       state.addressBookData = res.data.user_list || [] | ||||
|       state.company_name = res.data.company_name || '' | ||||
|       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) => { | ||||
|   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) | ||||
|       if (res.code === 200) { | ||||
|         let sysUserInfo = res.data | ||||
| @ -555,7 +820,7 @@ const resetAddressBookModal = () => { | ||||
|     state.addressBookCurrentTab = 'employeeAddressBook' | ||||
|     state.addressBookSearchNickName = '' | ||||
|     state.groupChatListSearchGroupName = '' | ||||
|     state.addressBookTableWidth = 800 | ||||
|     state.addressBookTableWidth = 1142 | ||||
|     state.clickKey = 3 | ||||
|     state.treeRefreshCount++ | ||||
|     state.addressBookPage = 1 | ||||
| @ -563,7 +828,9 @@ const resetAddressBookModal = () => { | ||||
|     state.groupChatListPage = 1 | ||||
|     state.groupChatListPageSize = 10 | ||||
|     getDepPoisUser() | ||||
|     getMyFriends() | ||||
|     getUserGroupChatList() | ||||
|     state.addFriendList = [] | ||||
|   }) | ||||
| } | ||||
| //处理页数变化 | ||||
| @ -599,6 +866,32 @@ const changeGroupChatListSearch = (value) => { | ||||
|     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 = () => { | ||||
|   let params = { | ||||
| @ -628,6 +921,19 @@ const handleGroupChatListPaginationSize = (value) => { | ||||
|   state.groupChatListPage = 1 | ||||
|   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) => { | ||||
|   console.log(searchText, searchResultKey, talk_type, receiver_id) | ||||
| @ -766,7 +1072,7 @@ const handleEnterSearchResultChat = () => { | ||||
|       <n-dropdown | ||||
|         trigger="click" | ||||
|         :options="state.chatSearchOptions" | ||||
|         style="width: 248px; height: 677px;" | ||||
|         style="width: 248px; height: 677px" | ||||
|         :show="state.showSearchDropdown" | ||||
|         @clickoutside="state.showSearchDropdown = false" | ||||
|       > | ||||
| @ -774,7 +1080,7 @@ const handleEnterSearchResultChat = () => { | ||||
|           placeholder="搜索好友 / 群聊" | ||||
|           v-model:value.trim="searchKeyword" | ||||
|           clearable | ||||
|           style="width: 78%;" | ||||
|           style="width: 78%" | ||||
|           @click="state.showSearchDropdown = true" | ||||
|         > | ||||
|           <!-- <template #prefix> | ||||
| @ -787,12 +1093,15 @@ const handleEnterSearchResultChat = () => { | ||||
|           <n-icon :component="Plus" /> | ||||
|         </template> | ||||
|       </n-button> --> | ||||
|       <img | ||||
|         style="width: 19px; height: 20px; cursor: pointer;" | ||||
|       <!-- <img | ||||
|         style="width: 19px; height: 20px; cursor: pointer" | ||||
|         src="@/assets/image/chatList/addressBook.png" | ||||
|         alt="" | ||||
|         @click="showAddressBookModal" | ||||
|       /> | ||||
|       /> --> | ||||
|       <n-dropdown :options="option" @select="handleSelect"> | ||||
|         <n-button> <n-icon :component="AddOne" /></n-button> | ||||
|       </n-dropdown> | ||||
|     </header> | ||||
| 
 | ||||
|     <!-- 置顶栏目 --> | ||||
| @ -838,7 +1147,7 @@ const handleEnterSearchResultChat = () => { | ||||
|       <template v-else> | ||||
|         <TalkItem | ||||
|           v-for="item in items" | ||||
|           :key="item.index_name" | ||||
|           :key="item.index_name + item.unread_num" | ||||
|           :data="item" | ||||
|           :avatar="item.avatar" | ||||
|           :username="item.remark || item.name" | ||||
| @ -853,6 +1162,12 @@ const handleEnterSearchResultChat = () => { | ||||
| 
 | ||||
|   <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 | ||||
|     v-model:show="state.isShowAddressBookModal" | ||||
|     title="通讯录" | ||||
| @ -864,13 +1179,16 @@ const handleEnterSearchResultChat = () => { | ||||
|   > | ||||
|     <template #content> | ||||
|       <div class="custom-modal-content"> | ||||
|         <n-card style="padding: 0 12px;"> | ||||
|         <n-card style="padding: 0 12px"> | ||||
|           <n-tabs | ||||
|             type="line" | ||||
|             @update:value="handleAddressBookTabChange" | ||||
|             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-tabs> | ||||
|           <xSearchForm | ||||
| @ -887,11 +1205,25 @@ const handleEnterSearchResultChat = () => { | ||||
|             @change="changeGroupChatListSearch" | ||||
|             :cols="3" | ||||
|           ></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 | ||||
|             class="addressBook-content" | ||||
|             v-if="state.addressBookCurrentTab == 'employeeAddressBook'" | ||||
|           > | ||||
|             <div class="addressBook-tree" v-if="!state.addressBookSearchNickName"> | ||||
|             <!-- 隐藏组织架构树 --> | ||||
|             <div class="addressBook-tree" v-if="!state.addressBookSearchNickName && 0"> | ||||
|               <fl-tree | ||||
|                 :data="state.treeData" | ||||
|                 :expandedKeys="state.expandedKeys" | ||||
| @ -926,6 +1258,36 @@ const handleEnterSearchResultChat = () => { | ||||
|               </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-table"> | ||||
|               <xNDataTable | ||||
| @ -957,6 +1319,41 @@ const handleEnterSearchResultChat = () => { | ||||
|       </div> | ||||
|     </template> | ||||
|   </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 | ||||
|     v-model:show="state.isShowSearchRecordModal" | ||||
| @ -969,7 +1366,7 @@ const handleEnterSearchResultChat = () => { | ||||
|   > | ||||
|     <template #content> | ||||
|       <div class="search-record-modal-content"> | ||||
|         <n-card style="padding: 0 12px;"> | ||||
|         <n-card style="padding: 0 12px"> | ||||
|           <div class="search-record-input"> | ||||
|             <span class="search-record-input-title">搜索</span> | ||||
|             <n-input | ||||
|  | ||||
| @ -82,7 +82,7 @@ const { loadConfig, records, onLoad, onRefreshLoad, onJumpMessage, onLoadMoreDow | ||||
| ) | ||||
| const uploadsStore = useUploadsStore() | ||||
| const { useMessage } = useUtil() | ||||
| const { dropdown, showDropdownMenu, closeDropdownMenu } = useMenu() | ||||
| const { dropdown, showDropdownMenu, closeDropdownMenu, isOneMonthBefore } = useMenu() | ||||
| const { showUserInfoModal } = useInject() | ||||
| const dialogueStore = useDialogueStore() | ||||
| const userStore = useUserStore() | ||||
| @ -184,7 +184,7 @@ const onCopyText = (data: ITalkRecord) => { | ||||
|       return clipboard(htmlDecode(data.extra.content), () => useMessage.success('复制成功')) | ||||
|     } | ||||
|   } | ||||
|   console.log('data.extra?.url',data.extra?.url) | ||||
|   console.log('data.extra?.url', data.extra?.url) | ||||
|   if (data.extra?.url) { | ||||
|     return clipboardImage(data.extra.url, () => { | ||||
|       useMessage.success('复制成功') | ||||
| @ -330,13 +330,15 @@ const onContextMenuHandle = (key: string) => { | ||||
| } | ||||
| 
 | ||||
| const onRowClick = (item: ITalkRecord) => { | ||||
|   if (dialogueStore.isOpenMultiSelect) { | ||||
|     console.log('item.msg_type',item.msg_type) | ||||
|   if (dialogueStore.isOpenMultiSelect && isOneMonthBefore(item.created_at.split(' ')[0])) { | ||||
|     console.log('item.msg_type', item.msg_type) | ||||
|     if (ForwardableMessageType.includes(item.msg_type)) { | ||||
|       item.isCheck = !item.isCheck | ||||
|     } else { | ||||
|       useMessage.info('此类消息不支持转发') | ||||
|     } | ||||
|   } else { | ||||
|     useMessage.info('只支持转发近一个月内的消息') | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| @ -363,7 +365,7 @@ watch( | ||||
|       } | ||||
|       currentSessionKey.value = newSessionKey | ||||
|     } | ||||
|      | ||||
| 
 | ||||
|     let specialParams = undefined | ||||
|     if (newProps.specifiedMsg) { | ||||
|       try { | ||||
| @ -377,7 +379,7 @@ watch( | ||||
|         } | ||||
|       } catch (e) {} | ||||
|     } | ||||
|      | ||||
| 
 | ||||
|     // 检查是否需要阻止刷新 | ||||
|     if (dialogueStore.noRefreshRecords) { | ||||
|       // 清除之前的定时器(如果存在) | ||||
| @ -391,7 +393,7 @@ watch( | ||||
|       }, 3000) | ||||
|       return | ||||
|     } | ||||
|      | ||||
| 
 | ||||
|     onLoad( | ||||
|       { | ||||
|         receiver_id: newProps.receiver_id, | ||||
| @ -816,8 +818,10 @@ const onCustomSkipBottomEvent = () => { | ||||
|         > | ||||
|           <!-- 多选按钮 --> | ||||
|           <aside v-if="dialogueStore.isOpenMultiSelect" class="checkbox-column shrink-0"> | ||||
|             <!-- 近一个月外的消息多选框禁用 {{ item }} --> | ||||
|             <n-checkbox | ||||
|               size="small" | ||||
|               :disabled="!isOneMonthBefore(item.created_at.split(' ')[0])" | ||||
|               :checked="item.isCheck" | ||||
|               @update:checked="item.isCheck = !item.isCheck" | ||||
|             /> | ||||
| @ -855,7 +859,14 @@ const onCustomSkipBottomEvent = () => { | ||||
|             </div> | ||||
|             <div | ||||
|               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)" | ||||
|             > | ||||
|               <component | ||||
| @ -872,7 +883,7 @@ const onCustomSkipBottomEvent = () => { | ||||
|                 " | ||||
|                 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"> | ||||
|                     <ExclamationCircleFilled /> | ||||
|                   </n-icon> | ||||
| @ -917,14 +928,14 @@ const onCustomSkipBottomEvent = () => { | ||||
|               <n-popover | ||||
|                 trigger="click" | ||||
|                 placement="bottom-end" | ||||
|                 style="height: 382px; padding: 0;" | ||||
|                 style="height: 382px; padding: 0" | ||||
|                 v-if="props.talk_type === 2" | ||||
|               > | ||||
|                 <template #trigger> | ||||
|                   <span | ||||
|                     v-if="props.talk_type === 2" | ||||
|                     @click="toShowMessageReadDetail(item)" | ||||
|                     style="cursor: pointer;" | ||||
|                     style="cursor: pointer" | ||||
|                   > | ||||
|                     已读 ({{ item?.read_total_num || 0 }}/{{ | ||||
|                       props.num - 1 > 0 ? props.num - 1 : 0 | ||||
| @ -946,11 +957,12 @@ const onCustomSkipBottomEvent = () => { | ||||
|                     </n-tab> | ||||
|                   </n-tabs> | ||||
|                   <div class="talk-read-list"> | ||||
|                     <n-infinite-scroll style="height: 340px;" @load="loadMoreReadListDetail"> | ||||
|                     <n-infinite-scroll style="height: 340px" @load="loadMoreReadListDetail"> | ||||
|                       <div | ||||
|                         class="talk-read-list-item" | ||||
|                         v-for="(talkReadDetailItem, | ||||
|                         talkReadDetailIndex) in state.talkReadListDetail" | ||||
|                         v-for="( | ||||
|                           talkReadDetailItem, talkReadDetailIndex | ||||
|                         ) in state.talkReadListDetail" | ||||
|                         :key="talkReadDetailIndex" | ||||
|                       > | ||||
|                         <avatarModule | ||||
| @ -970,10 +982,10 @@ const onCustomSkipBottomEvent = () => { | ||||
|                           }" | ||||
|                         ></avatarModule> | ||||
|                         <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 | ||||
|                           }}</span> | ||||
|                           <span style="font-size: 12px; color: #999; line-height: 14px;">{{ | ||||
|                           <span style="font-size: 12px; color: #999; line-height: 14px">{{ | ||||
|                             talkReadDetailItem.jobNum | ||||
|                           }}</span> | ||||
|                         </div> | ||||
| @ -1005,7 +1017,7 @@ const onCustomSkipBottomEvent = () => { | ||||
|     :show="dropdown.show" | ||||
|     :x="dropdown.x" | ||||
|     :y="dropdown.y" | ||||
|     style="width: 142px;" | ||||
|     style="width: 142px" | ||||
|     :options="dropdown.options" | ||||
|     @select="onContextMenuHandle" | ||||
|     @clickoutside="closeDropdownMenu" | ||||
|  | ||||
| @ -16,7 +16,7 @@ import Editor from '@/components/editor/Editor.vue' | ||||
| import MultiSelectFooter from './MultiSelectFooter.vue' | ||||
| import HistoryRecord from '@/components/talk/HistoryRecord.vue' | ||||
| import {scrollToBottom} from '@/utils/dom.ts' | ||||
| import CustomEditor from '@/components/editor/CustomEditor.vue' | ||||
|  import CustomEditor from '@/components/editor/CustomEditor.vue' | ||||
| const userStore = useUserStore() | ||||
| const talkStore = useTalkStore() | ||||
| const editorStore = useEditorStore() | ||||
|  | ||||
| @ -48,25 +48,33 @@ export function useMenu() { | ||||
|     y: 0, | ||||
|     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) => { | ||||
|   //  dropdown.item = Object.assign({}, item)
 | ||||
|   dropdown.item = item | ||||
|   dropdown.item.is_self_action = true | ||||
|     dropdown.options = [] | ||||
|     if ([4].includes(item.msg_type)) { | ||||
|       if(item.is_convert_text === 1){ | ||||
|       if (item.is_convert_text === 1) { | ||||
|         dropdown.options.push({ label: '关闭转文字', key: 'closeConvertText' }) | ||||
|       }else{ | ||||
|       } else { | ||||
|         dropdown.options.push({ label: '转文字', key: 'convertText' }) | ||||
|       } | ||||
|      | ||||
|     } | ||||
|     if ([1, 3].includes(item.msg_type)) { | ||||
|       dropdown.options.push({ label: '复制', key: 'copy' }) | ||||
|     } | ||||
| 
 | ||||
|     dropdown.options.push({ label: '多选', key: 'multiSelect' }) | ||||
|     if (isOneMonthBefore(new Date(item.created_at.split(' ')[0]))) { | ||||
|       // 根据时间判断只有近一个月内的消息才能支持多选
 | ||||
|       dropdown.options.push({ label: '多选', key: 'multiSelect' }) | ||||
|     } | ||||
|     dropdown.options.push({ label: '引用', key: 'quote' }) | ||||
|     if (canAddRevokeOption(uid, item, (dialogueStore.groupInfo as any).is_manager)) { | ||||
|       dropdown.options.push({ label: '撤回', key: 'revoke' }); | ||||
| @ -74,8 +82,6 @@ export function useMenu() { | ||||
|     | ||||
|     dropdown.options.push({ label: '删除', key: 'delete' }) | ||||
| 
 | ||||
| 
 | ||||
| 
 | ||||
|     // if ([3, 4, 5].includes(item.msg_type)) {
 | ||||
|     //   dropdown.options.push({ label: '下载', key: 'download' })
 | ||||
|     // }
 | ||||
| @ -83,7 +89,6 @@ export function useMenu() { | ||||
|     // if ([3].includes(item.msg_type)) {
 | ||||
|     //   dropdown.options.push({ label: '收藏', key: 'collect' })
 | ||||
|     // }
 | ||||
|     | ||||
| 
 | ||||
|     dropdown.x = e.clientX | ||||
|     dropdown.y = e.clientY | ||||
| @ -95,5 +100,5 @@ export function useMenu() { | ||||
|     dropdown.item = {} | ||||
|   } | ||||
| 
 | ||||
|   return { dropdown, showDropdownMenu, closeDropdownMenu } | ||||
|   return { dropdown, showDropdownMenu, closeDropdownMenu, isOneMonthBefore } | ||||
| } | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user