Compare commits
	
		
			3 Commits
		
	
	
		
			a5d9009910
			...
			b7eea81200
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| b7eea81200 | |||
| df1cc68b59 | |||
| bbb5dd99fc | 
| @ -263,9 +263,20 @@ | |||||||
|                   @editorChange="onEditorChange" |                   @editorChange="onEditorChange" | ||||||
|                   style="width: 100%; flex: 1; height: 100%;" |                   style="width: 100%; flex: 1; height: 100%;" | ||||||
|                   @click="onEditorClick" |                   @click="onEditorClick" | ||||||
|  |                   v-if="state.canUseQuillEditor" | ||||||
|                 /> |                 /> | ||||||
|                 <!-- <tm-input type=textarea autoHeight focusColor="#F9F9F9" color="#F9F9F9" :inputPadding="[12]" |                 <tm-input | ||||||
|               placeholder=""></tm-input> --> |                   type="textarea" | ||||||
|  |                   autoHeight | ||||||
|  |                   focusColor="#F9F9F9" | ||||||
|  |                   color="#F9F9F9" | ||||||
|  |                   :inputPadding="[12]" | ||||||
|  |                   placeholder="" | ||||||
|  |                   v-if="!state.canUseQuillEditor" | ||||||
|  |                   @update:modelValue="onTextAreaChange" | ||||||
|  |                   v-model="state.textAreaValue" | ||||||
|  |                   @input="onTextAreaInput" | ||||||
|  |                 ></tm-input> | ||||||
|                 <div class="quote-area" v-if="state?.quoteInfo"> |                 <div class="quote-area" v-if="state?.quoteInfo"> | ||||||
|                   <span |                   <span | ||||||
|                     v-if="state?.quoteInfo?.msg_type === 1" |                     v-if="state?.quoteInfo?.msg_type === 1" | ||||||
| @ -326,7 +337,17 @@ | |||||||
|               </div> |               </div> | ||||||
|             </div> |             </div> | ||||||
|             <div v-if="state.isOpenEmojiPanel" class="mt-[50rpx]"> |             <div v-if="state.isOpenEmojiPanel" class="mt-[50rpx]"> | ||||||
|               <emojiPanel @on-select="onEmoticonEvent" /> |               <emojiPanel | ||||||
|  |                 @on-select=" | ||||||
|  |                   (data) => { | ||||||
|  |                     if (!state.canUseQuillEditor) { | ||||||
|  |                       onTextAreaEmoticon(data) | ||||||
|  |                     } else { | ||||||
|  |                       onEmoticonEvent(data) | ||||||
|  |                     } | ||||||
|  |                   } | ||||||
|  |                 " | ||||||
|  |               /> | ||||||
|             </div> |             </div> | ||||||
|             <div v-if="state.isOpenFilePanel" class="mt-[16rpx]"> |             <div v-if="state.isOpenFilePanel" class="mt-[16rpx]"> | ||||||
|               <filePanel |               <filePanel | ||||||
| @ -629,6 +650,8 @@ const state = ref({ | |||||||
|   lastMentionPosition: -1, // 添加新的状态来记录上一次@的位置 |   lastMentionPosition: -1, // 添加新的状态来记录上一次@的位置 | ||||||
|   isLoading: false, //发送按钮loading |   isLoading: false, //发送按钮loading | ||||||
|   lastSelection: 0, |   lastSelection: 0, | ||||||
|  |   canUseQuillEditor: true, //是否可以使用quill编辑器,如果版本不支持,则使用普通输入框 | ||||||
|  |   textAreaValue: '', //普通输入框的值 | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| uniOnload(async (options) => { | uniOnload(async (options) => { | ||||||
| @ -728,7 +751,78 @@ const onSendMessage = (data = {}, callBack, showLoading = false) => { | |||||||
|     }) |     }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | const onTextAreaChange = (value) => { | ||||||
|  |   state.value.textAreaValue = value | ||||||
|  | 
 | ||||||
|  |   // 更新mentionUserIds数组,确保与输入框中的@内容一致 | ||||||
|  |   const atPattern = /@([^@\s]+)(?:\s|$)/g | ||||||
|  |   const matches = Array.from(value.matchAll(atPattern)) | ||||||
|  |   const mentionedUsers = matches.map((match) => match[1].trim().toLowerCase()) | ||||||
|  | 
 | ||||||
|  |   // 获取当前对话中的所有成员信息 | ||||||
|  |   const members = dialogueStore.members || [] | ||||||
|  | 
 | ||||||
|  |   // 根据@的用户名找到对应的user_id | ||||||
|  |   const newMentionUserIds = mentionedUsers | ||||||
|  |     .map((username) => { | ||||||
|  |       if (username === '所有人') { | ||||||
|  |         return 0 // 如果是@所有人,返回0 | ||||||
|  |       } | ||||||
|  |       const member = members.find( | ||||||
|  |         (m) => m.nickname.trim().toLowerCase() === username, | ||||||
|  |       ) | ||||||
|  |       return member ? member.id : null | ||||||
|  |     }) | ||||||
|  |     .filter((id) => id !== null) | ||||||
|  | 
 | ||||||
|  |   // 只有在输入框中有@内容时才更新mentionUserIds | ||||||
|  |   if (value.includes('@')) { | ||||||
|  |     state.value.mentionUserIds = newMentionUserIds | ||||||
|  |   } else { | ||||||
|  |     state.value.mentionUserIds = [] | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 处理输入框输入事件 | ||||||
|  | const onTextAreaInput = (value) => { | ||||||
|  |   console.log(value, 'value') | ||||||
|  |   if (value.length > 0) { | ||||||
|  |     if (value[value.length - 1] === '@') { | ||||||
|  |       if (talkParams.type === 1) { | ||||||
|  |         return | ||||||
|  |       } | ||||||
|  |       state.value.isShowMentionSelect = true | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| const onSendMessageClick = () => { | const onSendMessageClick = () => { | ||||||
|  |   if (!state.value.canUseQuillEditor) { | ||||||
|  |     // 处理普通输入框的发送消息 | ||||||
|  |     if (state.value.textAreaValue.trim() === '') { | ||||||
|  |       return | ||||||
|  |     } | ||||||
|  |     if (state.value.textAreaValue.length > 1024) { | ||||||
|  |       return message.warning('发送内容超长,请分条发送') | ||||||
|  |     } | ||||||
|  |     let message = { | ||||||
|  |       type: 'text', | ||||||
|  |       content: state.value.textAreaValue, | ||||||
|  |       quote_id: state.value.quoteInfo ? state.value.quoteInfo.msg_id : null, // 添加quote_id | ||||||
|  |       mentions: state.value.mentionUserIds, // 使用最终的用户ID数组 | ||||||
|  |     } | ||||||
|  |     console.log(message) | ||||||
|  |     onSendMessage( | ||||||
|  |       message, | ||||||
|  |       (ok) => { | ||||||
|  |         if (!ok) return | ||||||
|  |         state.value.textAreaValue = '' | ||||||
|  |         state.value.quoteInfo = null // 发送后清除引用信息 | ||||||
|  |       }, | ||||||
|  |       true, | ||||||
|  |     ) | ||||||
|  |     return | ||||||
|  |   } | ||||||
|   // 发送前确保 mentionUserIds 是最新的 |   // 发送前确保 mentionUserIds 是最新的 | ||||||
|   updateMentionUserIds() |   updateMentionUserIds() | ||||||
| 
 | 
 | ||||||
| @ -879,8 +973,46 @@ const onEmoticonEvent = (data) => { | |||||||
|     emit('editor-event', fn) |     emit('editor-event', fn) | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | let calcDelta = false | ||||||
| const onEditorChange = () => { | const onEditorChange = () => { | ||||||
|  |   if(calcDelta){ | ||||||
|  |     calcDelta = false | ||||||
|  |     return | ||||||
|  |   } | ||||||
|  |   // 以下逻辑是 光标与@之间存在其他内容时 不触发选人弹窗 | ||||||
|  |   const qtext = getQuill().getText() | ||||||
|  |   let selectIdx = getQuill().getSelection().index | ||||||
|  |   const textBeforeCursor = qtext.substring(0, selectIdx) | ||||||
|  |   if (textBeforeCursor[0]?.charCodeAt(0) === 10) { | ||||||
|  |     const delta = getQuill().getContents() | ||||||
|  |     const ops = delta.ops || [] | ||||||
|  |     if (ops[0].insert === '\n') { | ||||||
|  |       ops.splice(0, 1) | ||||||
|  |       getQuill().setContents(delta) | ||||||
|  |       getQuill().setSelection(getQuill().getText().length, 0) | ||||||
|  |     } | ||||||
|  |     // for (let i = 0; i < ops.length; i++) { | ||||||
|  |     //   if (ops[i].insert === '\n') { | ||||||
|  |     //     console.error('有空格') | ||||||
|  |     //     ops.splice(i, 1) | ||||||
|  |     //     calcDelta = true | ||||||
|  |     //   } | ||||||
|  |     // } | ||||||
|  |     // getQuill().setContents(delta) | ||||||
|  |     // getQuill().setSelection(getQuill().getText().length, 0) | ||||||
|  |   } | ||||||
|  |   let endWithAt = false | ||||||
|  |   if ( | ||||||
|  |     textBeforeCursor[textBeforeCursor.length - 1] === '@' || | ||||||
|  |     textBeforeCursor[textBeforeCursor.length - 2]?.charCodeAt(0) === 64 | ||||||
|  |   ) { | ||||||
|  |     endWithAt = true | ||||||
|  |   } | ||||||
|  |   if (endWithAt) { | ||||||
|  |     state.value.cursorMention = true | ||||||
|  |   } else { | ||||||
|  |     state.value.cursorMention = false | ||||||
|  |   } | ||||||
|   if (getQuill().getText() !== state.value.lastMentionText) { |   if (getQuill().getText() !== state.value.lastMentionText) { | ||||||
|     state.value.lastMentionTriggered = false |     state.value.lastMentionTriggered = false | ||||||
|   } |   } | ||||||
| @ -1350,6 +1482,11 @@ const clearMultiSelect = () => { | |||||||
| 
 | 
 | ||||||
| const initData = async () => { | const initData = async () => { | ||||||
|   const dialogueList = getDialogueList(talkParams.index_name) |   const dialogueList = getDialogueList(talkParams.index_name) | ||||||
|  |   let doLocalPaging = false | ||||||
|  |   if(dialogueList?.records?.length > 0){ | ||||||
|  |     doLocalPaging = true | ||||||
|  |   } | ||||||
|  |   console.error('dialogueList', dialogueList) | ||||||
| 
 | 
 | ||||||
|   let objT = { |   let objT = { | ||||||
|     uid: talkParams.uid, |     uid: talkParams.uid, | ||||||
| @ -1360,7 +1497,9 @@ const initData = async () => { | |||||||
|     no_limit: dialogueList ? 1 : 0, |     no_limit: dialogueList ? 1 : 0, | ||||||
|   } |   } | ||||||
|   await onLoad({ ...objT }) |   await onLoad({ ...objT }) | ||||||
|   zpagingRef.value?.setLocalPaging(records.value) |   if(doLocalPaging){ | ||||||
|  |     zpagingRef.value?.setLocalPaging(dialogueList.records) | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| //点击跳转到聊天设置页面 | //点击跳转到聊天设置页面 | ||||||
| @ -1404,6 +1543,10 @@ const confirmMentionSelect = () => { | |||||||
|       selectMemberByAlphabetRef.value.confirmSelectMembers() |       selectMemberByAlphabetRef.value.confirmSelectMembers() | ||||||
|     } |     } | ||||||
|     hideMentionSelect() |     hideMentionSelect() | ||||||
|  | 
 | ||||||
|  |     // 重置选中人数和多选状态 | ||||||
|  |     state.value.selectedMembersNum = 0 | ||||||
|  |     state.value.mentionIsMulSelect = false | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1415,6 +1558,14 @@ const getSelectResult = (mentionSelect) => { | |||||||
| 
 | 
 | ||||||
| //处理要提醒人的消息样式 | //处理要提醒人的消息样式 | ||||||
| const getMentionSelectLists = (mentionSelectList) => { | const getMentionSelectLists = (mentionSelectList) => { | ||||||
|  |   if (!state.value.canUseQuillEditor) { | ||||||
|  |     if (mentionSelectList.length > 0) { | ||||||
|  |       mentionSelectList.forEach((item) => { | ||||||
|  |         onTextAreaMention(item) | ||||||
|  |       }) | ||||||
|  |     } | ||||||
|  |     return | ||||||
|  |   } | ||||||
|   const quill = getQuill() |   const quill = getQuill() | ||||||
|   const mention = quill.getModule('mention') |   const mention = quill.getModule('mention') | ||||||
| 
 | 
 | ||||||
| @ -1648,7 +1799,11 @@ let currentPressItem = null | |||||||
| const handleAvatarTouchStart = (item) => { | const handleAvatarTouchStart = (item) => { | ||||||
|   currentPressItem = item |   currentPressItem = item | ||||||
|   avatarPressTimer = setTimeout(() => { |   avatarPressTimer = setTimeout(() => { | ||||||
|  |     if (!state.value.canUseQuillEditor) { | ||||||
|  |       onTextAreaMention(item) | ||||||
|  |     } else { | ||||||
|       doMentionUser(item) |       doMentionUser(item) | ||||||
|  |     } | ||||||
|   }, 500) |   }, 500) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1661,6 +1816,13 @@ const handleAvatarTouchEnd = () => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| onMounted(async () => { | onMounted(async () => { | ||||||
|  |   if (uni.getSystemInfoSync().osName === 'ios') { | ||||||
|  |     let versions = uni.getSystemInfoSync().osVersion.split('.') | ||||||
|  |     if (Number(versions[0]) < 17) { | ||||||
|  |       console.error('ios版本低于17') | ||||||
|  |       state.value.canUseQuillEditor = false | ||||||
|  |     } | ||||||
|  |   } | ||||||
|   if (typeof plus !== 'undefined') { |   if (typeof plus !== 'undefined') { | ||||||
|     const webview = plus.webview.currentWebview() |     const webview = plus.webview.currentWebview() | ||||||
|     webview.setStyle({ |     webview.setStyle({ | ||||||
| @ -1716,24 +1878,10 @@ onUnmounted(() => { | |||||||
| // 修改防抖函数的实现 | // 修改防抖函数的实现 | ||||||
| const showMentionSelectDebounced = (quill) => { | const showMentionSelectDebounced = (quill) => { | ||||||
|   const text = quill.getText() |   const text = quill.getText() | ||||||
| 
 |   // 光标不在@后第一位时,不触发选人弹窗 | ||||||
|   // 以下逻辑是 光标与@之间存在其他内容时 不触发选人弹窗 |   if (!state.value.cursorMention) { | ||||||
|   const selection = quill.getSelection() |  | ||||||
|   if (selection) { |  | ||||||
|     state.lastSelection = selection.index |  | ||||||
|     if ( |  | ||||||
|       text[selection.index - 1].charCodeAt(0) !== 64 && |  | ||||||
|       text[selection.index - 1].charCodeAt(0) !== 32 && |  | ||||||
|       text[selection.index - 1].charCodeAt(0) !== 10 |  | ||||||
|     ) { |  | ||||||
|       uni.showToast({ |  | ||||||
|         title: text[selection.index - 1], |  | ||||||
|         icon: 'none', |  | ||||||
|       }) |  | ||||||
|       state.value.lastMentionTriggered = false |  | ||||||
|     return |     return | ||||||
|   } |   } | ||||||
|   } |  | ||||||
|   // 以下逻辑是 记录触发@时,用户的输入框内容,下次一样内容得@不会再次触发 |   // 以下逻辑是 记录触发@时,用户的输入框内容,下次一样内容得@不会再次触发 | ||||||
|   if (text !== state.value.lastMentionText) { |   if (text !== state.value.lastMentionText) { | ||||||
|     state.value.lastMentionTriggered = false |     state.value.lastMentionTriggered = false | ||||||
| @ -1749,6 +1897,45 @@ const showMentionSelectDebounced = (quill) => { | |||||||
|   state.value.isShowMentionSelect = true |   state.value.isShowMentionSelect = true | ||||||
|   // quill.blur() |   // quill.blur() | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | // 处理普通输入框的表情插入 | ||||||
|  | const onTextAreaEmoticon = (data) => { | ||||||
|  |   if (data.type == 1) { | ||||||
|  |     if (data.img) { | ||||||
|  |       // 如果是图片表情,插入对应的文本标记 | ||||||
|  |       state.value.textAreaValue += `${data.value}` | ||||||
|  |     } else { | ||||||
|  |       // 如果是文本表情,直接插入 | ||||||
|  |       state.value.textAreaValue += data.value | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // 处理普通输入框的长按@功能 | ||||||
|  | const onTextAreaMention = (user) => { | ||||||
|  |   if (talkParams.type === 1) { | ||||||
|  |     return | ||||||
|  |   } | ||||||
|  |   if (state.value.textAreaValue.length > 0) { | ||||||
|  |     if ( | ||||||
|  |       state.value.textAreaValue[state.value.textAreaValue.length - 1] === '@' | ||||||
|  |     ) { | ||||||
|  |       state.value.textAreaValue = state.value.textAreaValue.slice(0, -1) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   state.value.textAreaValue += `@${user.nickname} ` | ||||||
|  | 
 | ||||||
|  |   // 先创建新数组,再赋值 | ||||||
|  |   const newMentionUserIds = state.value.mentionUserIds | ||||||
|  |     ? [...state.value.mentionUserIds] | ||||||
|  |     : [] | ||||||
|  |   newMentionUserIds.push(user.nickname === '所有人' ? 0 : user.id) | ||||||
|  |   state.value.mentionUserIds = newMentionUserIds | ||||||
|  | 
 | ||||||
|  |   if (state.value.isShowMentionSelect) { | ||||||
|  |     state.value.isShowMentionSelect = false | ||||||
|  |   } | ||||||
|  | } | ||||||
| </script> | </script> | ||||||
| <style scoped lang="less"> | <style scoped lang="less"> | ||||||
| .dialog-page { | .dialog-page { | ||||||
| @ -2045,6 +2232,11 @@ const showMentionSelectDebounced = (quill) => { | |||||||
|       display: inline-block; |       display: inline-block; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
|  | 
 | ||||||
|  |   :deep(.round-3) { | ||||||
|  |     max-height: 320rpx; | ||||||
|  |     overflow-y: scroll; | ||||||
|  |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| :deep(.wd-action-sheet) { | :deep(.wd-action-sheet) { | ||||||
|  | |||||||
| @ -23,7 +23,7 @@ class WsSocket { | |||||||
|       lockReconnect: false, |       lockReconnect: false, | ||||||
|       setTimeout: null, // 计时器对象
 |       setTimeout: null, // 计时器对象
 | ||||||
|       time: 3000, // 重连间隔时间
 |       time: 3000, // 重连间隔时间
 | ||||||
|       number: 10000000 // 重连次数
 |       number: 20 // 重连次数
 | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user