Compare commits
	
		
			No commits in common. "b956b4ef79c1a20188eda6d781d45798591fa550" and "efd61b30f4ac92a010cb088364ee71c2b82892bb" have entirely different histories.
		
	
	
		
			b956b4ef79
			...
			efd61b30f4
		
	
		
							
								
								
									
										2
									
								
								env/.env.test
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								env/.env.test
									
									
									
									
										vendored
									
									
								
							| @ -7,6 +7,6 @@ VUE_APP_PREVIEW=false | |||||||
| #VITE_SOCKET_API=ws://192.168.88.21:9504 | #VITE_SOCKET_API=ws://192.168.88.21:9504 | ||||||
|  VITE_BASE_API=http://114.218.158.24:8503 |  VITE_BASE_API=http://114.218.158.24:8503 | ||||||
|  VITE_SOCKET_API=ws://114.218.158.24:8504 |  VITE_SOCKET_API=ws://114.218.158.24:8504 | ||||||
| VITE_EPR_BASEURL=http://172.16.100.93:8503 | VITE_EPR_BASEURL=http://114.218.158.24:9020 | ||||||
| VITE_PAGE_URL=http://172.16.100.93:9032 | VITE_PAGE_URL=http://172.16.100.93:9032 | ||||||
| VUE_APP_WEBSITE_NAME="" | VUE_APP_WEBSITE_NAME="" | ||||||
| @ -24,11 +24,9 @@ | |||||||
|     "@vicons/ionicons5": "^0.13.0", |     "@vicons/ionicons5": "^0.13.0", | ||||||
|     "@vueup/vue-quill": "^1.2.0", |     "@vueup/vue-quill": "^1.2.0", | ||||||
|     "@vueuse/core": "^10.7.0", |     "@vueuse/core": "^10.7.0", | ||||||
|     "@vueuse/rxjs": "^13.4.0", |  | ||||||
|     "ant-design-vue": "^4.2.6", |     "ant-design-vue": "^4.2.6", | ||||||
|     "axios": "^1.6.2", |     "axios": "^1.6.2", | ||||||
|     "dayjs": "^1.11.13", |     "dayjs": "^1.11.13", | ||||||
|     "dexie": "^4.0.11", |  | ||||||
|     "highlight.js": "^11.5.0", |     "highlight.js": "^11.5.0", | ||||||
|     "js-audio-recorder": "^1.0.7", |     "js-audio-recorder": "^1.0.7", | ||||||
|     "lodash-es": "^4.17.21", |     "lodash-es": "^4.17.21", | ||||||
| @ -38,7 +36,6 @@ | |||||||
|     "quill": "^1.3.7", |     "quill": "^1.3.7", | ||||||
|     "quill-image-uploader": "^1.3.0", |     "quill-image-uploader": "^1.3.0", | ||||||
|     "quill-mention": "^4.1.0", |     "quill-mention": "^4.1.0", | ||||||
|     "rxjs": "^7.8.2", |  | ||||||
|     "sortablejs": "^1.15.6", |     "sortablejs": "^1.15.6", | ||||||
|     "viewerjs": "^1.11.7", |     "viewerjs": "^1.11.7", | ||||||
|     "vue": "^3.3.11", |     "vue": "^3.3.11", | ||||||
|  | |||||||
							
								
								
									
										7385
									
								
								pnpm-lock.yaml
									
									
									
									
									
								
							
							
						
						
									
										7385
									
								
								pnpm-lock.yaml
									
									
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @ -1223,8 +1223,15 @@ const handleEditorClick = (event) => { | |||||||
|     <section class="el-container is-vertical"> |     <section class="el-container is-vertical"> | ||||||
|       <header class="el-header toolbar bdr-t"> |       <header class="el-header toolbar bdr-t"> | ||||||
|         <div class="tools pr-30px"> |         <div class="tools pr-30px"> | ||||||
|           <n-popover placement="top-start" trigger="click" raw :show-arrow="false" :width="300" ref="emoticonRef" |           <n-popover | ||||||
|             style="width: 500px; height: 250px; border-radius: 10px; overflow: hidden"> |             placement="top-start" | ||||||
|  |             trigger="click" | ||||||
|  |             raw | ||||||
|  |             :show-arrow="false" | ||||||
|  |             :width="300" | ||||||
|  |             ref="emoticonRef" | ||||||
|  |             style="width: 500px; height: 250px; border-radius: 10px; overflow: hidden" | ||||||
|  |           > | ||||||
|             <template #trigger> |             <template #trigger> | ||||||
|               <div class="item pointer"> |               <div class="item pointer"> | ||||||
|                 <n-icon size="18" class="icon" :component="SmilingFace" /> |                 <n-icon size="18" class="icon" :component="SmilingFace" /> | ||||||
| @ -1233,7 +1240,13 @@ const handleEditorClick = (event) => { | |||||||
|             </template> |             </template> | ||||||
|             <MeEditorEmoticon @on-select="onEmoticonEvent" /> |             <MeEditorEmoticon @on-select="onEmoticonEvent" /> | ||||||
|           </n-popover> |           </n-popover> | ||||||
|           <div class="item pointer" v-for="nav in navs" :key="nav.title" v-show="nav.show" @click="nav.click"> |           <div | ||||||
|  |             class="item pointer" | ||||||
|  |             v-for="nav in navs" | ||||||
|  |             :key="nav.title" | ||||||
|  |             v-show="nav.show" | ||||||
|  |             @click="nav.click" | ||||||
|  |           > | ||||||
|             <n-icon size="18" class="icon" :component="nav.icon" /> |             <n-icon size="18" class="icon" :component="nav.icon" /> | ||||||
|             <p class="tip-title">{{ nav.title }}</p> |             <p class="tip-title">{{ nav.title }}</p> | ||||||
|           </div> |           </div> | ||||||
| @ -1248,15 +1261,31 @@ const handleEditorClick = (event) => { | |||||||
|         </div> |         </div> | ||||||
|       </header> |       </header> | ||||||
|       <main class="el-main height100"> |       <main class="el-main height100"> | ||||||
|         <div ref="editorRef" class="custom-editor" contenteditable="true" :placeholder="placeholder" |         <div | ||||||
|           @input="handleInput" @keydown="handleKeydown" @paste="handlePaste" @focus="handleFocus" @blur="handleBlur"> |           ref="editorRef" | ||||||
|         </div> |           class="custom-editor" | ||||||
|         <div v-if="showMention && dialogueStore.talk.talk_type === 2" class="mention-list py-5px" |           contenteditable="true" | ||||||
|           :style="{ top: mentionPosition.top + 'px', left: mentionPosition.left + 'px' }"> |           :placeholder="placeholder" | ||||||
|  |           @input="handleInput" | ||||||
|  |           @keydown="handleKeydown" | ||||||
|  |           @paste="handlePaste" | ||||||
|  |           @focus="handleFocus" | ||||||
|  |           @blur="handleBlur" | ||||||
|  |         ></div> | ||||||
|  |         <div | ||||||
|  |           v-if="showMention && dialogueStore.talk.talk_type === 2" | ||||||
|  |           class="mention-list py-5px" | ||||||
|  |           :style="{ top: mentionPosition.top + 'px', left: mentionPosition.left + 'px' }" | ||||||
|  |         > | ||||||
|           <ul class="max-h-140px w-163px overflow-auto hide-scrollbar"> |           <ul class="max-h-140px w-163px overflow-auto hide-scrollbar"> | ||||||
|             <li v-for="(member, index) in mentionList" :key="member.user_id || member.id" |             <li | ||||||
|               class="cursor-pointer px-14px h-42px" :class="{ 'bg-#EEE9F9': index === selectedMentionIndex }" |               v-for="(member, index) in mentionList" | ||||||
|               @mousedown.prevent="handleMentionSelectByMouse(member)" @mouseover="selectedMentionIndex = index"> |               :key="member.user_id || member.id" | ||||||
|  |               class="cursor-pointer px-14px h-42px" | ||||||
|  |               :class="{ 'bg-#EEE9F9': index === selectedMentionIndex }" | ||||||
|  |               @mousedown.prevent="handleMentionSelectByMouse(member)" | ||||||
|  |               @mouseover="selectedMentionIndex = index" | ||||||
|  |             > | ||||||
|               <div class="flex items-center border-b-1px border-b-solid border-b-#F8F8F8 h-full"> |               <div class="flex items-center border-b-1px border-b-solid border-b-#F8F8F8 h-full"> | ||||||
|                 <img class="w-26px h-26px rounded-50% mr-11px" :src="member.avatar" alt=""> |                 <img class="w-26px h-26px rounded-50% mr-11px" :src="member.avatar" alt=""> | ||||||
|                 <span>{{ member.nickname }}</span> |                 <span>{{ member.nickname }}</span> | ||||||
| @ -1276,17 +1305,14 @@ const handleEditorClick = (event) => { | |||||||
| .editor { | .editor { | ||||||
|   --tip-bg-color: rgb(241 241 241 / 90%); |   --tip-bg-color: rgb(241 241 241 / 90%); | ||||||
|   height: 100%; |   height: 100%; | ||||||
| 
 |  | ||||||
|   .toolbar { |   .toolbar { | ||||||
|     height: 38px; |     height: 38px; | ||||||
|     display: flex; |     display: flex; | ||||||
| 
 |  | ||||||
|     .tools { |     .tools { | ||||||
|       height: 40px; |       height: 40px; | ||||||
|       flex: auto; |       flex: auto; | ||||||
|       display: flex; |       display: flex; | ||||||
|       align-items: center; |       align-items: center; | ||||||
| 
 |  | ||||||
|       .item { |       .item { | ||||||
|         display: flex; |         display: flex; | ||||||
|         align-items: center; |         align-items: center; | ||||||
| @ -1295,7 +1321,6 @@ const handleEditorClick = (event) => { | |||||||
|         margin: 0 2px; |         margin: 0 2px; | ||||||
|         position: relative; |         position: relative; | ||||||
|         user-select: none;  |         user-select: none;  | ||||||
| 
 |  | ||||||
|         .tip-title { |         .tip-title { | ||||||
|           display: none; |           display: none; | ||||||
|           position: absolute; |           position: absolute; | ||||||
| @ -1312,7 +1337,6 @@ const handleEditorClick = (event) => { | |||||||
|           user-select: none; |           user-select: none; | ||||||
|           z-index: 999999999999;  |           z-index: 999999999999;  | ||||||
|         } |         } | ||||||
| 
 |  | ||||||
|         &:hover { |         &:hover { | ||||||
|           .tip-title { |           .tip-title { | ||||||
|             display: block; |             display: block; | ||||||
| @ -1321,7 +1345,6 @@ const handleEditorClick = (event) => { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   :deep(.editor-file) { |   :deep(.editor-file) { | ||||||
|     display: inline-block; |     display: inline-block; | ||||||
|     padding: 5px 10px; |     padding: 5px 10px; | ||||||
| @ -1338,7 +1361,6 @@ const handleEditorClick = (event) => { | |||||||
|     overflow: hidden; |     overflow: hidden; | ||||||
|     text-overflow: ellipsis;  |     text-overflow: ellipsis;  | ||||||
|     white-space: nowrap;  |     white-space: nowrap;  | ||||||
| 
 |  | ||||||
|     &::after { |     &::after { | ||||||
|       content: attr(data-size); |       content: attr(data-size); | ||||||
|       position: absolute; |       position: absolute; | ||||||
| @ -1346,12 +1368,10 @@ const handleEditorClick = (event) => { | |||||||
|       color: #757575; |       color: #757575; | ||||||
|       font-size: 12px; |       font-size: 12px; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     &:hover { |     &:hover { | ||||||
|       background-color: #e3f2fd; |       background-color: #e3f2fd; | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   :deep(.editor-emoji) { |   :deep(.editor-emoji) { | ||||||
|     display: inline-block; |     display: inline-block; | ||||||
|     width: 24px; |     width: 24px; | ||||||
| @ -1359,7 +1379,6 @@ const handleEditorClick = (event) => { | |||||||
|     vertical-align: middle;  |     vertical-align: middle;  | ||||||
|     margin: 0 2px; |     margin: 0 2px; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   :deep(.editor-quote) { |   :deep(.editor-quote) { | ||||||
|     margin-bottom: 8px; |     margin-bottom: 8px; | ||||||
|     padding: 8px 12px; |     padding: 8px 12px; | ||||||
| @ -1375,22 +1394,18 @@ const handleEditorClick = (event) => { | |||||||
|     cursor: pointer;  |     cursor: pointer;  | ||||||
|     user-select: none;  |     user-select: none;  | ||||||
|     transition: background-color 0.2s ease;  |     transition: background-color 0.2s ease;  | ||||||
| 
 |  | ||||||
|     &:hover { |     &:hover { | ||||||
|       background-color: var(--im-message-left-bg-hover-color, #eaeaea); |       background-color: var(--im-message-left-bg-hover-color, #eaeaea); | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     .quote-content-wrapper { |     .quote-content-wrapper { | ||||||
|       flex: 1; |       flex: 1; | ||||||
|       overflow: hidden; |       overflow: hidden; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     .quote-title { |     .quote-title { | ||||||
|       color: var(--im-primary-color, #409eff); |       color: var(--im-primary-color, #409eff); | ||||||
|       margin-bottom: 4px; |       margin-bottom: 4px; | ||||||
|       font-weight: 500; |       font-weight: 500; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     .quote-content { |     .quote-content { | ||||||
|       color: var(--im-text-color, #333); |       color: var(--im-text-color, #333); | ||||||
|       word-break: break-all;  |       word-break: break-all;  | ||||||
| @ -1401,14 +1416,12 @@ const handleEditorClick = (event) => { | |||||||
|       -webkit-line-clamp: 2;  |       -webkit-line-clamp: 2;  | ||||||
|       -webkit-box-orient: vertical; |       -webkit-box-orient: vertical; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     .quote-image img { |     .quote-image img { | ||||||
|       max-width: 100px; |       max-width: 100px; | ||||||
|       max-height: 60px; |       max-height: 60px; | ||||||
|       border-radius: 3px; |       border-radius: 3px; | ||||||
|       pointer-events: none;  |       pointer-events: none;  | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     .quote-close { |     .quote-close { | ||||||
|       width: 18px; |       width: 18px; | ||||||
|       height: 18px; |       height: 18px; | ||||||
| @ -1421,14 +1434,12 @@ const handleEditorClick = (event) => { | |||||||
|       font-size: 16px; |       font-size: 16px; | ||||||
|       margin-left: 8px; |       margin-left: 8px; | ||||||
|       user-select: none; |       user-select: none; | ||||||
| 
 |  | ||||||
|       &:hover { |       &:hover { | ||||||
|         background-color: rgba(0, 0, 0, 0.2); |         background-color: rgba(0, 0, 0, 0.2); | ||||||
|         color: #333; |         color: #333; | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .custom-editor { |   .custom-editor { | ||||||
|     width: 100%; |     width: 100%; | ||||||
|     height: 100%; |     height: 100%; | ||||||
| @ -1441,53 +1452,44 @@ const handleEditorClick = (event) => { | |||||||
|     color: #333; |     color: #333; | ||||||
|     background: transparent; |     background: transparent; | ||||||
|     overflow-y: auto; |     overflow-y: auto; | ||||||
| 
 |  | ||||||
|     &:empty:before { |     &:empty:before { | ||||||
|       content: attr(placeholder); |       content: attr(placeholder); | ||||||
|       color: #999; |       color: #999; | ||||||
|       pointer-events: none; |       pointer-events: none; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     &::-webkit-scrollbar { |     &::-webkit-scrollbar { | ||||||
|       width: 3px; |       width: 3px; | ||||||
|       height: 3px; |       height: 3px; | ||||||
|       background-color: unset; |       background-color: unset; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     &::-webkit-scrollbar-thumb { |     &::-webkit-scrollbar-thumb { | ||||||
|       border-radius: 3px; |       border-radius: 3px; | ||||||
|       background-color: transparent; |       background-color: transparent; | ||||||
|     } |     } | ||||||
| 
 |  | ||||||
|     &:hover { |     &:hover { | ||||||
|       &::-webkit-scrollbar-thumb { |       &::-webkit-scrollbar-thumb { | ||||||
|         background-color: var(--im-scrollbar-thumb); |         background-color: var(--im-scrollbar-thumb); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .custom-editor:empty::before { |   .custom-editor:empty::before { | ||||||
|     content: attr(placeholder); |     content: attr(placeholder); | ||||||
|     color: #999; |     color: #999; | ||||||
|     pointer-events: none;  |     pointer-events: none;  | ||||||
|     font-family: PingFang SC, Microsoft YaHei, 'Alibaba PuHuiTi 2.0 45' !important; |     font-family: PingFang SC, Microsoft YaHei, 'Alibaba PuHuiTi 2.0 45' !important; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .custom-editor:focus { |   .custom-editor:focus { | ||||||
|     outline: none; |     outline: none; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .mention:hover { |   .mention:hover { | ||||||
|     background-color: #bae7ff; |     background-color: #bae7ff; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .editor-emoji { |   .editor-emoji { | ||||||
|     width: 20px; |     width: 20px; | ||||||
|     height: 20px; |     height: 20px; | ||||||
|     vertical-align: middle; |     vertical-align: middle; | ||||||
|     margin: 0 2px; |     margin: 0 2px; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .mention-list { |   .mention-list { | ||||||
|     position: absolute; |     position: absolute; | ||||||
|     background-color: white; |     background-color: white; | ||||||
| @ -1496,7 +1498,6 @@ const handleEditorClick = (event) => { | |||||||
|     box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1); |     box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1); | ||||||
|     z-index: 1000; |     z-index: 1000; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .quote-card { |   .quote-card { | ||||||
|     background: #f5f5f5; |     background: #f5f5f5; | ||||||
|     border-left: 3px solid #1890ff; |     border-left: 3px solid #1890ff; | ||||||
| @ -1505,22 +1506,18 @@ const handleEditorClick = (event) => { | |||||||
|     border-radius: 4px; |     border-radius: 4px; | ||||||
|     position: relative; |     position: relative; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .quote-content { |   .quote-content { | ||||||
|     font-size: 12px; |     font-size: 12px; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .quote-title { |   .quote-title { | ||||||
|     font-weight: bold; |     font-weight: bold; | ||||||
|     color: #1890ff; |     color: #1890ff; | ||||||
|     margin-bottom: 4px; |     margin-bottom: 4px; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .quote-text { |   .quote-text { | ||||||
|     color: #666; |     color: #666; | ||||||
|     line-height: 1.4; |     line-height: 1.4; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .quote-close { |   .quote-close { | ||||||
|     position: absolute; |     position: absolute; | ||||||
|     top: 4px; |     top: 4px; | ||||||
| @ -1531,7 +1528,6 @@ const handleEditorClick = (event) => { | |||||||
|     color: #999; |     color: #999; | ||||||
|     font-size: 12px; |     font-size: 12px; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .edit-tip { |   .edit-tip { | ||||||
|     background: #fff7e6; |     background: #fff7e6; | ||||||
|     border: 1px solid #ffd591; |     border: 1px solid #ffd591; | ||||||
| @ -1544,7 +1540,6 @@ const handleEditorClick = (event) => { | |||||||
|     align-items: center; |     align-items: center; | ||||||
|     gap: 8px; |     gap: 8px; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   .edit-tip button { |   .edit-tip button { | ||||||
|     background: none; |     background: none; | ||||||
|     border: none; |     border: none; | ||||||
| @ -1553,13 +1548,11 @@ const handleEditorClick = (event) => { | |||||||
|     margin-left: auto; |     margin-left: auto; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 |  | ||||||
| html[theme-mode='dark'] { | html[theme-mode='dark'] { | ||||||
|   .editor { |   .editor { | ||||||
|     --tip-bg-color: #48484d; |     --tip-bg-color: #48484d; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 |  | ||||||
| :deep(.editor-image-wrapper.image-upload-loading::before) { | :deep(.editor-image-wrapper.image-upload-loading::before) { | ||||||
|   content: ''; |   content: ''; | ||||||
|   position: absolute; |   position: absolute; | ||||||
| @ -1575,23 +1568,19 @@ html[theme-mode='dark'] { | |||||||
|   animation: spin 0.6s linear infinite; |   animation: spin 0.6s linear infinite; | ||||||
|   z-index: 1; |   z-index: 1; | ||||||
| } | } | ||||||
| 
 |  | ||||||
| :deep(.editor-image-wrapper.image-upload-loading img) { | :deep(.editor-image-wrapper.image-upload-loading img) { | ||||||
|   opacity: 0.5; |   opacity: 0.5; | ||||||
| } | } | ||||||
| 
 |  | ||||||
| @keyframes spin { | @keyframes spin { | ||||||
|   to { |   to { | ||||||
|     transform: rotate(360deg); |     transform: rotate(360deg); | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 |  | ||||||
| .hide-scrollbar { | .hide-scrollbar { | ||||||
|   &::-webkit-scrollbar { |   &::-webkit-scrollbar { | ||||||
|     width: 0; |     width: 0; | ||||||
|     display: none; |     display: none; | ||||||
|   } |   } | ||||||
| 
 |  | ||||||
|   scrollbar-width: none; |   scrollbar-width: none; | ||||||
|   -ms-overflow-style: none; |   -ms-overflow-style: none; | ||||||
| } | } | ||||||
|  | |||||||
| @ -32,7 +32,7 @@ class Read extends Base { | |||||||
| 
 | 
 | ||||||
|   handle() { |   handle() { | ||||||
|     if (this.type == 'total') { |     if (this.type == 'total') { | ||||||
|    |       console.error('====接收到了新版已读回执全量=====', this.resource) | ||||||
|       const readList = this.resource.result |       const readList = this.resource.result | ||||||
|       if (readList.length > 0) { |       if (readList.length > 0) { | ||||||
|         readList.forEach((item) => { |         readList.forEach((item) => { | ||||||
|  | |||||||
| @ -227,13 +227,14 @@ class Talk extends Base { | |||||||
|         }) |         }) | ||||||
|       }, 1000) |       }, 1000) | ||||||
|     } |     } | ||||||
|     console.log('输出加载1') | 
 | ||||||
|     // 获取聊天面板元素节点
 |     // 获取聊天面板元素节点
 | ||||||
|     const el = document.getElementById('imChatPanel') |     const el = document.getElementById('imChatPanel') | ||||||
|     if (!el) return |     if (!el) return | ||||||
| 
 | 
 | ||||||
|     // 判断的滚动条是否在底部
 |     // 判断的滚动条是否在底部
 | ||||||
|     const isBottom = isScrollAtBottom(el) |     const isBottom = isScrollAtBottom(el) | ||||||
|  | 
 | ||||||
|     if (isBottom || record.user_id == this.getAccountId()) { |     if (isBottom || record.user_id == this.getAccountId()) { | ||||||
|       scrollToBottom() |       scrollToBottom() | ||||||
|     } else { |     } else { | ||||||
|  | |||||||
| @ -130,19 +130,19 @@ export const useTalkRecord = (uid: number) => { | |||||||
|       cursor: loadConfig.cursor, |       cursor: loadConfig.cursor, | ||||||
|       limit: 30 |       limit: 30 | ||||||
|     } |     } | ||||||
|     // 如果不是从本地数据库加载的,则设置加载状态为0(加载中)
 | 
 | ||||||
|     if (loadConfig.status !== 2 && loadConfig.status !== 3) { |  | ||||||
|     loadConfig.status = 0 |     loadConfig.status = 0 | ||||||
|     } |  | ||||||
| 
 | 
 | ||||||
|     let scrollHeight = 0 |     let scrollHeight = 0 | ||||||
|  |     console.log('加载数据列表load') | ||||||
|     const el = document.getElementById('imChatPanel') |     const el = document.getElementById('imChatPanel') | ||||||
|     if (el) { |     if (el) { | ||||||
|       scrollHeight = el.scrollHeight |       scrollHeight = el.scrollHeight | ||||||
|     } |     } | ||||||
|  | 
 | ||||||
|     const { data, code } = await ServeTalkRecords(request) |     const { data, code } = await ServeTalkRecords(request) | ||||||
|     if (code != 200) { |     if (code != 200) { | ||||||
|       return (loadConfig.status = (loadConfig.status === 2 || loadConfig.status === 3) ? loadConfig.status : 1) // 如果已经从本地加载了数据,保持原状态
 |       return (loadConfig.status = 1) | ||||||
|     } |     } | ||||||
|     // 防止对话切换过快,数据渲染错误
 |     // 防止对话切换过快,数据渲染错误
 | ||||||
|     if ( |     if ( | ||||||
| @ -154,64 +154,6 @@ export const useTalkRecord = (uid: number) => { | |||||||
| 
 | 
 | ||||||
|     const items = (data.items || []).map((item: ITalkRecord) => formatTalkRecord(uid, item)) |     const items = (data.items || []).map((item: ITalkRecord) => formatTalkRecord(uid, item)) | ||||||
| 
 | 
 | ||||||
|     // 同步到本地数据库
 |  | ||||||
|     try { |  | ||||||
|       const { batchAddOrUpdateMessages } = await import('@/utils/db') |  | ||||||
|       await batchAddOrUpdateMessages(data.items || [], params.talk_type, params.receiver_id, true, 'sequence') |  | ||||||
|       console.log('聊天记录已同步到本地数据库') |  | ||||||
|     } catch (error) { |  | ||||||
|       console.error('同步聊天记录到本地数据库失败:', error) |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // 如果是从本地数据库加载的数据,且服务器返回的数据与本地数据相同,则不需要更新UI
 |  | ||||||
|     if ((loadConfig.status === 2 || loadConfig.status === 3) && request.cursor === 0) { |  | ||||||
|       try { |  | ||||||
|         // 获取最新的本地数据库消息进行比较
 |  | ||||||
|         const { getMessages } = await import('@/utils/db') |  | ||||||
|         const localMessages = await getMessages( |  | ||||||
|           params.talk_type, |  | ||||||
|           uid, |  | ||||||
|           params.receiver_id, |  | ||||||
|           items.length || 30, // 获取与服务器返回数量相同的消息
 |  | ||||||
|           0 // 从第一页开始
 |  | ||||||
|         ) |  | ||||||
|          |  | ||||||
|         // 格式化本地消息,确保与服务器消息结构一致
 |  | ||||||
|         const formattedLocalMessages = localMessages.map((item: ITalkRecord) => formatTalkRecord(uid, item)) |  | ||||||
|     |  | ||||||
|          |  | ||||||
|         // 改进比较逻辑:检查消息数量和所有消息的ID是否匹配
 |  | ||||||
|         if (formattedLocalMessages.length === items.length && formattedLocalMessages.length > 0) { |  | ||||||
|           // 创建消息ID映射,用于快速查找
 |  | ||||||
|           const serverMsgMap = new Map() |  | ||||||
|           items.forEach(item => serverMsgMap.set(item.msg_id, item)) |  | ||||||
|            |  | ||||||
|           // 检查每条本地消息是否与服务器消息匹配
 |  | ||||||
|           const allMatch = formattedLocalMessages.every(localMsg => { |  | ||||||
|             const serverMsg = serverMsgMap.get(localMsg.msg_id) |  | ||||||
|             // 检查消息是否存在且关键状态是否一致(考虑撤回、已读等状态变化)
 |  | ||||||
|             return serverMsg &&  |  | ||||||
|                    serverMsg.is_revoke === localMsg.is_revoke &&  |  | ||||||
|                    serverMsg.is_read === localMsg.is_read &&  |  | ||||||
|                    (serverMsg.send_status === localMsg.send_status ||  |  | ||||||
|                     (!serverMsg.send_status && !localMsg.send_status)) && |  | ||||||
|                    serverMsg.content === localMsg.content |  | ||||||
|           }) |  | ||||||
|            |  | ||||||
|           if (allMatch) { |  | ||||||
|             console.log('本地数据与服务器数据一致,无需更新UI') |  | ||||||
|             return |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         // 数据不一致,需要更新UI
 |  | ||||||
|         console.log('本地数据与服务器数据不一致,更新UI') |  | ||||||
|       } catch (error) { |  | ||||||
|         console.error('比较本地数据和服务器数据时出错:', error) |  | ||||||
|         // 出错时默认更新UI
 |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (request.cursor == 0) { |     if (request.cursor == 0) { | ||||||
|       // 判断是否是初次加载
 |       // 判断是否是初次加载
 | ||||||
|       dialogueStore.clearDialogueRecord() |       dialogueStore.clearDialogueRecord() | ||||||
| @ -225,6 +167,7 @@ export const useTalkRecord = (uid: number) => { | |||||||
| 
 | 
 | ||||||
|     nextTick(() => { |     nextTick(() => { | ||||||
|       const el = document.getElementById('imChatPanel') |       const el = document.getElementById('imChatPanel') | ||||||
|  | 
 | ||||||
|       if (el) { |       if (el) { | ||||||
|         if (request.cursor == 0) { |         if (request.cursor == 0) { | ||||||
|           // el.scrollTop = el.scrollHeight
 |           // el.scrollTop = el.scrollHeight
 | ||||||
| @ -232,12 +175,6 @@ export const useTalkRecord = (uid: number) => { | |||||||
|           // setTimeout(() => {
 |           // setTimeout(() => {
 | ||||||
|           //   el.scrollTop = el.scrollHeight + 1000
 |           //   el.scrollTop = el.scrollHeight + 1000
 | ||||||
|           // }, 500)
 |           // }, 500)
 | ||||||
|           console.log('滚动到底部') |  | ||||||
|            |  | ||||||
|           // 在初次加载完成后恢复上传任务
 |  | ||||||
|           // 确保在所有聊天记录加载完成后再恢复上传任务
 |  | ||||||
|           dialogueStore.restoreUploadTasks() |  | ||||||
|            |  | ||||||
|           scrollToBottom() |           scrollToBottom() | ||||||
|         } else { |         } else { | ||||||
|           el.scrollTop = el.scrollHeight - scrollHeight |           el.scrollTop = el.scrollHeight - scrollHeight | ||||||
| @ -252,7 +189,9 @@ export const useTalkRecord = (uid: number) => { | |||||||
| 
 | 
 | ||||||
|   // 获取当前消息的最小 sequence
 |   // 获取当前消息的最小 sequence
 | ||||||
|   const getMinSequence = () => { |   const getMinSequence = () => { | ||||||
|  |     console.error('records.value', records.value) | ||||||
|     if (!records.value.length) return 0 |     if (!records.value.length) return 0 | ||||||
|  |     console.error(Math.min(...records.value.map((item) => item.sequence))) | ||||||
|     return Math.min(...records.value.map((item) => item.sequence)) |     return Math.min(...records.value.map((item) => item.sequence)) | ||||||
|   } |   } | ||||||
|   // 获取当前消息的最大 sequence
 |   // 获取当前消息的最大 sequence
 | ||||||
| @ -261,56 +200,13 @@ export const useTalkRecord = (uid: number) => { | |||||||
|     return Math.max(...records.value.map((item) => item.sequence)) |     return Math.max(...records.value.map((item) => item.sequence)) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // 从本地数据库加载聊天记录
 |  | ||||||
|   const loadFromLocalDB = async (params: Params) => { |  | ||||||
|     try { |  | ||||||
|       // 导入 getMessages 函数
 |  | ||||||
|       const { getMessages } = await import('@/utils/db') |  | ||||||
|       // 从本地数据库获取聊天记录
 |  | ||||||
|       const localMessages = await getMessages( |  | ||||||
|         params.talk_type, |  | ||||||
|         uid, |  | ||||||
|         params.receiver_id, |  | ||||||
|         params.limit || 30, |  | ||||||
|         0 // 从第一页开始
 |  | ||||||
|         // 不传入 maxSequence 参数,获取最新的消息
 |  | ||||||
|       ) |  | ||||||
|       // 如果有本地数据
 |  | ||||||
|       if (localMessages && localMessages.length > 0) { |  | ||||||
|         // 清空现有记录
 |  | ||||||
|         dialogueStore.clearDialogueRecord() |  | ||||||
|          |  | ||||||
|         // 格式化并添加记录
 |  | ||||||
|         const formattedMessages = localMessages.map((item: ITalkRecord) => formatTalkRecord(uid, item)) |  | ||||||
|         dialogueStore.unshiftDialogueRecord(formattedMessages) |  | ||||||
|          |  | ||||||
|         // 设置加载状态为完成(3表示从本地数据库加载完成)
 |  | ||||||
|         loadConfig.status = 3 |  | ||||||
|          |  | ||||||
|         // 恢复上传任务
 |  | ||||||
|         dialogueStore.restoreUploadTasks() |  | ||||||
|          |  | ||||||
|         // 滚动到底部
 |  | ||||||
|         nextTick(() => { |  | ||||||
|           scrollToBottom() |  | ||||||
|         }) |  | ||||||
|          |  | ||||||
|         return true |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       return false |  | ||||||
|     } catch (error) { |  | ||||||
|       console.error('从本地数据库加载聊天记录失败:', error) |  | ||||||
|       return false |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   /** |   /** | ||||||
|    * 加载数据主入口,支持指定消息定位模式 |    * 加载数据主入口,支持指定消息定位模式 | ||||||
|    * @param params 原有参数 |    * @param params 原有参数 | ||||||
|    * @param options 可选,{ specifiedMsg } 指定消息对象 |    * @param options 可选,{ specifiedMsg } 指定消息对象 | ||||||
|    */ |    */ | ||||||
|   const onLoad = async (params: Params, options?: LoadOptions) => { |   const onLoad = (params: Params, options?: LoadOptions) => { | ||||||
|  |     // 如果会话切换,重置所有状态
 | ||||||
|     if ( |     if ( | ||||||
|       params.talk_type !== loadConfig.talk_type || |       params.talk_type !== loadConfig.talk_type || | ||||||
|       params.receiver_id !== loadConfig.receiver_id |       params.receiver_id !== loadConfig.receiver_id | ||||||
| @ -325,6 +221,7 @@ export const useTalkRecord = (uid: number) => { | |||||||
|     // 新增:支持指定消息定位模式,参数以传入为准合并
 |     // 新增:支持指定消息定位模式,参数以传入为准合并
 | ||||||
|     if (options?.specifiedMsg?.cursor !== undefined) { |     if (options?.specifiedMsg?.cursor !== undefined) { | ||||||
|       loadConfig.specialParams = { ...options.specifiedMsg } // 记录特殊参数,供分页加载用
 |       loadConfig.specialParams = { ...options.specifiedMsg } // 记录特殊参数,供分页加载用
 | ||||||
|  |       console.error('options', options) | ||||||
|       loadConfig.status = 0 // 复用主流程 loading 状态
 |       loadConfig.status = 0 // 复用主流程 loading 状态
 | ||||||
|       // 以 params 为基础,合并 specifiedMsg 的所有字段(只要有就覆盖)
 |       // 以 params 为基础,合并 specifiedMsg 的所有字段(只要有就覆盖)
 | ||||||
|       const contextParams = { |       const contextParams = { | ||||||
| @ -334,7 +231,6 @@ export const useTalkRecord = (uid: number) => { | |||||||
|       //msg_id是用来做定位的,不做参数,所以这里清空
 |       //msg_id是用来做定位的,不做参数,所以这里清空
 | ||||||
|       contextParams.msg_id = '' |       contextParams.msg_id = '' | ||||||
|       ServeTalkRecords(contextParams).then(({ data, code }) => { |       ServeTalkRecords(contextParams).then(({ data, code }) => { | ||||||
|         console.log('data',data) |  | ||||||
|         if (code !== 200) { |         if (code !== 200) { | ||||||
|           loadConfig.status = 2 |           loadConfig.status = 2 | ||||||
|           return |           return | ||||||
| @ -426,8 +322,6 @@ export const useTalkRecord = (uid: number) => { | |||||||
|               }) |               }) | ||||||
|             } else { |             } else { | ||||||
|               // 其他情况滚动到底部
 |               // 其他情况滚动到底部
 | ||||||
|               // 在特殊参数模式下也需要恢复上传任务
 |  | ||||||
|               dialogueStore.restoreUploadTasks() |  | ||||||
|               scrollToBottom() |               scrollToBottom() | ||||||
|             } |             } | ||||||
|           } |           } | ||||||
| @ -437,22 +331,14 @@ export const useTalkRecord = (uid: number) => { | |||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     loadConfig.specialParams = undefined // 普通模式清空
 |     loadConfig.specialParams = undefined // 普通模式清空
 | ||||||
|      |  | ||||||
|     // 设置初始加载状态为0(加载中)
 |  | ||||||
|     loadConfig.status = 0 |  | ||||||
|      |  | ||||||
|     // 先从本地数据库加载数据
 |  | ||||||
|     const hasLocalData = await loadFromLocalDB(params) |  | ||||||
|      |  | ||||||
|     // 无论是否有本地数据,都从服务器获取最新数据
 |  | ||||||
|     // 原有逻辑
 |     // 原有逻辑
 | ||||||
|     console.log('onLoad()执行load') |  | ||||||
|     load(params) |     load(params) | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // 向上加载更多(兼容特殊参数模式)
 |   // 向上加载更多(兼容特殊参数模式)
 | ||||||
|   const onRefreshLoad = () => { |   const onRefreshLoad = () => { | ||||||
|     if (loadConfig.status == 1 || loadConfig.status == 3) { |     console.error('loadConfig.status', loadConfig.status) | ||||||
|  |     if (loadConfig.status == 1) { | ||||||
|       console.log('specialParams', loadConfig.specialParams) |       console.log('specialParams', loadConfig.specialParams) | ||||||
|       // 判断是否是特殊参数模式
 |       // 判断是否是特殊参数模式
 | ||||||
|       if (loadConfig.specialParams && typeof loadConfig.specialParams === 'object') { |       if (loadConfig.specialParams && typeof loadConfig.specialParams === 'object') { | ||||||
| @ -483,7 +369,6 @@ export const useTalkRecord = (uid: number) => { | |||||||
|         } else { |         } else { | ||||||
|           // 如果不匹配,重置为普通模式
 |           // 如果不匹配,重置为普通模式
 | ||||||
|           resetLoadConfig() |           resetLoadConfig() | ||||||
|           console.log('load执行2') |  | ||||||
|           load({ |           load({ | ||||||
|             receiver_id: loadConfig.receiver_id, |             receiver_id: loadConfig.receiver_id, | ||||||
|             talk_type: loadConfig.talk_type, |             talk_type: loadConfig.talk_type, | ||||||
| @ -492,7 +377,6 @@ export const useTalkRecord = (uid: number) => { | |||||||
|         } |         } | ||||||
|       } else { |       } else { | ||||||
|         // 原有逻辑
 |         // 原有逻辑
 | ||||||
|                console.log('load执行3') |  | ||||||
|         load({ |         load({ | ||||||
|           receiver_id: loadConfig.receiver_id, |           receiver_id: loadConfig.receiver_id, | ||||||
|           talk_type: loadConfig.talk_type, |           talk_type: loadConfig.talk_type, | ||||||
|  | |||||||
| @ -8,13 +8,11 @@ import router from './router' | |||||||
| import App from './App.vue' | import App from './App.vue' | ||||||
| import * as plugins from './plugins' | import * as plugins from './plugins' | ||||||
| import request from "@/api/index.js"; | import request from "@/api/index.js"; | ||||||
| 
 |  | ||||||
| if (window.__POWERED_BY_WUJIE__) { | if (window.__POWERED_BY_WUJIE__) { | ||||||
|   // eslint-disable-next-line
 |   // eslint-disable-next-line
 | ||||||
|   window.__webpack_public_path__ = window.__WUJIE_PUBLIC_PATH__; |   window.__webpack_public_path__ = window.__WUJIE_PUBLIC_PATH__; | ||||||
| } | } | ||||||
| async function bootstrap() { | async function bootstrap() { | ||||||
| 
 |  | ||||||
|   const app = createApp(App) |   const app = createApp(App) | ||||||
| 
 | 
 | ||||||
|   app.use(router) |   app.use(router) | ||||||
|  | |||||||
| @ -15,9 +15,7 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
|     return { |     return { | ||||||
|       // 对话索引(聊天对话的唯一索引)
 |       // 对话索引(聊天对话的唯一索引)
 | ||||||
|       index_name: '', |       index_name: '', | ||||||
|       globalUploadList:[], | 
 | ||||||
|       // 添加一个映射,用于快速查找每个会话的上传任务
 |  | ||||||
|       uploadTaskMap: {},  // 格式: { "talk_type_receiver_id": [task1, task2, ...] }
 |  | ||||||
|       // 对话节点
 |       // 对话节点
 | ||||||
|       talk: { |       talk: { | ||||||
|         avatar:'', |         avatar:'', | ||||||
| @ -131,10 +129,8 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
|       if (data.talk_type == 2) { |       if (data.talk_type == 2) { | ||||||
|         this.updateGroupMembers() |         this.updateGroupMembers() | ||||||
|         this.getGroupInfo() |         this.getGroupInfo() | ||||||
|       } |  | ||||||
| 
 | 
 | ||||||
|       // 注意:上传任务的恢复将在聊天记录加载完成后进行
 |       } | ||||||
|       // 在useTalkRecord.ts的onLoad方法中,会在加载完聊天记录后调用restoreUploadTasks方法
 |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 更新提及列表
 |     // 更新提及列表
 | ||||||
| @ -175,12 +171,10 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
| 
 | 
 | ||||||
|     // 数组头部压入对话记录
 |     // 数组头部压入对话记录
 | ||||||
|     unshiftDialogueRecord(records) { |     unshiftDialogueRecord(records) { | ||||||
|       console.log('unshiftDialogueRecord') |  | ||||||
|       this.records.unshift(...records) |       this.records.unshift(...records) | ||||||
|     }, |     }, | ||||||
|     //数组尾部加入更多对话记录
 |     //数组尾部加入更多对话记录
 | ||||||
|     addDialogueRecordForLoadMore(records){ |     addDialogueRecordForLoadMore(records){ | ||||||
|             console.log('addDialogueRecordForLoadMore') |  | ||||||
|       this.records.push(...records) |       this.records.push(...records) | ||||||
|     }, |     }, | ||||||
|     async getGroupInfo(){ |     async getGroupInfo(){ | ||||||
| @ -192,55 +186,24 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|     // 推送对话记录
 |     // 推送对话记录
 | ||||||
|     async addDialogueRecord(record) { |     addDialogueRecord(record) { | ||||||
|       // TOOD 需要通过 sequence 排序,保证消息一致性
 |       // TOOD 需要通过 sequence 排序,保证消息一致性
 | ||||||
|       // this.records.splice(index, 0, record)
 |       // this.records.splice(index, 0, record)
 | ||||||
|       this.records.push(record) |  | ||||||
| 
 | 
 | ||||||
|       // 同步到本地数据库
 |       this.records.push(record) | ||||||
|       try { |  | ||||||
|         const { addMessage } = await import('@/utils/db') |  | ||||||
|         await addMessage(record) |  | ||||||
|       } catch (error) { |  | ||||||
|         console.error('同步消息到本地数据库失败:', error) |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 更新对话记录
 |     // 更新对话记录
 | ||||||
|     async updateDialogueRecord(params) { |     updateDialogueRecord(params) { | ||||||
|       const { msg_id = '' } = params |       const { msg_id = '' } = params | ||||||
| 
 | 
 | ||||||
|       const item = this.records.find((item) => item.msg_id === msg_id) |       const item = this.records.find((item) => item.msg_id === msg_id) | ||||||
| 
 | 
 | ||||||
|       if (item) { |       item && Object.assign(item, params) | ||||||
|         Object.assign(item, params) |  | ||||||
|          |  | ||||||
|         // 同步到本地数据库
 |  | ||||||
|         try { |  | ||||||
|           // 如果是撤回消息
 |  | ||||||
|           if (params.is_revoke === 1) { |  | ||||||
|             const { revokeMessage } = await import('@/utils/db') |  | ||||||
|             await revokeMessage(msg_id) |  | ||||||
|           } |  | ||||||
|         } catch (error) { |  | ||||||
|           console.error('同步消息更新到本地数据库失败:', error) |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 批量删除对话记录
 |     // 批量删除对话记录
 | ||||||
|     async batchDelDialogueRecord(msgIds = []) { |     batchDelDialogueRecord(msgIds = []) { | ||||||
|       // 同步到本地数据库
 |  | ||||||
|       try { |  | ||||||
|         const { deleteMessage } = await import('@/utils/db') |  | ||||||
|         for (const msgid of msgIds) { |  | ||||||
|           await deleteMessage(msgid) |  | ||||||
|         } |  | ||||||
|       } catch (error) { |  | ||||||
|         console.error('同步消息删除到本地数据库失败:', error) |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       // 从内存中删除
 |  | ||||||
|       msgIds.forEach((msgid) => { |       msgIds.forEach((msgid) => { | ||||||
|         const index = this.records.findIndex((item) => item.msg_id === msgid) |         const index = this.records.findIndex((item) => item.msg_id === msgid) | ||||||
| 
 | 
 | ||||||
| @ -329,16 +292,6 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
| 
 | 
 | ||||||
|     // 更新视频上传进度
 |     // 更新视频上传进度
 | ||||||
|     updateUploadProgress(uploadId, percentage) { |     updateUploadProgress(uploadId, percentage) { | ||||||
|       // 更新全局列表中的进度
 |  | ||||||
|       const globalTask = this.globalUploadList.find(item =>  |  | ||||||
|         item.extra && item.extra.is_uploading && item.extra.upload_id === uploadId |  | ||||||
|       ) |  | ||||||
|        |  | ||||||
|       if (globalTask) { |  | ||||||
|         globalTask.extra.percentage = percentage |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       // 更新当前会话记录中的进度
 |  | ||||||
|       const record = this.records.find(item =>  |       const record = this.records.find(item =>  | ||||||
|         item.extra && item.extra.is_uploading && item.extra.upload_id === uploadId |         item.extra && item.extra.is_uploading && item.extra.upload_id === uploadId | ||||||
|       ) |       ) | ||||||
| @ -348,44 +301,6 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|      |      | ||||||
|     // 添加上传任务
 |  | ||||||
|     addUploadTask(task) { |  | ||||||
|       // 添加到全局列表
 |  | ||||||
|       this.globalUploadList.push(task) |  | ||||||
|        |  | ||||||
|       // 添加到会话映射
 |  | ||||||
|       const sessionKey = `${task.talk_type}_${task.receiver_id}` |  | ||||||
|       if (!this.uploadTaskMap[sessionKey]) { |  | ||||||
|         this.uploadTaskMap[sessionKey] = [] |  | ||||||
|       } |  | ||||||
|       this.uploadTaskMap[sessionKey].push(task) |  | ||||||
|        |  | ||||||
|       // 同时添加到当前会话记录
 |  | ||||||
|       this.addDialogueRecord(task) |  | ||||||
|     }, |  | ||||||
|      |  | ||||||
|     // 上传完成后移除任务
 |  | ||||||
|     removeUploadTask(uploadId) { |  | ||||||
|       // 从全局列表中找到任务
 |  | ||||||
|       const taskIndex = this.globalUploadList.findIndex(item => item.msg_id === uploadId) |  | ||||||
|        |  | ||||||
|       if (taskIndex >= 0) { |  | ||||||
|         const task = this.globalUploadList[taskIndex] |  | ||||||
|         const sessionKey = `${task.talk_type}_${task.receiver_id}` |  | ||||||
|          |  | ||||||
|         // 从会话映射中移除
 |  | ||||||
|         if (this.uploadTaskMap[sessionKey]) { |  | ||||||
|           const mapIndex = this.uploadTaskMap[sessionKey].findIndex(item => item.msg_id === uploadId) |  | ||||||
|           if (mapIndex >= 0) { |  | ||||||
|             this.uploadTaskMap[sessionKey].splice(mapIndex, 1) |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         // 从全局列表中移除
 |  | ||||||
|         this.globalUploadList.splice(taskIndex, 1) |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|      |  | ||||||
|     // 视频上传完成后更新消息
 |     // 视频上传完成后更新消息
 | ||||||
|     completeUpload(uploadId, videoInfo) { |     completeUpload(uploadId, videoInfo) { | ||||||
|       const record = this.records.find(item =>  |       const record = this.records.find(item =>  | ||||||
| @ -402,135 +317,6 @@ export const useDialogueStore = defineStore('dialogue', { | |||||||
|     // 更新会话信息
 |     // 更新会话信息
 | ||||||
|     updateDialogueTalk(params){ |     updateDialogueTalk(params){ | ||||||
|       Object.assign(this.talk, params) |       Object.assign(this.talk, params) | ||||||
|     }, |  | ||||||
|      |  | ||||||
|     // 根据 insert_sequence 将任务插入到 records 数组的正确位置(使用优化的二分查找)
 |  | ||||||
|     insertTaskAtCorrectPosition(task) { |  | ||||||
|       const len = this.records.length |  | ||||||
|        |  | ||||||
|       // 快速路径:如果数组为空或任务应该插入到末尾
 |  | ||||||
|       if (len === 0) { |  | ||||||
|         this.records.push(task) |  | ||||||
|         return |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       // 快速路径:检查是否应该插入到开头或末尾(避免二分查找的开销)
 |  | ||||||
|       if (task.insert_sequence < this.records[0].sequence) { |  | ||||||
|         this.records.unshift(task) |  | ||||||
|         return |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       if (task.insert_sequence >= this.records[len - 1].sequence) { |  | ||||||
|         this.records.push(task) |  | ||||||
|         return |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       // 使用优化的二分查找算法找到插入位置
 |  | ||||||
|       let low = 0 |  | ||||||
|       let high = len - 1 |  | ||||||
|        |  | ||||||
|       // 二分查找优化:使用位运算加速计算中点
 |  | ||||||
|       while (low <= high) { |  | ||||||
|         const mid = (low + high) >>> 1 // 无符号右移代替 Math.floor((low + high) / 2)
 |  | ||||||
|         if (this.records[mid].sequence <= task.insert_sequence) { |  | ||||||
|           low = mid + 1 |  | ||||||
|         } else { |  | ||||||
|           high = mid - 1 |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       // 在找到的位置插入任务
 |  | ||||||
|       this.records.splice(low, 0, task) |  | ||||||
|     }, |  | ||||||
|      |  | ||||||
|     // 恢复当前会话的上传任务
 |  | ||||||
|     restoreUploadTasks() { |  | ||||||
|       // 获取当前会话的sessionKey
 |  | ||||||
|       const sessionKey = `${this.talk.talk_type}_${this.talk.receiver_id}` |  | ||||||
|        |  | ||||||
|       // 检查是否有需要恢复的上传任务
 |  | ||||||
|       if (!this.uploadTaskMap[sessionKey] || this.uploadTaskMap[sessionKey].length === 0) { |  | ||||||
|         return |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       // 性能优化:缓存数组长度和本地变量,减少属性查找
 |  | ||||||
|       const tasks = this.uploadTaskMap[sessionKey] |  | ||||||
|       const tasksLength = tasks.length |  | ||||||
|        |  | ||||||
|       // 如果只有一个任务,直接处理
 |  | ||||||
|       if (tasksLength === 1) { |  | ||||||
|         this.insertTaskAtCorrectPosition(tasks[0]) |  | ||||||
|         return |  | ||||||
|       } |  | ||||||
|        |  | ||||||
|       // 性能优化:对于少量任务,避免创建新数组和排序开销
 |  | ||||||
|       if (tasksLength <= 10) { |  | ||||||
|         // 找出最小的 insert_sequence
 |  | ||||||
|         let minIndex = 0 |  | ||||||
|         for (let i = 1; i < tasksLength; i++) { |  | ||||||
|           if (tasks[i].insert_sequence < tasks[minIndex].insert_sequence) { |  | ||||||
|             minIndex = i |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         // 按顺序插入任务
 |  | ||||||
|         let inserted = 0 |  | ||||||
|         let currentMin = tasks[minIndex] |  | ||||||
|         this.insertTaskAtCorrectPosition(currentMin) |  | ||||||
|         inserted++ |  | ||||||
|          |  | ||||||
|         while (inserted < tasksLength) { |  | ||||||
|           minIndex = -1 |  | ||||||
|           let minSequence = Infinity |  | ||||||
|            |  | ||||||
|           // 找出剩余任务中 insert_sequence 最小的
 |  | ||||||
|           for (let i = 0; i < tasksLength; i++) { |  | ||||||
|             const task = tasks[i] |  | ||||||
|             if (task !== currentMin && task.insert_sequence < minSequence) { |  | ||||||
|               minIndex = i |  | ||||||
|               minSequence = task.insert_sequence |  | ||||||
|             } |  | ||||||
|           } |  | ||||||
|            |  | ||||||
|           if (minIndex !== -1) { |  | ||||||
|             currentMin = tasks[minIndex] |  | ||||||
|             this.insertTaskAtCorrectPosition(currentMin) |  | ||||||
|             inserted++ |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       } else { |  | ||||||
|         // 对于大量任务,使用排序后批量处理
 |  | ||||||
|         // 创建一个新数组并排序,避免修改原数组
 |  | ||||||
|         const sortedTasks = [...tasks].sort((a, b) => a.insert_sequence - b.insert_sequence) |  | ||||||
|          |  | ||||||
|         // 性能优化:使用 requestAnimationFrame 进行批处理,更好地配合浏览器渲染周期
 |  | ||||||
|         const batchSize = 50 // 每批处理的任务数量
 |  | ||||||
|         const totalBatches = Math.ceil(sortedTasks.length / batchSize) |  | ||||||
|          |  | ||||||
|         const processBatch = (batchIndex) => { |  | ||||||
|           const startIndex = batchIndex * batchSize |  | ||||||
|           const endIndex = Math.min(startIndex + batchSize, sortedTasks.length) |  | ||||||
|            |  | ||||||
|           // 处理当前批次的任务
 |  | ||||||
|           for (let i = startIndex; i < endIndex; i++) { |  | ||||||
|             this.insertTaskAtCorrectPosition(sortedTasks[i]) |  | ||||||
|           } |  | ||||||
|            |  | ||||||
|           // 如果还有更多批次,安排下一个批次
 |  | ||||||
|           if (batchIndex < totalBatches - 1) { |  | ||||||
|             // 使用 requestAnimationFrame 配合浏览器渲染周期
 |  | ||||||
|             // 如果不支持,回退到 setTimeout
 |  | ||||||
|             if (typeof requestAnimationFrame !== 'undefined') { |  | ||||||
|               requestAnimationFrame(() => processBatch(batchIndex + 1)) |  | ||||||
|             } else { |  | ||||||
|               setTimeout(() => processBatch(batchIndex + 1), 0) |  | ||||||
|             } |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         // 开始处理第一批
 |  | ||||||
|         processBatch(0) |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| }) | }) | ||||||
|  | |||||||
| @ -3,7 +3,6 @@ import { ServeGetTalkList, ServeCreateTalkList } from '@/api/chat' | |||||||
| import { formatTalkItem, ttime, KEY_INDEX_NAME } from '@/utils/talk' | import { formatTalkItem, ttime, KEY_INDEX_NAME } from '@/utils/talk' | ||||||
| import { useEditorDraftStore } from './editor-draft' | import { useEditorDraftStore } from './editor-draft' | ||||||
| import { ISession } from '@/types/chat' | import { ISession } from '@/types/chat' | ||||||
| import { getConversations, addOrUpdateConversation, deleteConversation, getConversation } from '@/utils/db' |  | ||||||
| 
 | 
 | ||||||
| interface TalkStoreState { | interface TalkStoreState { | ||||||
|   loadStatus: number |   loadStatus: number | ||||||
| @ -46,103 +45,56 @@ export const useTalkStore = defineStore('talk', { | |||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 更新对话节点
 |     // 更新对话节点
 | ||||||
|     async updateItem(params: any) { |     updateItem(params: any) { | ||||||
|       const item = this.items.find((item) => item.index_name === params.index_name) |       const item = this.items.find((item) => item.index_name === params.index_name) | ||||||
| 
 | 
 | ||||||
|       if (item) { |       item && Object.assign(item, params) | ||||||
|         Object.assign(item, params) |  | ||||||
|          |  | ||||||
|         // 同步更新本地数据库
 |  | ||||||
|         try { |  | ||||||
|           await addOrUpdateConversation(JSON.parse(JSON.stringify(item))) |  | ||||||
|         } catch (error) { |  | ||||||
|           console.error('更新本地会话失败:', error) |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 新增对话节点
 |     // 新增对话节点
 | ||||||
|     async addItem(params: any) { |     addItem(params: any) { | ||||||
|       this.items = [params, ...this.items] |       this.items = [params, ...this.items] | ||||||
|        |  | ||||||
|       // 同步添加到本地数据库
 |  | ||||||
|       try { |  | ||||||
|         await addOrUpdateConversation(JSON.parse(JSON.stringify(params))) |  | ||||||
|       } catch (error) { |  | ||||||
|         console.error('添加本地会话失败:', error) |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 移除对话节点
 |     // 移除对话节点
 | ||||||
|     async delItem(index_name: string) { |     delItem(index_name: string) { | ||||||
|       const i = this.items.findIndex((item) => item.index_name === index_name) |       const i = this.items.findIndex((item) => item.index_name === index_name) | ||||||
| 
 | 
 | ||||||
|       if (i >= 0) { |       if (i >= 0) { | ||||||
|         const item = this.items[i] |  | ||||||
|         this.items.splice(i, 1) |         this.items.splice(i, 1) | ||||||
|          |  | ||||||
|         // 同步从本地数据库删除
 |  | ||||||
|         try { |  | ||||||
|           // 从本地数据库中查找并删除会话
 |  | ||||||
|           const [talkType, receiverId] = index_name.split('_') |  | ||||||
|           const conversation = await getConversation(Number(talkType), Number(receiverId)) |  | ||||||
|            |  | ||||||
|           if (conversation && conversation.id) { |  | ||||||
|             await deleteConversation(conversation.id, false) // 不删除相关消息
 |  | ||||||
|           } |  | ||||||
|         } catch (error) { |  | ||||||
|           console.error('删除本地会话失败:', error) |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       this.items = [...this.items] |       this.items = [...this.items] | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 更新对话消息
 |     // 更新对话消息
 | ||||||
|     async updateMessage(params: any) { |     updateMessage(params: any) { | ||||||
|       const item = this.items.find((item) => item.index_name === params.index_name) |       const item = this.items.find((item) => item.index_name === params.index_name) | ||||||
| 
 | 
 | ||||||
|       if (item) { |       if (item) { | ||||||
|         item.unread_num++ |         item.unread_num++ | ||||||
|         item.msg_text = params.msg_text |         item.msg_text = params.msg_text | ||||||
|         item.updated_at = params.updated_at |         item.updated_at = params.updated_at | ||||||
|          |  | ||||||
|         // 同步更新本地数据库中的会话信息
 |  | ||||||
|         try { |  | ||||||
|           await addOrUpdateConversation(JSON.parse(JSON.stringify(item))) |  | ||||||
|         } catch (error) { |  | ||||||
|           console.error('更新本地会话消息失败:', error) |  | ||||||
|         } |  | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 更新联系人备注
 |     // 更新联系人备注
 | ||||||
|     async setRemark(params: any) { |     setRemark(params: any) { | ||||||
|       const item = this.items.find((item) => item.index_name === `1_${params.user_id}`) |       const item = this.items.find((item) => item.index_name === `1_${params.user_id}`) | ||||||
| 
 | 
 | ||||||
|       if (item) { |       item && (item.remark = params.remark) | ||||||
|         item.remark = params.remark |  | ||||||
|          |  | ||||||
|         // 同步更新本地数据库
 |  | ||||||
|         try { |  | ||||||
|           await addOrUpdateConversation(JSON.parse(JSON.stringify(item))) |  | ||||||
|         } catch (error) { |  | ||||||
|           console.error('更新本地联系人备注失败:', error) |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     // 加载会话列表
 |     // 加载会话列表
 | ||||||
|     async loadTalkList() { |     loadTalkList() { | ||||||
|       this.loadStatus = 2 |       this.loadStatus = 2 | ||||||
| 
 | 
 | ||||||
|       try { |       const resp = ServeGetTalkList() | ||||||
|         // 先从本地数据库加载会话列表
 | 
 | ||||||
|         const localConversations = await getConversations() |       resp.then(({ code, data }) => { | ||||||
|         if (localConversations && localConversations.length > 0) { |         if (code == 200) { | ||||||
|           // 将本地会话列表转换为应用所需格式
 | 
 | ||||||
|           this.items = localConversations.map((item: any) => { |           this.items = data.items.map((item: any) => { | ||||||
|             // 确保本地存储的会话格式与应用一致
 |  | ||||||
|             const value = formatTalkItem(item) |             const value = formatTalkItem(item) | ||||||
| 
 | 
 | ||||||
|             const draft = useEditorDraftStore().items[value.index_name] |             const draft = useEditorDraftStore().items[value.index_name] | ||||||
| @ -156,63 +108,22 @@ export const useTalkStore = defineStore('talk', { | |||||||
|             return value |             return value | ||||||
|           }) |           }) | ||||||
| 
 | 
 | ||||||
|           // 设置为加载完成状态,因为已从本地加载了数据,不需要等待服务器数据就可以显示
 |  | ||||||
|           this.loadStatus = 3 |           this.loadStatus = 3 | ||||||
|  |         } else { | ||||||
|  |           this.loadStatus = 4 | ||||||
|         } |         } | ||||||
| 
 |  | ||||||
|         // 从服务器获取最新会话列表
 |  | ||||||
|         const resp = await ServeGetTalkList() |  | ||||||
| 
 |  | ||||||
|         if (resp.code == 200) { |  | ||||||
|           // 将服务器返回的会话列表转换为应用所需格式
 |  | ||||||
|           const serverItems = resp.data.items.map((item: any) => { |  | ||||||
|             const value = formatTalkItem(item) |  | ||||||
| 
 |  | ||||||
|             const draft = useEditorDraftStore().items[value.index_name] |  | ||||||
|             if (draft) { |  | ||||||
|               value.draft_text = JSON.parse(draft).text || '' |  | ||||||
|             } |  | ||||||
| 
 |  | ||||||
|             if (value.is_robot == 1) { |  | ||||||
|               value.is_online = 1 |  | ||||||
|             } |  | ||||||
|             return value |  | ||||||
|       }) |       }) | ||||||
| 
 | 
 | ||||||
|           // 更新状态和本地数据库
 |       resp.catch(() => { | ||||||
|           this.items = serverItems |  | ||||||
|            |  | ||||||
|           // 将最新的会话列表保存到本地数据库
 |  | ||||||
|           for (const item of serverItems) { |  | ||||||
|             await addOrUpdateConversation(item) |  | ||||||
|           } |  | ||||||
| 
 |  | ||||||
|           this.loadStatus = 3 |  | ||||||
|         } else { |  | ||||||
|           // 如果服务器请求失败但本地有数据,保持使用本地数据
 |  | ||||||
|           if (this.items.length === 0) { |  | ||||||
|         this.loadStatus = 4 |         this.loadStatus = 4 | ||||||
|           } else { |       }) | ||||||
|             this.loadStatus = 3 |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       } catch (error) { |  | ||||||
|         console.error('加载会话列表失败:', error) |  | ||||||
|          |  | ||||||
|         // 如果有本地数据,即使服务器请求失败也显示本地数据
 |  | ||||||
|         if (this.items.length === 0) { |  | ||||||
|           this.loadStatus = 4 |  | ||||||
|         } else { |  | ||||||
|           this.loadStatus = 3 |  | ||||||
|         } |  | ||||||
|       } |  | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     findTalkIndex(index_name: string) { |     findTalkIndex(index_name: string) { | ||||||
|       return this.items.findIndex((item: ISession) => item.index_name === index_name) |       return this.items.findIndex((item: ISession) => item.index_name === index_name) | ||||||
|     }, |     }, | ||||||
| 
 | 
 | ||||||
|     async toTalk(talk_type: number, receiver_id: number, router: any) { |     toTalk(talk_type: number, receiver_id: number, router: any) { | ||||||
|       const route = { |       const route = { | ||||||
|         path: '/message', |         path: '/message', | ||||||
|         query: { |         query: { | ||||||
| @ -225,31 +136,13 @@ export const useTalkStore = defineStore('talk', { | |||||||
|         return router.push(route) |         return router.push(route) | ||||||
|       } |       } | ||||||
| 
 | 
 | ||||||
|       try { |       ServeCreateTalkList({ | ||||||
|         // 先检查本地数据库中是否有该会话
 |  | ||||||
|         const localConversation = await getConversation(talk_type, receiver_id) |  | ||||||
|          |  | ||||||
|         if (localConversation) { |  | ||||||
|           // 如果本地有该会话,直接添加到列表中
 |  | ||||||
|           if (this.findTalkIndex(`${talk_type}_${receiver_id}`) === -1) { |  | ||||||
|             this.addItem(formatTalkItem(localConversation)) |  | ||||||
|           } |  | ||||||
|            |  | ||||||
|           sessionStorage.setItem(KEY_INDEX_NAME, `${talk_type}_${receiver_id}`) |  | ||||||
|           return router.push(route) |  | ||||||
|         } |  | ||||||
|          |  | ||||||
|         // 如果本地没有,则从服务器创建
 |  | ||||||
|         const { code, data, message } = await ServeCreateTalkList({ |  | ||||||
|         talk_type, |         talk_type, | ||||||
|         receiver_id |         receiver_id | ||||||
|         }) |       }).then(({ code, data, message }) => { | ||||||
|          |  | ||||||
|         if (code == 200) { |         if (code == 200) { | ||||||
|           const formattedItem = formatTalkItem(data) |  | ||||||
|            |  | ||||||
|           if (this.findTalkIndex(`${talk_type}_${receiver_id}`) === -1) { |           if (this.findTalkIndex(`${talk_type}_${receiver_id}`) === -1) { | ||||||
|             await this.addItem(formattedItem) // 使用 await 确保本地数据库同步更新
 |             this.addItem(formatTalkItem(data)) | ||||||
|           } |           } | ||||||
| 
 | 
 | ||||||
|           sessionStorage.setItem(KEY_INDEX_NAME, `${talk_type}_${receiver_id}`) |           sessionStorage.setItem(KEY_INDEX_NAME, `${talk_type}_${receiver_id}`) | ||||||
| @ -257,10 +150,7 @@ export const useTalkStore = defineStore('talk', { | |||||||
|         } else { |         } else { | ||||||
|           window['$message'].info(message) |           window['$message'].info(message) | ||||||
|         } |         } | ||||||
|       } catch (error) { |       }) | ||||||
|         console.error('创建会话失败:', error) |  | ||||||
|         window['$message'].error('创建会话失败,请稍后再试') |  | ||||||
|       } |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| }) | }) | ||||||
|  | |||||||
| @ -1,13 +1,7 @@ | |||||||
| import { defineStore } from 'pinia' | import { defineStore } from 'pinia' | ||||||
| // import { message } from 'naive-ui'
 | import { ServeFindFileSplitInfo, ServeFileSubareaUpload } from '@/api/upload' | ||||||
| import { | import { ServeSendTalkFile } from '@/api/chat' | ||||||
|   ServeSendTalkFile | import { uploadImg } from '@/api/upload' | ||||||
| } from '@/api/chat' |  | ||||||
| import {  |  | ||||||
|   uploadImg, |  | ||||||
|   ServeFindFileSplitInfo, |  | ||||||
|   ServeFileSubareaUpload  |  | ||||||
| } from '@/api/upload' |  | ||||||
| import { | import { | ||||||
|   useDialogueStore |   useDialogueStore | ||||||
| } from '@/store' | } from '@/store' | ||||||
| @ -146,12 +140,12 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|           this.triggerUpload(upload_id, clientUploadId) |           this.triggerUpload(upload_id, clientUploadId) | ||||||
|         } else { |         } else { | ||||||
|           message.error(res.message) |           message.error(res.message) | ||||||
|           this.handleUploadError(upload_id, clientUploadId) |           onProgress(-1) // 通知上传失败
 | ||||||
|         } |         } | ||||||
|       } catch (error) { |       } catch (error) { | ||||||
|         console.error("初始化分片上传失败:", error); |         console.error("初始化分片上传失败:", error); | ||||||
|         message.error("初始化上传失败,请重试") |         message.error("初始化上传失败,请重试") | ||||||
|         this.handleUploadError(upload_id, clientUploadId) |         onProgress(-1) | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|      |      | ||||||
| @ -207,20 +201,26 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|             this.triggerUpload(uploadId, clientUploadId) |             this.triggerUpload(uploadId, clientUploadId) | ||||||
|           } |           } | ||||||
|         } else { |         } else { | ||||||
|  |           updatedItem.onProgress(-1) | ||||||
|           // 上传失败处理
 |           // 上传失败处理
 | ||||||
|           console.error(`分片上传失败,错误码: ${res.code},错误信息: ${res.message || '未知错误'}`); |           console.error(`分片上传失败,错误码: ${res.code},错误信息: ${res.message || '未知错误'}`); | ||||||
|           this.handleUploadError(uploadId, clientUploadId || '') |           updatedItem.status = 3 | ||||||
|  |            | ||||||
|  |          | ||||||
|         } |         } | ||||||
|       } catch (error) { |       } catch (error) { | ||||||
|  |         updatedItem.onProgress(-1) | ||||||
|         console.error("分片上传错误:", error); |         console.error("分片上传错误:", error); | ||||||
|          |          | ||||||
|         // 获取最新的项目状态
 |         // 获取最新的项目状态
 | ||||||
|  |         // 这里不应该重新定义变量,而是使用已有的updatedItem
 | ||||||
|  |         // const updatedItem = this.findItem(uploadId)
 | ||||||
|         if (!updatedItem) return |         if (!updatedItem) return | ||||||
|          |          | ||||||
|         // 如果是暂停导致的错误,不改变状态
 |         // 如果是暂停导致的错误,不改变状态
 | ||||||
|         if (updatedItem.is_paused) return |         if (updatedItem.is_paused) return | ||||||
|          |          | ||||||
|         this.handleUploadError(uploadId, clientUploadId || '') |         updatedItem.status = 3 | ||||||
|       } |       } | ||||||
|     }, |     }, | ||||||
|      |      | ||||||
| @ -244,10 +244,6 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|           talk_type: item.talk_type |           talk_type: item.talk_type | ||||||
|         }) |         }) | ||||||
|          |          | ||||||
|         // 从DialogueStore中移除上传任务
 |  | ||||||
|         const dialogueStore = useDialogueStore() |  | ||||||
|         dialogueStore.removeUploadTask(clientUploadId) |  | ||||||
|          |  | ||||||
|         if (item.onComplete) { |         if (item.onComplete) { | ||||||
|           item.onComplete(item) |           item.onComplete(item) | ||||||
|         } |         } | ||||||
| @ -295,21 +291,5 @@ export const useUploadsStore = defineStore('uploads', { | |||||||
|       // 从上传列表中移除旧的上传项
 |       // 从上传列表中移除旧的上传项
 | ||||||
|       this.items = this.items.filter(i => i.client_upload_id !== clientUploadId) |       this.items = this.items.filter(i => i.client_upload_id !== clientUploadId) | ||||||
|     }, |     }, | ||||||
|      |  | ||||||
|     // 上传失败处理
 |  | ||||||
|     async handleUploadError(uploadId: string, clientUploadId: string) { |  | ||||||
|       const item = this.findItem(uploadId) |  | ||||||
|       if (!item) return |  | ||||||
|        |  | ||||||
|       item.status = 3 // 设置为上传失败状态
 |  | ||||||
|        |  | ||||||
|       // 从DialogueStore中移除上传任务
 |  | ||||||
|       const dialogueStore = useDialogueStore() |  | ||||||
|       dialogueStore.removeUploadTask(clientUploadId) |  | ||||||
|        |  | ||||||
|       if (item.onProgress) { |  | ||||||
|         item.onProgress(-1) // 通知上传失败
 |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|   } |   } | ||||||
| }) | }) | ||||||
|  | |||||||
							
								
								
									
										381
									
								
								src/utils/db.js
									
									
									
									
									
								
							
							
						
						
									
										381
									
								
								src/utils/db.js
									
									
									
									
									
								
							| @ -1,381 +0,0 @@ | |||||||
| 
 |  | ||||||
| import Dexie from 'dexie'; |  | ||||||
| 
 |  | ||||||
| export const db = new Dexie('chatHistory'); |  | ||||||
| 
 |  | ||||||
| // 定义数据库表结构和索引
 |  | ||||||
| // 版本3:优化了索引,提高了查询和排序性能
 |  | ||||||
| db.version(4).stores({ |  | ||||||
|   /** |  | ||||||
|    * 聊天记录表 |  | ||||||
|    * - msg_id: 消息唯一ID (主键) |  | ||||||
|    * - sequence: 消息序列号,用于排序 |  | ||||||
|    * - [talk_type+receiver_id]: 复合索引,用于快速查询会话消息 |  | ||||||
|    * - created_at: 消息创建时间,用于排序 |  | ||||||
|    * - [talk_type+receiver_id+sequence]: 复合索引,用于高效分页查询 |  | ||||||
|    */ |  | ||||||
|   messages: 'msg_id, sequence, [talk_type+receiver_id], created_at, [talk_type+receiver_id+sequence]', |  | ||||||
| 
 |  | ||||||
|   /** |  | ||||||
|    * 会话表 |  | ||||||
|    * - ++id: 自增主键 |  | ||||||
|    * - &index_name: 唯一索引 (talk_type + '_' + receiver_id) |  | ||||||
|    * - updated_at: 索引,用于排序 |  | ||||||
|    * - is_top: 索引,用于置顶排序 |  | ||||||
|    */ |  | ||||||
|   conversations: 'id, &index_name, talk_type, receiver_id, updated_at, unread_num, is_top', |  | ||||||
| }); |  | ||||||
| 
 |  | ||||||
| db.on('ready', () => { |  | ||||||
|   console.log(`数据库已就绪,版本: ${db.verno}`); |  | ||||||
| }); |  | ||||||
| 
 |  | ||||||
| /** 消息类型常量 */ |  | ||||||
| export const MessageType = { |  | ||||||
|   TEXT: 1, // 文本消息
 |  | ||||||
|   IMAGE: 2, // 图片消息
 |  | ||||||
|   FILE: 3, // 文件消息
 |  | ||||||
|   AUDIO: 4, // 语音消息
 |  | ||||||
|   VIDEO: 5, // 视频消息
 |  | ||||||
|   LOCATION: 6, // 位置消息
 |  | ||||||
|   CARD: 7, // 名片消息
 |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| /** 会话类型常量 */ |  | ||||||
| export const TalkType = { |  | ||||||
|   PRIVATE: 1, // 私聊
 |  | ||||||
|   GROUP: 2, // 群聊
 |  | ||||||
| }; |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 生成一个简单的UUID |  | ||||||
|  * @returns {string} UUID |  | ||||||
|  */ |  | ||||||
| function generateUUID() { |  | ||||||
|   return 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, c => { |  | ||||||
|     const r = (Math.random() * 16) | 0; |  | ||||||
|     const v = c === 'x' ? r : (r & 0x3) | 0x8; |  | ||||||
|     return v.toString(16); |  | ||||||
|   }); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // #region 消息操作
 |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 添加或更新一条聊天记录 |  | ||||||
|  * @param {object} message - 消息对象 |  | ||||||
|  * @returns {Promise<string>} 消息ID |  | ||||||
|  */ |  | ||||||
| export async function addMessage(message) { |  | ||||||
|   try { |  | ||||||
|     if (!message.msg_id) { |  | ||||||
|       message.msg_id = generateUUID(); |  | ||||||
|     } |  | ||||||
|     if (!message.created_at) { |  | ||||||
|       message.created_at = new Date().toISOString().replace('T', ' ').substring(0, 19); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // 使用 put 方法,如果主键已存在则更新,否则添加
 |  | ||||||
|     await db.messages.put(message); |  | ||||||
|     return message.msg_id; |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('添加或更新消息失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 批量添加或更新聊天记录 |  | ||||||
|  * @param {Array<object>} messages - 消息对象数组 |  | ||||||
|  * @returns {Promise<void>} |  | ||||||
|  */ |  | ||||||
| export async function batchAddOrUpdateMessages(messages) { |  | ||||||
|   try { |  | ||||||
|     if (!Array.isArray(messages) || messages.length === 0) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     const messagesToStore = messages.map(message => { |  | ||||||
|       if (!message.msg_id) { |  | ||||||
|         message.msg_id = generateUUID(); |  | ||||||
|       } |  | ||||||
|       if (!message.created_at) { |  | ||||||
|         message.created_at = new Date().toISOString().replace('T', ' ').substring(0, 19); |  | ||||||
|       } |  | ||||||
|       return message; |  | ||||||
|     }); |  | ||||||
| 
 |  | ||||||
|     await db.messages.bulkPut(messagesToStore); |  | ||||||
| 
 |  | ||||||
|     // 更新最后一条消息到会话
 |  | ||||||
|     const latestMessage = messagesToStore[messagesToStore.length - 1]; |  | ||||||
|     if (latestMessage) { |  | ||||||
|       await updateConversationLastMessage(latestMessage); |  | ||||||
|     } |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('批量添加或更新消息失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 获取指定会话的聊天记录 |  | ||||||
|  * @param {number} talkType - 会话类型 (1:私聊, 2:群聊) |  | ||||||
|  * @param {number} userId - 当前用户ID |  | ||||||
|  * @param {number} receiverId - 接收者ID (私聊为对方用户ID,群聊为群ID) |  | ||||||
|  * @param {number} [limit=30] - 限制返回的记录数量 |  | ||||||
|  * @param {number|null} [maxSequence=null] - 最大sequence值,用于分页加载更早的消息 |  | ||||||
|  * @returns {Promise<Array<object>>} 消息列表 (按sequence升序排列) |  | ||||||
|  */ |  | ||||||
| export async function getMessages(talkType, userId, receiverId, limit = 30, maxSequence = null) { |  | ||||||
|   try { |  | ||||||
|     let collection; |  | ||||||
| 
 |  | ||||||
|     if (maxSequence !== null) { |  | ||||||
|       // 加载更多:查询 sequence 小于 maxSequence 的消息
 |  | ||||||
|       collection = db.messages |  | ||||||
|         .where('[talk_type+receiver_id+sequence]') |  | ||||||
|         .between([talkType, receiverId, 0], [talkType, receiverId, maxSequence], true, false); |  | ||||||
|     } else { |  | ||||||
|       // 首次加载:查询指定会话的所有消息
 |  | ||||||
|       collection = db.messages.where({ '[talk_type+receiver_id]': [talkType, receiverId] }); |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     // 1. reverse() - 利用索引倒序排列,获取最新的消息
 |  | ||||||
|     // 2. limit() - 限制数量,实现分页
 |  | ||||||
|     // 3. toArray() - 执行查询
 |  | ||||||
|     const messages = await collection.reverse().limit(limit).toArray(); |  | ||||||
| 
 |  | ||||||
|     // 再次 reverse() - 将获取到的分页消息按时间正序排列,以便于在界面上显示
 |  | ||||||
|     return messages.reverse(); |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('获取消息失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 标记指定会话的所有消息为已读 |  | ||||||
|  * @param {number} talkType - 会话类型 |  | ||||||
|  * @param {number} userId - 当前用户ID |  | ||||||
|  * @param {number} receiverId - 接收者ID |  | ||||||
|  * @returns {Promise<number>} 更新的消息数量 |  | ||||||
|  */ |  | ||||||
| export async function markMessagesAsRead(talkType, userId, receiverId) { |  | ||||||
|   try { |  | ||||||
|     let query; |  | ||||||
|     if (talkType === TalkType.PRIVATE) { |  | ||||||
|       // 私聊:只标记对方发给我的未读消息
 |  | ||||||
|       query = db.messages |  | ||||||
|         .where('[talk_type+receiver_id]') |  | ||||||
|         .equals([talkType, userId]) |  | ||||||
|         .and(item => item.user_id === receiverId && item.is_read === 0); |  | ||||||
|     } else { |  | ||||||
|       // 群聊:标记群里所有非自己的未读消息
 |  | ||||||
|       query = db.messages |  | ||||||
|         .where('[talk_type+receiver_id]') |  | ||||||
|         .equals([talkType, receiverId]) |  | ||||||
|         .and(item => item.user_id !== userId && item.is_read === 0); |  | ||||||
|     } |  | ||||||
|     return await query.modify({ is_read: 1 }); |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('批量标记消息已读失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 撤回消息 |  | ||||||
|  * @param {string} msgId - 消息ID |  | ||||||
|  * @returns {Promise<number>} 更新记录数 (1或0) |  | ||||||
|  */ |  | ||||||
| export async function revokeMessage(msgId) { |  | ||||||
|   try { |  | ||||||
|     return await db.messages.update(msgId, { is_revoke: 1 }); |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('撤回消息失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 删除消息 |  | ||||||
|  * @param {string} msgId - 消息ID |  | ||||||
|  * @returns {Promise<void>} |  | ||||||
|  */ |  | ||||||
| export async function deleteMessage(msgId) { |  | ||||||
|   try { |  | ||||||
|     await db.messages.delete(msgId); |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('删除消息失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // #endregion 消息操作
 |  | ||||||
| 
 |  | ||||||
| // #region 会话操作
 |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 添加或更新会话 |  | ||||||
|  * @param {object} conversation - 会话对象 |  | ||||||
|  * @returns {Promise<number>} 会话ID |  | ||||||
|  */ |  | ||||||
| export async function addOrUpdateConversation(conversation) { |  | ||||||
|   try { |  | ||||||
|     // put 方法会根据唯一索引 index_name 自动判断是添加还是更新
 |  | ||||||
|     return await db.conversations.put(conversation); |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('添加或更新会话失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 获取所有会话列表 |  | ||||||
|  * @param {boolean} [includeEmpty=false] - 是否包含没有最后一条消息的会话 |  | ||||||
|  * @returns {Promise<Array<object>>} 会话列表 (按置顶和更新时间排序) |  | ||||||
|  */ |  | ||||||
| export async function getConversations(includeEmpty = false) { |  | ||||||
|   try { |  | ||||||
|     const filterFn = item => !includeEmpty ? (item.msg_text && item.msg_text.length > 0) : true; |  | ||||||
| 
 |  | ||||||
|     // 分别查询置顶和非置顶会话,以利用索引并优化性能
 |  | ||||||
|     const topConversationsPromise = db.conversations |  | ||||||
|       .where('is_top') |  | ||||||
|       .equals(1) |  | ||||||
|       .sortBy('updated_at') |  | ||||||
|       .then(arr => arr.reverse().filter(filterFn)); |  | ||||||
| 
 |  | ||||||
|     const otherConversationsPromise = db.conversations |  | ||||||
|       .where('is_top') |  | ||||||
|       .notEqual(1) |  | ||||||
|       .sortBy('updated_at') |  | ||||||
|       .then(arr => arr.reverse().filter(filterFn)); |  | ||||||
| 
 |  | ||||||
|     const [topConversations, otherConversations] = await Promise.all([ |  | ||||||
|       topConversationsPromise, |  | ||||||
|       otherConversationsPromise, |  | ||||||
|     ]); |  | ||||||
| 
 |  | ||||||
|     return [...topConversations, ...otherConversations]; |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('获取会话列表失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 获取指定会话 |  | ||||||
|  * @param {number} talkType - 会话类型 |  | ||||||
|  * @param {number} receiverId - 接收者ID |  | ||||||
|  * @returns {Promise<object|undefined>} 会话对象 |  | ||||||
|  */ |  | ||||||
| export async function getConversation(talkType, receiverId) { |  | ||||||
|   try { |  | ||||||
|     const indexName = `${talkType}_${receiverId}`; |  | ||||||
|     return await db.conversations.get({ index_name: indexName }); |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('获取会话失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 更新会话的未读消息数 |  | ||||||
|  * @param {number} talkType - 会话类型 |  | ||||||
|  * @param {number} receiverId - 接收者ID |  | ||||||
|  * @param {number|null} unreadNum - 未读消息数。如果为null,则自增1 |  | ||||||
|  * @returns {Promise<number>} 更新的记录数 |  | ||||||
|  */ |  | ||||||
| export async function updateConversationUnreadNum(talkType, receiverId, unreadNum = null) { |  | ||||||
|   try { |  | ||||||
|     const indexName = `${talkType}_${receiverId}`; |  | ||||||
|     const conversation = await db.conversations.get({ index_name: indexName }); |  | ||||||
| 
 |  | ||||||
|     if (conversation) { |  | ||||||
|       const newUnreadNum = unreadNum === null ? (conversation.unread_num || 0) + 1 : unreadNum; |  | ||||||
|       return await db.conversations.update(conversation.id, { unread_num: newUnreadNum }); |  | ||||||
|     } |  | ||||||
|     return 0; |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('更新会话未读数失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 清空会话的未读消息数 |  | ||||||
|  * @param {number} talkType - 会话类型 |  | ||||||
|  * @param {number} receiverId - 接收者ID |  | ||||||
|  * @returns {Promise<number>} 更新的记录数 |  | ||||||
|  */ |  | ||||||
| export function clearConversationUnreadNum(talkType, receiverId) { |  | ||||||
|   return updateConversationUnreadNum(talkType, receiverId, 0); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 删除会话及其相关的消息 |  | ||||||
|  * @param {number} conversationId - 会话ID |  | ||||||
|  * @param {boolean} [deleteMessages=false] - 是否同时删除相关的消息记录 |  | ||||||
|  * @returns {Promise<void>} |  | ||||||
|  */ |  | ||||||
| export async function deleteConversation(conversationId, deleteMessages = false) { |  | ||||||
|   try { |  | ||||||
|     await db.transaction('rw', db.conversations, db.messages, async () => { |  | ||||||
|       const conversation = await db.conversations.get(conversationId); |  | ||||||
|       if (!conversation) return; |  | ||||||
| 
 |  | ||||||
|       // 删除会话
 |  | ||||||
|       await db.conversations.delete(conversationId); |  | ||||||
| 
 |  | ||||||
|       // 如果需要,删除关联的消息
 |  | ||||||
|       if (deleteMessages) { |  | ||||||
|         const { talk_type, receiver_id } = conversation; |  | ||||||
|         await db.messages.where({ '[talk_type+receiver_id]': [talk_type, receiver_id] }).delete(); |  | ||||||
|       } |  | ||||||
|     }); |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('删除会话失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /** |  | ||||||
|  * 更新会话的最后一条消息摘要 |  | ||||||
|  * @param {object} message - 消息对象 |  | ||||||
|  * @returns {Promise<number>} 更新的记录数 |  | ||||||
|  */ |  | ||||||
| export async function updateConversationLastMessage(message) { |  | ||||||
|   try { |  | ||||||
|     const { talk_type, user_id, receiver_id, msg_type } = message; |  | ||||||
|     const targetReceiverId = talk_type === TalkType.PRIVATE ? (user_id === receiver_id ? user_id : receiver_id) : receiver_id; |  | ||||||
|     const indexName = `${talk_type}_${targetReceiverId}`; |  | ||||||
| 
 |  | ||||||
|     const conversation = await db.conversations.get({ index_name: indexName }); |  | ||||||
|     if (!conversation) return 0; |  | ||||||
| 
 |  | ||||||
|     let msgText = ''; |  | ||||||
|     switch (msg_type) { |  | ||||||
|       case MessageType.TEXT: msgText = message.content || ''; break; |  | ||||||
|       case MessageType.IMAGE: msgText = '[图片]'; break; |  | ||||||
|       case MessageType.FILE: msgText = '[文件]'; break; |  | ||||||
|       case MessageType.AUDIO: msgText = '[语音]'; break; |  | ||||||
|       case MessageType.VIDEO: msgText = '[视频]'; break; |  | ||||||
|       case MessageType.LOCATION: msgText = '[位置]'; break; |  | ||||||
|       case MessageType.CARD: msgText = '[名片]'; break; |  | ||||||
|       default: msgText = '[未知消息]'; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     return await db.conversations.update(conversation.id, { |  | ||||||
|       msg_text: msgText, |  | ||||||
|       content: message.content || '', |  | ||||||
|       updated_at: message.created_at, |  | ||||||
|     }); |  | ||||||
|   } catch (error) { |  | ||||||
|     console.error('更新会话最后消息失败:', error); |  | ||||||
|     throw error; |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| // #endregion 会话操作
 |  | ||||||
| @ -54,6 +54,7 @@ request.interceptors.request.use((config) => { | |||||||
| 
 | 
 | ||||||
| // 响应拦截器
 | // 响应拦截器
 | ||||||
| request.interceptors.response.use((response) => { | request.interceptors.response.use((response) => { | ||||||
|  |   console.log('response.data.status',response.data.status) | ||||||
|   if(response.data.code !==200&&response.data.status!==0){ |   if(response.data.code !==200&&response.data.status!==0){ | ||||||
|     window['$message'].warning(response.data.msg) |     window['$message'].warning(response.data.msg) | ||||||
|   } |   } | ||||||
|  | |||||||
| @ -145,7 +145,8 @@ watch( | |||||||
|     if (talkParams.type !== 2) { |     if (talkParams.type !== 2) { | ||||||
|       ServeCheckFriend({ receiver_id: newValue.receiver_id, talk_type: 1 }).then((res) => { |       ServeCheckFriend({ receiver_id: newValue.receiver_id, talk_type: 1 }).then((res) => { | ||||||
|         if (res?.code === 200) { |         if (res?.code === 200) { | ||||||
|           isFriend.value = res.data.is_friend |           console.log(res, 'ress') | ||||||
|  |           isFriend.value = !res.data.is_friend | ||||||
|         } |         } | ||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
| @ -536,7 +537,7 @@ const clearSelectedDateTime = () => { | |||||||
|     <main class="el-main relative"> |     <main class="el-main relative"> | ||||||
|       <div |       <div | ||||||
|         class="p-[15px] pt-[10px] w-[100%] z-99 absolute" |         class="p-[15px] pt-[10px] w-[100%] z-99 absolute" | ||||||
|         v-if="!isFriend && talkParams.type !== 2" |         v-if="isFriend && talkParams.type !== 2" | ||||||
|       > |       > | ||||||
|         <div |         <div | ||||||
|           class="bg-[#FFFFFF] w-[100%] p-[10px] text-[14px] flex justify-between" |           class="bg-[#FFFFFF] w-[100%] p-[10px] text-[14px] flex justify-between" | ||||||
|  | |||||||
| @ -519,7 +519,6 @@ const items = computed((): ISession[] => { | |||||||
| 
 | 
 | ||||||
|   return [...topItems, ...normalItems] |   return [...topItems, ...normalItems] | ||||||
| }) | }) | ||||||
| setTimeout(()=>{console.log('items',items)},2000) |  | ||||||
| watch( | watch( | ||||||
|   () => state.addressBookSearchNickName, |   () => state.addressBookSearchNickName, | ||||||
|   (newValue, oldValue) => { |   (newValue, oldValue) => { | ||||||
| @ -594,7 +593,7 @@ const indexName = computed(() => dialogueStore.index_name) | |||||||
| // 切换会话 | // 切换会话 | ||||||
| const onTabTalk = (item: ISession, follow = false) => { | const onTabTalk = (item: ISession, follow = false) => { | ||||||
|   console.log('onTabTalk') |   console.log('onTabTalk') | ||||||
| console.log('item.index_name === indexName.value',item.index_name === indexName.value) | 
 | ||||||
|   if (item.index_name === indexName.value) return |   if (item.index_name === indexName.value) return | ||||||
| 
 | 
 | ||||||
|   searchKeyword.value = '' |   searchKeyword.value = '' | ||||||
| @ -639,7 +638,7 @@ const onReload = () => { | |||||||
| // 初始化加载 | // 初始化加载 | ||||||
| const onInitialize = () => { | const onInitialize = () => { | ||||||
|   let index_name = getCacheIndexName() |   let index_name = getCacheIndexName() | ||||||
|   console.log('index_name',index_name) | 
 | ||||||
|   index_name && onTabTalk(talkStore.findItem(index_name), true) |   index_name && onTabTalk(talkStore.findItem(index_name), true) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -3,7 +3,7 @@ import { watch, onMounted, ref, nextTick, onUnmounted } from 'vue' | |||||||
| import { NDropdown, NCheckbox, NPopover, NInfiniteScroll } from 'naive-ui' | import { NDropdown, NCheckbox, NPopover, NInfiniteScroll } from 'naive-ui' | ||||||
| import { Loading, MoreThree, ToTop } from '@icon-park/vue-next' | import { Loading, MoreThree, ToTop } from '@icon-park/vue-next' | ||||||
| import { bus } from '@/utils/event-bus' | import { bus } from '@/utils/event-bus' | ||||||
| import { useDialogueStore, useTalkStore } from '@/store' | import { useDialogueStore } from '@/store' | ||||||
| import { formatTime, parseTime } from '@/utils/datetime' | import { formatTime, parseTime } from '@/utils/datetime' | ||||||
| import { clipboard, htmlDecode, clipboardImage } from '@/utils/common' | import { clipboard, htmlDecode, clipboardImage } from '@/utils/common' | ||||||
| import { downloadImage } from '@/utils/functions' | import { downloadImage } from '@/utils/functions' | ||||||
| @ -19,11 +19,8 @@ import RevokeMessage from '@/components/talk/message/RevokeMessage.vue' | |||||||
| import { voiceToText, ServeMessageReadDetail } from '@/api/chat.js' | import { voiceToText, ServeMessageReadDetail } from '@/api/chat.js' | ||||||
| import { confirmBox } from '@/components/confirm-box/service.js' | import { confirmBox } from '@/components/confirm-box/service.js' | ||||||
| import ws from '@/connect' | import ws from '@/connect' | ||||||
| import { useRouter } from 'vue-router' |  | ||||||
| import avatarModule from '@/components/avatar-module/index.vue' | import avatarModule from '@/components/avatar-module/index.vue' | ||||||
| 
 | 
 | ||||||
| const router = useRouter() |  | ||||||
| 
 |  | ||||||
| // 定义消息已读状态接口 | // 定义消息已读状态接口 | ||||||
| interface ReadStatus { | interface ReadStatus { | ||||||
|   msg_ids: string[] |   msg_ids: string[] | ||||||
| @ -89,28 +86,11 @@ const { dropdown, showDropdownMenu, closeDropdownMenu, isOneMonthBefore } = useM | |||||||
| const { showUserInfoModal } = useInject() | const { showUserInfoModal } = useInject() | ||||||
| const dialogueStore = useDialogueStore() | const dialogueStore = useDialogueStore() | ||||||
| const userStore = useUserStore() | const userStore = useUserStore() | ||||||
| const talkStore = useTalkStore() |  | ||||||
| // const showUserInfoModal = (uid: number) => { | // const showUserInfoModal = (uid: number) => { | ||||||
| //   userStore.getUserInfo(uid) | //   userStore.getUserInfo(uid) | ||||||
| // } | // } | ||||||
| // 置底按钮 | // 置底按钮 | ||||||
| const skipBottom = ref(false) | const skipBottom = ref(false) | ||||||
| const goToMessage = (result) => { |  | ||||||
|   const talk_type = props.talk_type |  | ||||||
|   const receiver_id = props.receiver_id |  | ||||||
|   dialogueStore.specifiedMsg = encodeURIComponent( |  | ||||||
|     JSON.stringify({ |  | ||||||
|       talk_type, |  | ||||||
|       receiver_id, |  | ||||||
|       msg_id: result.msg_id, |  | ||||||
|       cursor: result.sequence - 15 > 0 ? result.sequence - 15 : 0, |  | ||||||
|       direction: 'down', |  | ||||||
|       sort_sequence: 'asc', |  | ||||||
|       create_time: result.created_at |  | ||||||
|     }) |  | ||||||
|   ) |  | ||||||
|   talkStore.toTalk(talk_type, receiver_id, router) |  | ||||||
| } |  | ||||||
| // 是否显示消息时间 | // 是否显示消息时间 | ||||||
| const isShowTalkTime = (index: number, datetime: string) => { | const isShowTalkTime = (index: number, datetime: string) => { | ||||||
|   if (datetime == undefined) { |   if (datetime == undefined) { | ||||||
| @ -350,16 +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])) { | ||||||
|     if (!isOneMonthBefore(item.created_at.split(' ')[0])) { |  | ||||||
|       return useMessage.info('只支持转发近一个月内的消息') |  | ||||||
|     } |  | ||||||
|     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('只支持转发近一个月内的消息') | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -372,7 +351,6 @@ let noRefreshTimer: number | null = null | |||||||
| watch( | watch( | ||||||
|   () => props, |   () => props, | ||||||
|   async (newProps) => { |   async (newProps) => { | ||||||
|     console.log('监听props',newProps) |  | ||||||
|     await nextTick() |     await nextTick() | ||||||
|     // 生成当前会话的唯一标识 |     // 生成当前会话的唯一标识 | ||||||
|     const newSessionKey = `${newProps.talk_type}_${newProps.receiver_id}` |     const newSessionKey = `${newProps.talk_type}_${newProps.receiver_id}` | ||||||
| @ -415,7 +393,7 @@ watch( | |||||||
|       }, 3000) |       }, 3000) | ||||||
|       return |       return | ||||||
|     } |     } | ||||||
|     console.log('fsd付大夫') | 
 | ||||||
|     onLoad( |     onLoad( | ||||||
|       { |       { | ||||||
|         receiver_id: newProps.receiver_id, |         receiver_id: newProps.receiver_id, | ||||||
| @ -425,7 +403,7 @@ watch( | |||||||
|       specialParams ? { specifiedMsg: specialParams } : undefined |       specialParams ? { specifiedMsg: specialParams } : undefined | ||||||
|     ) |     ) | ||||||
|   }, |   }, | ||||||
|   {  deep: true,immediate:true } |   { immediate: true, deep: true } | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| // onMounted(() => { | // onMounted(() => { | ||||||
| @ -556,6 +534,7 @@ const checkVisibleOutElements = () => { | |||||||
|     }) |     }) | ||||||
|     if (waitDoCheck.length > 0) { |     if (waitDoCheck.length > 0) { | ||||||
|       waitDoCheck.forEach((doCheckItem) => { |       waitDoCheck.forEach((doCheckItem) => { | ||||||
|  |         console.error('====组装了新版已读回执参数,需要发送socket=====', doCheckItem) | ||||||
|         ws.emit('im.message.listen.read', doCheckItem) |         ws.emit('im.message.listen.read', doCheckItem) | ||||||
|       }) |       }) | ||||||
|     } |     } | ||||||
| @ -615,6 +594,7 @@ watch( | |||||||
|       if (observer) { |       if (observer) { | ||||||
|         observer.disconnect() |         observer.disconnect() | ||||||
|       } |       } | ||||||
|  | 
 | ||||||
|       // 重新初始化观察者 |       // 重新初始化观察者 | ||||||
|       const options = { |       const options = { | ||||||
|         root: null, |         root: null, | ||||||
| @ -622,6 +602,7 @@ watch( | |||||||
|         rootMargin: '50px 0px' |         rootMargin: '50px 0px' | ||||||
|       } |       } | ||||||
|       observer = new IntersectionObserver(handleIntersection, options) |       observer = new IntersectionObserver(handleIntersection, options) | ||||||
|  | 
 | ||||||
|       // 重新观察所有消息元素 |       // 重新观察所有消息元素 | ||||||
|       const messageElements = document.querySelectorAll('.message-item') |       const messageElements = document.querySelectorAll('.message-item') | ||||||
|       messageElements.forEach((el) => { |       messageElements.forEach((el) => { | ||||||
| @ -789,7 +770,7 @@ const onCustomSkipBottomEvent = () => { | |||||||
|       <div class="load-toolbar pointer"> |       <div class="load-toolbar pointer"> | ||||||
|         <span v-if="loadConfig.status == 0"> 正在加载数据中 ... </span> |         <span v-if="loadConfig.status == 0"> 正在加载数据中 ... </span> | ||||||
|         <span v-else-if="loadConfig.status == 1" @click="onRefreshLoad"> 查看更多消息 ... </span> |         <span v-else-if="loadConfig.status == 1" @click="onRefreshLoad"> 查看更多消息 ... </span> | ||||||
|         <span v-else-if="loadConfig.status == 2 || loadConfig.status == 3" class="no-more"> 没有更多消息了 </span> |         <span v-else class="no-more"> 没有更多消息了 </span> | ||||||
|       </div> |       </div> | ||||||
| 
 | 
 | ||||||
|       <div |       <div | ||||||
| @ -926,11 +907,11 @@ const onCustomSkipBottomEvent = () => { | |||||||
| <n-icon class="more-tools pointer" :component="MoreThree" @click="onContextMenu($event, item)" /> | <n-icon class="more-tools pointer" :component="MoreThree" @click="onContextMenu($event, item)" /> | ||||||
| </div> --> | </div> --> | ||||||
|             </div> |             </div> | ||||||
|             <!-- @click="onJumpMessage(item.extra?.reply?.msg_id)" --> | 
 | ||||||
|             <div |             <div | ||||||
|               v-if="item.extra.reply" |               v-if="item.extra.reply" | ||||||
|               class="talk-reply pointer" |               class="talk-reply pointer" | ||||||
|               @click="goToMessage(item.extra?.reply)" |               @click="onJumpMessage(item.extra?.reply?.msg_id)" | ||||||
|             > |             > | ||||||
|               <n-icon :component="ToTop" size="14" class="icon-top" /> |               <n-icon :component="ToTop" size="14" class="icon-top" /> | ||||||
|               <span class="ellipsis"> |               <span class="ellipsis"> | ||||||
| @ -1075,7 +1056,6 @@ const onCustomSkipBottomEvent = () => { | |||||||
|     &.border { |     &.border { | ||||||
|       border-radius: 10px; |       border-radius: 10px; | ||||||
|       border: 1px solid var(--im-primary-color); |       border: 1px solid var(--im-primary-color); | ||||||
|       background-color: red; |  | ||||||
|     } |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -115,9 +115,6 @@ const onSendVideoEvent = async ({ data }) => { | |||||||
|   // 创建临时消息记录 |   // 创建临时消息记录 | ||||||
|   const tempMessage = { |   const tempMessage = { | ||||||
|     msg_id: uploadId, |     msg_id: uploadId, | ||||||
|     insert_sequence: dialogueStore.records.length > 0  |  | ||||||
|       ? dialogueStore.records[dialogueStore.records.length-1].sequence  |  | ||||||
|       : 0, |  | ||||||
|     sequence: Date.now(), |     sequence: Date.now(), | ||||||
|     talk_type: props.talk_type, |     talk_type: props.talk_type, | ||||||
|     msg_type: 5, // 视频消息类型 |     msg_type: 5, // 视频消息类型 | ||||||
| @ -140,8 +137,8 @@ const onSendVideoEvent = async ({ data }) => { | |||||||
|     float: 'right' // 我发送的消息显示在右侧 |     float: 'right' // 我发送的消息显示在右侧 | ||||||
|   } |   } | ||||||
|    |    | ||||||
|   // 使用新的方法添加上传任务 |   // 直接添加到对话记录中 | ||||||
|   dialogueStore.addUploadTask(tempMessage)   |   dialogueStore.addDialogueRecord(tempMessage)   | ||||||
|   nextTick(()=>{ |   nextTick(()=>{ | ||||||
|         scrollToBottom() |         scrollToBottom() | ||||||
|       }) |       }) | ||||||
| @ -155,6 +152,7 @@ const onSendVideoEvent = async ({ data }) => { | |||||||
|     }, |     }, | ||||||
|     async () => { |     async () => { | ||||||
|       dialogueStore.batchDelDialogueRecord([uploadId]) |       dialogueStore.batchDelDialogueRecord([uploadId]) | ||||||
|  |   | ||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
| @ -166,12 +164,13 @@ const onSendCodeEvent = ({ data, callBack }) => { | |||||||
| 
 | 
 | ||||||
| // 发送文件消息 | // 发送文件消息 | ||||||
| const onSendFileEvent = ({ data }) => { | const onSendFileEvent = ({ data }) => { | ||||||
|  |   let maxsize = 200 * 1024 * 1024 | ||||||
|  |   if (data.size > maxsize) { | ||||||
|  |     return window['$message'].warning('上传文件不能超过100M!') | ||||||
|  |   } | ||||||
|   const clientUploadId = `file-${Date.now()}-${Math.floor(Math.random() * 1000)}` |   const clientUploadId = `file-${Date.now()}-${Math.floor(Math.random() * 1000)}` | ||||||
|   const tempMessage = { |   const tempMessage = { | ||||||
|     msg_id: clientUploadId, |     msg_id: clientUploadId, | ||||||
|     insert_sequence: dialogueStore.records.length > 0  |  | ||||||
|       ? dialogueStore.records[dialogueStore.records.length-1].sequence  |  | ||||||
|       : 0, |  | ||||||
|     sequence: Date.now(), |     sequence: Date.now(), | ||||||
|     talk_type: props.talk_type, |     talk_type: props.talk_type, | ||||||
|     msg_type: 6, |     msg_type: 6, | ||||||
| @ -193,7 +192,7 @@ const onSendFileEvent = ({ data }) => { | |||||||
|     }, |     }, | ||||||
|     float: 'right' |     float: 'right' | ||||||
|   } |   } | ||||||
|   dialogueStore.addUploadTask(tempMessage) |   dialogueStore.addDialogueRecord(tempMessage) | ||||||
|   nextTick(()=>{ |   nextTick(()=>{ | ||||||
|         scrollToBottom() |         scrollToBottom() | ||||||
|       }) |       }) | ||||||
| @ -202,8 +201,8 @@ const onSendFileEvent = ({ data }) => { | |||||||
|       dialogueStore.updateUploadProgress(clientUploadId, percentage) |       dialogueStore.updateUploadProgress(clientUploadId, percentage) | ||||||
|     }, |     }, | ||||||
|     async () => { |     async () => { | ||||||
|       // 上传完成后,上传任务已经被removeUploadTask方法移除 |       dialogueStore.batchDelDialogueRecord([clientUploadId]) | ||||||
|       // 不需要再次从records中删除 |      | ||||||
|     } |     } | ||||||
|   ) |   ) | ||||||
| } | } | ||||||
|  | |||||||
| @ -46,9 +46,9 @@ export default defineConfig(({ mode }) => { | |||||||
|       vueJsx({}),  |       vueJsx({}),  | ||||||
|       compressPlugin(),  |       compressPlugin(),  | ||||||
|       UnoCSS(), |       UnoCSS(), | ||||||
|       vueDevTools({ |       // vueDevTools({
 | ||||||
|         launchEditor: 'trae', |       //   launchEditor: 'trae',
 | ||||||
|       }) |       // })
 | ||||||
|     ], |     ], | ||||||
|     define: { |     define: { | ||||||
|       __APP_ENV__: env.APP_ENV |       __APP_ENV__: env.APP_ENV | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user