Compare commits
	
		
			5 Commits
		
	
	
		
			a0cab6f763
			...
			09510a2400
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
| 09510a2400 | |||
| fe63a12e8e | |||
| 06bbec8d62 | |||
| b1d2fd47c4 | |||
| 81fb19092e | 
							
								
								
									
										16
									
								
								env/.env.development
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										16
									
								
								env/.env.development
									
									
									
									
										vendored
									
									
								
							| @ -5,4 +5,18 @@ VITE_DELETE_CONSOLE = false | |||||||
| # 是否开启sourcemap | # 是否开启sourcemap | ||||||
| VITE_SHOW_SOURCEMAP = true | VITE_SHOW_SOURCEMAP = true | ||||||
| 
 | 
 | ||||||
| VITE_SERVER_BASEURL = 'http://114.218.158.24:9020' | # 开发环境 | ||||||
|  | # VITE_SERVER_BASEURL = 'http://114.218.158.24:9020' | ||||||
|  | # VITE_SERVER_BASEURL = 'http://192.168.88.80:9040/api' | ||||||
|  | # 正式环境 作为测试使用 | ||||||
|  | # VITE_SERVER_BASEURL = 'https://warehouse.szjixun.cn/oa_backend' | ||||||
|  | 
 | ||||||
|  | #体制外 | ||||||
|  | # VITE_SERVER_BASEURL = 'https://oa-a.szjixun.cn/api' | ||||||
|  | 
 | ||||||
|  | # 体制内 | ||||||
|  | VITE_SERVER_BASEURL = 'https://oa-b.szjixun.cn/api' | ||||||
|  | # VITE_SERVER_BASEURL = 'https://oa-b.szjixun.cn/api' | ||||||
|  | # VITE_SERVER_BASEURL = 'https://erpapi.fontree.cn' | ||||||
|  | 
 | ||||||
|  | # VITE_SERVER_BASEURL = 'https://erptest.fontree.cn:9020' | ||||||
							
								
								
									
										11
									
								
								env/.env.production
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										11
									
								
								env/.env.production
									
									
									
									
										vendored
									
									
								
							| @ -5,6 +5,15 @@ VITE_DELETE_CONSOLE = true | |||||||
| # 是否开启sourcemap | # 是否开启sourcemap | ||||||
| VITE_SHOW_SOURCEMAP = false | VITE_SHOW_SOURCEMAP = false | ||||||
| 
 | 
 | ||||||
| VITE_SERVER_BASEURL = 'https://erpapi.fontree.cn' # 体制外 OA | # VITE_SERVER_BASEURL = 'https://erpapi.fontree.cn' # 体制外 OA | ||||||
|  | # 正式环境 作为测试使用 | ||||||
|  | # VITE_SERVER_BASEURL = 'https://warehouse.szjixun.cn/oa_backend'  | ||||||
|  | # VITE_SERVER_BASEURL = 'https://oa-a.szjixun.cn/api' | ||||||
|  | #体制外 | ||||||
|  | # VITE_SERVER_BASEURL = 'https://oa-a.szjixun.cn/api' | ||||||
|  | 
 | ||||||
|  | # 体制内 | ||||||
|  | VITE_SERVER_BASEURL = 'https://oa-b.szjixun.cn/api' | ||||||
|  | 
 | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -52,6 +52,7 @@ | |||||||
|     "build:app-ios": "uni build -p app-ios", |     "build:app-ios": "uni build -p app-ios", | ||||||
|     "build:custom": "uni build -p", |     "build:custom": "uni build -p", | ||||||
|     "build:h5": "uni build", |     "build:h5": "uni build", | ||||||
|  |     "build:h5-dev": "uni build --mode dev", | ||||||
|     "build": "uni build", |     "build": "uni build", | ||||||
|     "build:h5:ssr": "uni build --ssr", |     "build:h5:ssr": "uni build --ssr", | ||||||
|     "build:mp-alipay": "uni build -p mp-alipay", |     "build:mp-alipay": "uni build -p mp-alipay", | ||||||
| @ -106,6 +107,7 @@ | |||||||
|     "pinia": "2.0.36", |     "pinia": "2.0.36", | ||||||
|     "pinia-plugin-persistedstate": "3.2.1", |     "pinia-plugin-persistedstate": "3.2.1", | ||||||
|     "qs": "6.5.3", |     "qs": "6.5.3", | ||||||
|  |     "vconsole": "^3.15.1", | ||||||
|     "vue": "3.4.21", |     "vue": "3.4.21", | ||||||
|     "wot-design-uni": "^1.4.0", |     "wot-design-uni": "^1.4.0", | ||||||
|     "z-paging": "^2.8.4" |     "z-paging": "^2.8.4" | ||||||
|  | |||||||
| @ -70,6 +70,9 @@ dependencies: | |||||||
|   qs: |   qs: | ||||||
|     specifier: 6.5.3 |     specifier: 6.5.3 | ||||||
|     version: 6.5.3 |     version: 6.5.3 | ||||||
|  |   vconsole: | ||||||
|  |     specifier: ^3.15.1 | ||||||
|  |     version: 3.15.1 | ||||||
|   vue: |   vue: | ||||||
|     specifier: 3.4.21 |     specifier: 3.4.21 | ||||||
|     version: 3.4.21(typescript@5.8.3) |     version: 3.4.21(typescript@5.8.3) | ||||||
| @ -5555,6 +5558,11 @@ packages: | |||||||
|     engines: {node: '>= 0.6'} |     engines: {node: '>= 0.6'} | ||||||
|     dev: true |     dev: true | ||||||
| 
 | 
 | ||||||
|  |   /copy-text-to-clipboard@3.2.0: | ||||||
|  |     resolution: {integrity: sha512-RnJFp1XR/LOBDckxTib5Qjr/PMfkatD0MUCQgdpqS8MdKiNUzBjAQBEN6oUy+jW7LI93BBG3DtMB2KOOKpGs2Q==} | ||||||
|  |     engines: {node: '>=12'} | ||||||
|  |     dev: false | ||||||
|  | 
 | ||||||
|   /core-js-compat@3.42.0: |   /core-js-compat@3.42.0: | ||||||
|     resolution: {integrity: sha512-bQasjMfyDGyaeWKBIu33lHh9qlSR0MFE/Nmc6nMjf/iU9b3rSMdAYz1Baxrv4lPdGUsTqZudHA4jIGSJy0SWZQ==} |     resolution: {integrity: sha512-bQasjMfyDGyaeWKBIu33lHh9qlSR0MFE/Nmc6nMjf/iU9b3rSMdAYz1Baxrv4lPdGUsTqZudHA4jIGSJy0SWZQ==} | ||||||
|     dependencies: |     dependencies: | ||||||
| @ -8325,6 +8333,10 @@ packages: | |||||||
|     resolution: {integrity: sha512-ckmWDJjphvd/FvZawgygcUeQCxzvohjFO5RxTjj4eq8kw359gFF3E1brjfI+viLMxss5JrHTDRHZvu2/tuy0Qg==} |     resolution: {integrity: sha512-ckmWDJjphvd/FvZawgygcUeQCxzvohjFO5RxTjj4eq8kw359gFF3E1brjfI+viLMxss5JrHTDRHZvu2/tuy0Qg==} | ||||||
|     dev: true |     dev: true | ||||||
| 
 | 
 | ||||||
|  |   /mutation-observer@1.0.3: | ||||||
|  |     resolution: {integrity: sha512-M/O/4rF2h776hV7qGMZUH3utZLO/jK7p8rnNgGkjKUw8zCGjRQPxB8z6+5l8+VjRUQ3dNYu4vjqXYLr+U8ZVNA==} | ||||||
|  |     dev: false | ||||||
|  | 
 | ||||||
|   /nanoid@3.3.11: |   /nanoid@3.3.11: | ||||||
|     resolution: {integrity: sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==} |     resolution: {integrity: sha512-N8SpfPUnUp1bK+PMYW8qSWdl9U+wwNWI4QKxOYDy9JAro3WMX7p2OeVRF9v+347pnakNevPmiHhNmZ2HbFA76w==} | ||||||
|     engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} |     engines: {node: ^10 || ^12 || ^13.7 || ^14 || >=15.0.1} | ||||||
| @ -10495,6 +10507,15 @@ packages: | |||||||
|     engines: {node: '>= 0.8'} |     engines: {node: '>= 0.8'} | ||||||
|     dev: true |     dev: true | ||||||
| 
 | 
 | ||||||
|  |   /vconsole@3.15.1: | ||||||
|  |     resolution: {integrity: sha512-KH8XLdrq9T5YHJO/ixrjivHfmF2PC2CdVoK6RWZB4yftMykYIaXY1mxZYAic70vADM54kpMQF+dYmvl5NRNy1g==} | ||||||
|  |     dependencies: | ||||||
|  |       '@babel/runtime': 7.27.1 | ||||||
|  |       copy-text-to-clipboard: 3.2.0 | ||||||
|  |       core-js: 3.42.0 | ||||||
|  |       mutation-observer: 1.0.3 | ||||||
|  |     dev: false | ||||||
|  | 
 | ||||||
|   /vite-plugin-restart@0.4.2(vite@5.2.8): |   /vite-plugin-restart@0.4.2(vite@5.2.8): | ||||||
|     resolution: {integrity: sha512-9aWN2ScJ8hbT7aC8SDeZnsbWapnslz1vhNq6Vgf2GU9WdN4NExlrWhtnu7pmtOUG3Guj8y6lPcUZ+ls7SVP33w==} |     resolution: {integrity: sha512-9aWN2ScJ8hbT7aC8SDeZnsbWapnslz1vhNq6Vgf2GU9WdN4NExlrWhtnu7pmtOUG3Guj8y6lPcUZ+ls7SVP33w==} | ||||||
|     peerDependencies: |     peerDependencies: | ||||||
|  | |||||||
| @ -54,7 +54,7 @@ const httpInterceptor = { | |||||||
|     const userStore = useUserStore() |     const userStore = useUserStore() | ||||||
|     const { token } = userStore.userInfo as unknown as IUserInfo |     const { token } = userStore.userInfo as unknown as IUserInfo | ||||||
|     if (token) { |     if (token) { | ||||||
|       options.header.Authorization = `Bearer ${token}` |       options.header.Authorization = `${token}` | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
| } | } | ||||||
|  | |||||||
| @ -11,7 +11,7 @@ | |||||||
|   <div class="flex flex-col h-screen bg-#ffffff tops"> |   <div class="flex flex-col h-screen bg-#ffffff tops"> | ||||||
|     <!-- Navigation Bar --> |     <!-- Navigation Bar --> | ||||||
|     <div |     <div | ||||||
|       class="flex-none flex items-center justify-between px-5 py-3 bg-white shadow-md h-20 pt-10 z-999 fixed top-0 w-full box-border" |       class="flex-none flex items-center justify-between px-5 pb-6 bg-white shadow-md h-20 pt-15 z-999 fixed top-0 w-full box-border" | ||||||
|     > |     > | ||||||
|       <image src="/static/aichat/black.png" class="w-3 h-4.5" @click="goBack" /> |       <image src="/static/aichat/black.png" class="w-3 h-4.5" @click="goBack" /> | ||||||
|       <div class="text-lg font-medium ml-12">小墨</div> |       <div class="text-lg font-medium ml-12">小墨</div> | ||||||
| @ -160,15 +160,27 @@ | |||||||
|                       v-for="(file, fileIdx) in msg.content" |                       v-for="(file, fileIdx) in msg.content" | ||||||
|                       :key="fileIdx" |                       :key="fileIdx" | ||||||
|                       style="flex: 0 0 6rem" |                       style="flex: 0 0 6rem" | ||||||
|                       class="relative text-xs h-32 w-80 rounded-md overflow-hidden mr-1 c-black" |                       class="relative text-xs h-32 rounded-md overflow-hidden mr-1 c-black" | ||||||
|                     > |                     > | ||||||
|                       <!-- @click="previewVideo(msg.content)"--> |                       <!-- @click="previewVideo(msg.content)"--> | ||||||
|                       <video |                       <video | ||||||
|  |                         v-if="isVideo(file?.video_url?.poster)" | ||||||
|  |                         :src="file?.video_url?.url" | ||||||
|  |                         class="w-60 h-30" | ||||||
|  |                         controls | ||||||
|  |                         :poster=" | ||||||
|  |                           file?.video_url?.poster + | ||||||
|  |                           '?x-oss-process=video/snapshot,t_1,f_jpg,w_750,h_0,m_fast' | ||||||
|  |                         " | ||||||
|  |                       ></video> | ||||||
|  |                       <video | ||||||
|  |                         v-else | ||||||
|                         :src="file?.video_url?.url" |                         :src="file?.video_url?.url" | ||||||
|                         class="w-60 h-30" |                         class="w-60 h-30" | ||||||
|                         controls |                         controls | ||||||
|                         :poster="file?.video_url?.poster" |                         :poster="file?.video_url?.poster" | ||||||
|                       ></video> |                       ></video> | ||||||
|  |                       <!-- isVideo --> | ||||||
|                       <!-- <video |                       <!-- <video | ||||||
|                         :src="file.tempFilePath" |                         :src="file.tempFilePath" | ||||||
|                         class="w-60 h-30" |                         class="w-60 h-30" | ||||||
| @ -184,11 +196,17 @@ | |||||||
|                   </view> |                   </view> | ||||||
|                 </view> |                 </view> | ||||||
|                 <view |                 <view | ||||||
|                   v-if="msg.role === 'assistant' && msg.type === 'text'" |                   v-if=" | ||||||
|  |                     msg.role === 'assistant' && msg.type === 'text' && messages.length - 1 === idx | ||||||
|  |                   " | ||||||
|                   class="absolute bottom--3.5 flex space-x-3 ml-1" |                   class="absolute bottom--3.5 flex space-x-3 ml-1" | ||||||
|                 > |                 > | ||||||
|                   <image src="/static/aichat/copy.png" class="w-4 h-4" @click="copyText(msg)" /> |                   <image src="/static/aichat/copy.png" class="w-4 h-4" @click="copyText(msg)" /> | ||||||
|                   <image src="/static/aichat/resect.png" class="w-4.3 h-4" @click="refreshText()" /> |                   <image | ||||||
|  |                     src="/static/aichat/resect.png" | ||||||
|  |                     class="w-4.3 h-4" | ||||||
|  |                     @click="refreshText(msg)" | ||||||
|  |                   /> | ||||||
|                 </view> |                 </view> | ||||||
|               </view> |               </view> | ||||||
|               <image |               <image | ||||||
| @ -207,7 +225,7 @@ | |||||||
|     <!-- 底部上传预览 + 输入区 --> |     <!-- 底部上传预览 + 输入区 --> | ||||||
|     <div |     <div | ||||||
|       :class="[ |       :class="[ | ||||||
|         'fixed bottom-0 left-0 right-0 bg-white z-[80] overflow-hidden transition-all duration-300', |         'fixed bottom-0 left-0 right-0 bg-white z-[80] overflow-hidden transition-all duration-300 pb-40rpx', | ||||||
|         showActions ? (uploadList.length ? 'h-40' : 'h-40') : 'h-20', |         showActions ? (uploadList.length ? 'h-40' : 'h-40') : 'h-20', | ||||||
|       ]" |       ]" | ||||||
|     > |     > | ||||||
| @ -303,10 +321,11 @@ | |||||||
|       <div |       <div | ||||||
|         @click="toggleKnowledge" |         @click="toggleKnowledge" | ||||||
|         v-if="!uploadList.length" |         v-if="!uploadList.length" | ||||||
|         class="fixed left-[32rpx] right-0 z-[90] h-8 w-23 flex items-center justify-between px-3 box-border rounded-1 transition-all duration-300" |         style="line-height: 62rpx" | ||||||
|  |         class="fixed left-[32rpx] right-0 z-[90] w-23 flex items-center justify-between px-3 box-border rounded-1 transition-all duration-300" | ||||||
|         :class="[ |         :class="[ | ||||||
|           knowledgeOpen ? 'bg-#eee9f8' : 'bg-[#F9F9F9]', |           knowledgeOpen ? 'bg-#eee9f8' : 'bg-[#F9F9F9]', | ||||||
|           showActions ? (uploadList.length ? 'bottom-31' : 'bottom-41') : 'bottom-21', |           showActions ? (uploadList.length ? 'bottom-31' : 'bottom-46') : 'bottom-26', | ||||||
|         ]" |         ]" | ||||||
|       > |       > | ||||||
|         <image |         <image | ||||||
| @ -332,6 +351,7 @@ | |||||||
|           @focus="onFocus" |           @focus="onFocus" | ||||||
|           @confirm="sendText" |           @confirm="sendText" | ||||||
|           placeholder="想对我说点什么~" |           placeholder="想对我说点什么~" | ||||||
|  |           maxlength="5000" | ||||||
|           class="flex-1 h-10 px-3 border border-gray-100 bg-[#f9f9f9] rounded-1 focus:outline-none" |           class="flex-1 h-10 px-3 border border-gray-100 bg-[#f9f9f9] rounded-1 focus:outline-none" | ||||||
|         /> |         /> | ||||||
|         <!-- 将keyup替换为confirm --> |         <!-- 将keyup替换为confirm --> | ||||||
| @ -349,7 +369,6 @@ | |||||||
|           v-if="sendTextLoading && inputText.length <= 0" |           v-if="sendTextLoading && inputText.length <= 0" | ||||||
|           src="/static/aichat/enter-no.png" |           src="/static/aichat/enter-no.png" | ||||||
|           class="w-7 h-7" |           class="w-7 h-7" | ||||||
|           @click="sendText()" |  | ||||||
|           :disabled="loading" |           :disabled="loading" | ||||||
|           :class="[knowledgeOpen ? 'ml-2' : 'ml-0']" |           :class="[knowledgeOpen ? 'ml-2' : 'ml-0']" | ||||||
|         /> |         /> | ||||||
| @ -375,7 +394,7 @@ | |||||||
|         <view |         <view | ||||||
|           v-show="showActions" |           v-show="showActions" | ||||||
|           :class="[ |           :class="[ | ||||||
|             'flex justify-around items-center h-10 bg-white border-t border-t-solid border-[#E7E7E7]', |             'flex justify-around items-center h-10 bg-white border-t border-t-solid border-[#E7E7E7] pb-20', | ||||||
|             showActions ? (uploadList.length ? 'pt-10' : 'pt-10') : 'pt-0', |             showActions ? (uploadList.length ? 'pt-10' : 'pt-10') : 'pt-0', | ||||||
|           ]" |           ]" | ||||||
|         > |         > | ||||||
| @ -477,6 +496,7 @@ | |||||||
| import { ref, reactive, nextTick, watchEffect, watch } from 'vue' | import { ref, reactive, nextTick, watchEffect, watch } from 'vue' | ||||||
| import dayjs from 'dayjs' | import dayjs from 'dayjs' | ||||||
| import { useUserStore } from '@/store' | import { useUserStore } from '@/store' | ||||||
|  | // import store from '@/store' | ||||||
| import { getEnvBaseUrl } from '@/utils' | import { getEnvBaseUrl } from '@/utils' | ||||||
| import guid from '@/utils/guid.js' | import guid from '@/utils/guid.js' | ||||||
| import type { IGptRequestBody } from '@/service/index/foo' | import type { IGptRequestBody } from '@/service/index/foo' | ||||||
| @ -489,15 +509,17 @@ import { | |||||||
|   officeFileTypeList as fileType, |   officeFileTypeList as fileType, | ||||||
|   videoFileType as videoType, |   videoFileType as videoType, | ||||||
|   picFileType as picType, |   picFileType as picType, | ||||||
|  |   isJsonObject, | ||||||
| } from './utils/index' | } from './utils/index' | ||||||
| import 'dayjs/locale/zh-cn' | import 'dayjs/locale/zh-cn' | ||||||
| import { showToastErr, showToastOk, time_format3 } from '@/utils/tools' | import { showToastErr, showToastOk, time_format3 } from '@/utils/tools' | ||||||
| import { uploadFileChunk } from './utils/api.js' | import { uploadFileChunk } from './utils/api.js' | ||||||
| // import { TOKEN, AVATAR } from './utils/test' | // import { TOKEN, AVATAR } from './utils/test' | ||||||
| import { deepClone } from 'wot-design-uni/components/common/util' | import { deepClone } from 'wot-design-uni/components/common/util' | ||||||
| import { log } from 'console' | import VConsole from 'vconsole' | ||||||
| 
 | 
 | ||||||
| dayjs.locale('zh-cn') | dayjs.locale('zh-cn') | ||||||
|  | const store = useUserStore() | ||||||
| 
 | 
 | ||||||
| interface UploadFile { | interface UploadFile { | ||||||
|   id: string |   id: string | ||||||
| @ -520,8 +542,8 @@ interface IMessage { | |||||||
|   content: string | any[] |   content: string | any[] | ||||||
|   timestamp: Date |   timestamp: Date | ||||||
| } | } | ||||||
| const MAX_FILE_SIZE = 10 * 1024 * 1024 // 5mb属于大文件 | const MAX_FILE_SIZE = 5 * 1024 * 1024 // 5mb属于大文件 | ||||||
| const FILE_SLICE_SIZE = 10 * 1024 * 1024 // 分片大小 | const FILE_SLICE_SIZE = 2 * 1024 * 1024 // 分片大小 | ||||||
| const userAvatar = ref() | const userAvatar = ref() | ||||||
| const chatMode = ref('qwen-vl-plus') | const chatMode = ref('qwen-vl-plus') | ||||||
| let isUserOk = false // 用户确认使用 tongyi-app | let isUserOk = false // 用户确认使用 tongyi-app | ||||||
| @ -598,7 +620,7 @@ const listUuid = ref('') | |||||||
| 
 | 
 | ||||||
| async function createChatSession() { | async function createChatSession() { | ||||||
|   try { |   try { | ||||||
|     const createResp: any = await uni.request({ |     const res: any = await uni.request({ | ||||||
|       url: `${baseUrl}/chat/app/create`, |       url: `${baseUrl}/chat/app/create`, | ||||||
|       method: 'POST', |       method: 'POST', | ||||||
|       data: { |       data: { | ||||||
| @ -609,7 +631,12 @@ async function createChatSession() { | |||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|     // 如果后台返回新的会话信息,可以在这里处理,比如拿到 listUuid 等 |     // 如果后台返回新的会话信息,可以在这里处理,比如拿到 listUuid 等 | ||||||
|     listUuid.value = createResp.data.data.listUuid | 
 | ||||||
|  |     if (res.data.code === 0 && res?.data?.data?.listUuid) { | ||||||
|  |       listUuid.value = res?.data?.data?.listUuid | ||||||
|  |     } else { | ||||||
|  |       showToastErr(res.data.msg) | ||||||
|  |     } | ||||||
|   } catch (err) { |   } catch (err) { | ||||||
|     console.log(err) |     console.log(err) | ||||||
|   } |   } | ||||||
| @ -627,9 +654,9 @@ const state = reactive({ | |||||||
| }) | }) | ||||||
| const scrollTop = ref(0) | const scrollTop = ref(0) | ||||||
| async function fetchHistoryList() { | async function fetchHistoryList() { | ||||||
|   // if(state.page*state.pageSize>state.total && state.total!==null){ |   if (state.page >= state.total / state.pageSize + 1 && state.total !== null) { | ||||||
|   // 	return |     return | ||||||
|   // } |   } | ||||||
|   if (state.loading) { |   if (state.loading) { | ||||||
|     return |     return | ||||||
|   } |   } | ||||||
| @ -643,13 +670,23 @@ async function fetchHistoryList() { | |||||||
|         pageSize: state.pageSize, |         pageSize: state.pageSize, | ||||||
|       }, |       }, | ||||||
|       header: { |       header: { | ||||||
|         Authorization: token.value, |         // Authorization: token.value, | ||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|     if (resp.data && resp.data.data) { |     if (resp.data.code == 0 && resp.data.data) { | ||||||
|  |       if (rawList.value.length > 0) { | ||||||
|         rawList.value = rawList.value.concat(resp.data.data.data) |         rawList.value = rawList.value.concat(resp.data.data.data) | ||||||
|         state.total = resp.data.data.count //Math.ceil(resp.data.count/state.page) |         state.total = resp.data.data.count //Math.ceil(resp.data.count/state.page) | ||||||
|  |       } else { | ||||||
|  |         if (resp?.data?.data?.data) { | ||||||
|  |           rawList.value = resp.data.data.data | ||||||
|  |         } | ||||||
|  |         state.total = resp.data.data.count | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|       // scrollTop.value+=60; |       // scrollTop.value+=60; | ||||||
|  |     } else { | ||||||
|  |       showToastErr(resp.data.msg || '') | ||||||
|     } |     } | ||||||
|   } catch (err) { |   } catch (err) { | ||||||
|     console.log('err: ', err) |     console.log('err: ', err) | ||||||
| @ -662,17 +699,23 @@ const scrolltolowerLoadData = (e) => { | |||||||
| } | } | ||||||
| watch( | watch( | ||||||
|   () => state.page, |   () => state.page, | ||||||
|   async () => { |   async (newval) => { | ||||||
|  |     if (newval <= 0) { | ||||||
|  |       return | ||||||
|  |     } | ||||||
|     await fetchHistoryList() |     await fetchHistoryList() | ||||||
|   }, |   }, | ||||||
|   { deep: true }, |   { deep: true }, | ||||||
| ) | ) | ||||||
| async function openPopup() { | async function openPopup() { | ||||||
|   state.page++ |  | ||||||
|   showPopup.value = true |   showPopup.value = true | ||||||
|  |   state.page++ | ||||||
| } | } | ||||||
| function closePopup() { | function closePopup() { | ||||||
|   showPopup.value = false |   showPopup.value = false | ||||||
|  |   state.page = 0 | ||||||
|  |   state.total = null | ||||||
|  |   rawList.value = [] | ||||||
| } | } | ||||||
| function toggleFullscreen() { | function toggleFullscreen() { | ||||||
|   fullscreen.value = !fullscreen.value |   fullscreen.value = !fullscreen.value | ||||||
| @ -689,15 +732,17 @@ async function fetchHistoryDiets(value) { | |||||||
|         gptModel: chatMode.value, |         gptModel: chatMode.value, | ||||||
|       }, |       }, | ||||||
|       header: { |       header: { | ||||||
|         Authorization: token.value, |         // Authorization: token.value, | ||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|     if (resp && resp.data && resp.data.data) { |     if (resp.data.status === 0 && resp.data && resp.data.data) { | ||||||
|       const rawList = resp.data.data.detail // 假设后端直接返回消息数组 |       const rawList = resp?.data?.data?.detail // 假设后端直接返回消息数组 | ||||||
|       listUuid.value = resp.data.data.listUuid |       listUuid.value = resp?.data?.data?.listUuid | ||||||
|       // const newMessages = parseBackendMessages(JSON.parse(rawList)) |       // const newMessages = parseBackendMessages(JSON.parse(rawList)) | ||||||
|       // 用解析后的消息替换当前消息列表 |       // 用解析后的消息替换当前消息列表 | ||||||
|       messages.splice(0, messages.length, ...JSON.parse(rawList)) |       messages.splice(0, messages.length, ...JSON.parse(rawList)) | ||||||
|  |     } else { | ||||||
|  |       showToastErr(resp.msg || '') | ||||||
|     } |     } | ||||||
|   } catch (err) { |   } catch (err) { | ||||||
|     console.log('err: ', err) |     console.log('err: ', err) | ||||||
| @ -971,19 +1016,37 @@ onMounted(async () => { | |||||||
|   try { |   try { | ||||||
|     const init = async () => { |     const init = async () => { | ||||||
|       const wv = plus.webview.currentWebview() // 获取当前页面所属的 Webview 对象。 |       const wv = plus.webview.currentWebview() // 获取当前页面所属的 Webview 对象。 | ||||||
|       token.value = wv.token || uni.getStorageSync('token') || import.meta.env.VITE_DEV_TOKEN |       token.value = | ||||||
|  |         wv.token || | ||||||
|  |         uni.getStorageSync('token') || | ||||||
|  |         store.userInfo.token || | ||||||
|  |         import.meta.env.VITE_DEV_TOKEN | ||||||
|       userInfo.value = JSON.parse(wv.userInfo) || {} |       userInfo.value = JSON.parse(wv.userInfo) || {} | ||||||
|       refreshToken.value = wv.refreshToken || uni.getStorageSync('refreshToken') |       refreshToken.value = wv.refreshToken || uni.getStorageSync('refreshToken') | ||||||
|       statusBarHeight.value = wv.statusBarHeight || uni.getSystemInfoSync().statusBarHeight |       statusBarHeight.value = wv.statusBarHeight || uni.getSystemInfoSync().statusBarHeight | ||||||
|       userAvatar.value = userInfo.value.Avatar |       userAvatar.value = userInfo.value.Avatar | ||||||
|       mask.value = userInfo.value.ID |       mask.value = userInfo.value.ID | ||||||
|  |       store.setUserInfo({ | ||||||
|  |         token: token.value, | ||||||
|  |         avatar: userInfo.value.Avatar, | ||||||
|  |         refreshToken: refreshToken.value, | ||||||
|  |         statusBarHeight: statusBarHeight.value, | ||||||
|  |       }) | ||||||
|  |       if (userInfo.value.TelNum === '18639432358') { | ||||||
|  |         new VConsole() | ||||||
|  |       } | ||||||
|       await createChatSession() |       await createChatSession() | ||||||
|     } |     } | ||||||
|     init() |     init() | ||||||
|   } catch (e) { |   } catch (e) { | ||||||
|     console.error('onMounted e: ', e) |     console.error('onMounted e: ', e) | ||||||
|   } finally { |   } finally { | ||||||
|  |     // store.userInfo.token = '' | ||||||
|  |     // token.value = store.userInfo.token | ||||||
|   } |   } | ||||||
|  |   // if (JSON.parse(uni.getStorageSync('userInfo'))?.NickName === '常东方') { | ||||||
|  |   //   new VConsole() | ||||||
|  |   // } | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| function scrollToBottom() { | function scrollToBottom() { | ||||||
| @ -1024,7 +1087,6 @@ const previewVideo = (files) => { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function addMessage(msg) { | function addMessage(msg) { | ||||||
|   console.log('msg: ', msg) |  | ||||||
|   messages.push(msg) |   messages.push(msg) | ||||||
|   scrollToBottom() |   scrollToBottom() | ||||||
| } | } | ||||||
| @ -1159,9 +1221,8 @@ const startUploads = () => { | |||||||
|     if (file.size > MAX_FILE_SIZE) { |     if (file.size > MAX_FILE_SIZE) { | ||||||
|       bigFileUpload(file) |       bigFileUpload(file) | ||||||
|         .then((res) => { |         .then((res) => { | ||||||
|           console.log('全部成功 res: ', res) |  | ||||||
|           // const length=res.length; |           // const length=res.length; | ||||||
|           url = res.FullFileUrl |           // file.url = res.FullFileUrl | ||||||
|           file.status = 'success' |           file.status = 'success' | ||||||
|           uploadingCount.value-- |           uploadingCount.value-- | ||||||
|         }) |         }) | ||||||
| @ -1171,7 +1232,23 @@ const startUploads = () => { | |||||||
|           // uploadingCount.value-- |           // uploadingCount.value-- | ||||||
|         }) |         }) | ||||||
|     } else { |     } else { | ||||||
|       uploadFile(file) |       bigFileUpload(file) | ||||||
|  |         .then((res) => { | ||||||
|  |           console.log('全部成功 res: ', res) | ||||||
|  |           // const length=res.length; | ||||||
|  |           file.url = res.FullFileUrl | ||||||
|  |           file.status = 'success' | ||||||
|  |           uploadingCount.value-- | ||||||
|  |           console.log(file) | ||||||
|  |         }) | ||||||
|  |         .catch((rej) => { | ||||||
|  |           console.log('失败rej: ', rej) | ||||||
|  |           file.status = 'error' | ||||||
|  |           // uploadingCount.value-- | ||||||
|  |         }) | ||||||
|  |         .finally(() => { | ||||||
|  |           console.log('filesToUpload', filesToUpload) | ||||||
|  |         }) | ||||||
|     } |     } | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| @ -1255,7 +1332,6 @@ function bigFileUpload(file) { | |||||||
|   file.status = 'uploading' |   file.status = 'uploading' | ||||||
|   uploadingCount.value++ |   uploadingCount.value++ | ||||||
| 
 | 
 | ||||||
|   console.log('file: ', file) //progress |  | ||||||
|   return new Promise(async (resolve, reject) => { |   return new Promise(async (resolve, reject) => { | ||||||
|     const md5 = file.id |     const md5 = file.id | ||||||
|     const fileName = file.name |     const fileName = file.name | ||||||
| @ -1264,7 +1340,7 @@ function bigFileUpload(file) { | |||||||
|     const chunksList = await readFile(file, maxSize) |     const chunksList = await readFile(file, maxSize) | ||||||
|     const chunksLength = chunksList.length |     const chunksLength = chunksList.length | ||||||
|     const requestList = [] |     const requestList = [] | ||||||
|     const maxConcurrency = 3 // 一次最多3个 |     const maxConcurrency = 1 // 一次最多3个 | ||||||
|     let currentRunning = 0 |     let currentRunning = 0 | ||||||
|     let index = 0 |     let index = 0 | ||||||
|     let success = null |     let success = null | ||||||
| @ -1329,10 +1405,17 @@ function bigFileUpload(file) { | |||||||
|             currentRunning-- |             currentRunning-- | ||||||
|             if (chunkIndex === chunksLength - 1) { |             if (chunkIndex === chunksLength - 1) { | ||||||
|               if (errors.length === 0) { |               if (errors.length === 0) { | ||||||
|  |                 if (success.FullFileUrl || success.CoverUrl) { | ||||||
|                   file.status = 'success' |                   file.status = 'success' | ||||||
|  |                   file.FullFileUrl = success.FullFileUrl || success.CoverUrl | ||||||
|  | 
 | ||||||
|                   resolve(success) |                   resolve(success) | ||||||
|                 } else { |                 } else { | ||||||
|                   file.status = 'error' |                   file.status = 'error' | ||||||
|  |                 } | ||||||
|  |               } else { | ||||||
|  |                 file.status = 'error' | ||||||
|  | 
 | ||||||
|                 reject(errors) |                 reject(errors) | ||||||
|               } |               } | ||||||
|             } else if (index < chunksLength) { |             } else if (index < chunksLength) { | ||||||
| @ -1342,48 +1425,6 @@ function bigFileUpload(file) { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     next() |     next() | ||||||
|     // const request = (chunks, i) => { |  | ||||||
|     // 	const chunk = chunks.shift() |  | ||||||
|     // 	const params = { |  | ||||||
|     // 		FileMd5: md5, |  | ||||||
|     // 		Chunk: chunk, |  | ||||||
|     // 		ChunkFileName: `${fileName}_${i}`, |  | ||||||
|     // 		Total: chunksLength, |  | ||||||
|     // 		UseType: 100, |  | ||||||
|     // 		FileName: fileName, |  | ||||||
|     // 		Source: 'aiChat', |  | ||||||
|     // 	} |  | ||||||
| 
 |  | ||||||
|     // 	uploadFileChunk({ formData: params, tempFilePath: tempFilePath }) |  | ||||||
|     // 		.then((res) => { |  | ||||||
|     // 			// requestList.push(Promise.resolve(res)) |  | ||||||
|     // 			if (res.status === 0) { |  | ||||||
|     // 				file.progress = Math.ceil(((i + 1) / chunksLength) * 100) |  | ||||||
|     // 				Promise.resolve(JSON.parse(res.data)) |  | ||||||
|     // 			} else { |  | ||||||
|     // 				Promise.reject(res) |  | ||||||
|     // 			} |  | ||||||
|     // 		}) |  | ||||||
|     // 		.catch((rej) => { |  | ||||||
|     // 			// requestList.push(Promise.resolve(rej)) |  | ||||||
|     // 			Promise.reject(rej) |  | ||||||
|     // 		}) |  | ||||||
|     // 		.finally(() => { |  | ||||||
|     // 			if (chunks.length > 0) { |  | ||||||
|     // 				request(chunks, ++i) |  | ||||||
|     // 			} |  | ||||||
|     // 		}) |  | ||||||
|     // } |  | ||||||
|     // console.log('success,next: ', success, errors); |  | ||||||
|     // const promiseList=chunksList.map(request); |  | ||||||
|     // request(chunksList, 0) |  | ||||||
|     // return Promise.all(chunksList) |  | ||||||
|     // 	.then((res) => { |  | ||||||
|     // 		console.log('all res: ', res) |  | ||||||
|     // 	}) |  | ||||||
|     // 	.catch((rej) => { |  | ||||||
|     // 		console.log('all rej: ', rej) |  | ||||||
|     // 	}) |  | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| @ -1411,12 +1452,41 @@ const onPickImage = () => { | |||||||
|     }, |     }, | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
|  | // 调用原生 Android API 拍摄视频 | ||||||
| 
 | 
 | ||||||
| // 视频 |  | ||||||
| const onPickVideo = () => { | const onPickVideo = () => { | ||||||
|   uni.chooseVideo({ |   uni.chooseVideo({ | ||||||
|     sourceType: ['album', 'camera'], |     // sourceType: ['album', 'camera'], | ||||||
|  |     sourceType: ['album'], | ||||||
|  |     maxDuration: 60, | ||||||
|     compressed: true, |     compressed: true, | ||||||
|  |     camera: 'back', | ||||||
|  |     albumMode: 'custom', | ||||||
|  |     // extension: uploadConfig.video.supportType, | ||||||
|  |     success: (res: any) => { | ||||||
|  |       console.log(res) | ||||||
|  | 
 | ||||||
|  |       const tempFile = res.tempFile | ||||||
|  |       tempFile.path = res.tempFilePath | ||||||
|  | 
 | ||||||
|  |       // 开始上传 | ||||||
|  |       addUploadQueue([tempFile], uploadFileTypeEm.video) | ||||||
|  |     }, | ||||||
|  |     fail: (err) => { | ||||||
|  |       uni.showToast({ | ||||||
|  |         title: '选取视频失败', | ||||||
|  |         icon: 'none', | ||||||
|  |       }) | ||||||
|  |     }, | ||||||
|  |   }) | ||||||
|  | } | ||||||
|  | // 视频 | ||||||
|  | const onPickVideo2 = () => { | ||||||
|  |   uni.chooseVideo({ | ||||||
|  |     sourceType: ['album', 'camera'], | ||||||
|  |     maxDuration: 60, | ||||||
|  |     compressed: true, | ||||||
|  |     camera: 'back', | ||||||
|     // extension: uploadConfig.video.supportType, |     // extension: uploadConfig.video.supportType, | ||||||
|     success: (res: any) => { |     success: (res: any) => { | ||||||
|       console.log(res) |       console.log(res) | ||||||
| @ -1453,7 +1523,6 @@ const onPickFile = () => { | |||||||
|     type: 'all', |     type: 'all', | ||||||
| 
 | 
 | ||||||
|     success: (res: any) => { |     success: (res: any) => { | ||||||
|       console.log(res) |  | ||||||
|       // 开始上传 |       // 开始上传 | ||||||
|       addUploadQueue(res.tempFiles, uploadFileTypeEm.file) |       addUploadQueue(res.tempFiles, uploadFileTypeEm.file) | ||||||
|     }, |     }, | ||||||
| @ -1561,7 +1630,6 @@ const stopMsg = () => { | |||||||
|   stopStreamMsg = true |   stopStreamMsg = true | ||||||
| } | } | ||||||
| async function sendText() { | async function sendText() { | ||||||
|   console.log('uploadList: ', uploadList) |  | ||||||
|   if (uploadList.length > 0) { |   if (uploadList.length > 0) { | ||||||
|     const isUpLoading = uploadList.some((file) => { |     const isUpLoading = uploadList.some((file) => { | ||||||
|       // return file.status==="error" || file.status==="pending" |       // return file.status==="error" || file.status==="pending" | ||||||
| @ -1578,18 +1646,16 @@ async function sendText() { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   const msg = inputText.value.trim() |   const msg = inputText.value.trim() | ||||||
|   if (!msg && !refreshSend.value) { |   if (!msg) { | ||||||
|     return showToastErr('不可以发送空消息!') |     return showToastErr('不可以发送空消息!') | ||||||
|   } |   } | ||||||
|   // if (uploadList.length > 0) { |   // if (uploadList.length > 0) { | ||||||
|   // 	return showToastErr('请等待文件上传完成!') |   // 	return showToastErr('请等待文件上传完成!') | ||||||
|   // } |   // } | ||||||
|   if (!sendTextLoading.value) { |   if (!sendTextLoading.value) { | ||||||
|     sendTextLoading.value = true |  | ||||||
|     return showToastErr('正在接收消息请稍后') |     return showToastErr('正在接收消息请稍后') | ||||||
|   } |   } | ||||||
|   // 开启加载状态 | 
 | ||||||
|   sendTextLoading.value = false |  | ||||||
|   // 文本消息 |   // 文本消息 | ||||||
| 
 | 
 | ||||||
|   // 先判断是否上传文件,若有文件在判断文件类型,视频+图片不能与文档类文件同时上传 |   // 先判断是否上传文件,若有文件在判断文件类型,视频+图片不能与文档类文件同时上传 | ||||||
| @ -1741,7 +1807,7 @@ async function sendText() { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } else { |   } else { | ||||||
|     if (!refreshSend.value) { |     // if (!refreshSend.value) { | ||||||
|     //不重发时触发 |     //不重发时触发 | ||||||
|     addMessage({ |     addMessage({ | ||||||
|       role: 'user', |       role: 'user', | ||||||
| @ -1750,7 +1816,7 @@ async function sendText() { | |||||||
|       timestamp: new Date(), |       timestamp: new Date(), | ||||||
|       mask: 'new', |       mask: 'new', | ||||||
|     }) |     }) | ||||||
|     } |     // } | ||||||
|     // 纯文本时发送 |     // 纯文本时发送 | ||||||
|     chatMode.value = 'tongyi-app' //'tongyi-app'; qwen-long |     chatMode.value = 'tongyi-app' //'tongyi-app'; qwen-long | ||||||
| 
 | 
 | ||||||
| @ -1782,7 +1848,6 @@ async function sendText() { | |||||||
|   // 第一次发送纯文本消息,第二次发送图片+视频,第三次发送文档,此时因为历史消息都要一起发送给后端, |   // 第一次发送纯文本消息,第二次发送图片+视频,第三次发送文档,此时因为历史消息都要一起发送给后端, | ||||||
|   // 所以要想办法在遇到这种情况时,截断历史记录,主动为用户建立一个新的回话,但是不需要清空历史记录 |   // 所以要想办法在遇到这种情况时,截断历史记录,主动为用户建立一个新的回话,但是不需要清空历史记录 | ||||||
| 
 | 
 | ||||||
|   console.log('message: ', messages) |  | ||||||
|   uploadList.splice(0, uploadList.length) // 清空上传的文件 |   uploadList.splice(0, uploadList.length) // 清空上传的文件 | ||||||
|   const list = formatData(messages) |   const list = formatData(messages) | ||||||
| 
 | 
 | ||||||
| @ -1790,9 +1855,49 @@ async function sendText() { | |||||||
|   body.detail = JSON.stringify(messages) |   body.detail = JSON.stringify(messages) | ||||||
|   aiMsg.content = '' |   aiMsg.content = '' | ||||||
|   addMessage(aiMsg) |   addMessage(aiMsg) | ||||||
|  |   console.log('api body', body) | ||||||
|  | 
 | ||||||
|  |   send(body) | ||||||
|   // return |   // return | ||||||
|   // return |   // return | ||||||
|   // 没有上传文件,仅文字消息 |   // 没有上传文件,仅文字消息 | ||||||
|  | } | ||||||
|  | const spliceMsg = (list, model) => { | ||||||
|  |   let file = false | ||||||
|  |   let image = false | ||||||
|  |   let video = false | ||||||
|  |   const length = list.length | ||||||
|  |   console.log(list) | ||||||
|  | 
 | ||||||
|  |   for (let i = length - 1; i >= 0; i--) { | ||||||
|  |     const item = list[i] | ||||||
|  |     if (model === 'tongyi-long' && item.type === 'image' && item.type === 'video') { | ||||||
|  |       const index = length - i - 1 | ||||||
|  |       return list.splice(i - 1) | ||||||
|  |     } else if (model !== 'tongyi-long' && item.type === 'file') { | ||||||
|  |       const index = i | ||||||
|  |       return list.splice(index + 1) | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  |   return list | ||||||
|  | } | ||||||
|  | const send = async (body) => { | ||||||
|  |   // refreshSend.value = true;  // 正在重新发送 | ||||||
|  |   // 开启加载状态 | ||||||
|  |   sendTextLoading.value = false // 接收消息期间不可再次发送 | ||||||
|  |   const [aiMsg] = messages.slice(-1) | ||||||
|  |   const recordList = messages.slice(0, messages.length - 1) | ||||||
|  |   body.detail = JSON.stringify(recordList) | ||||||
|  |   // body.model = chatMode.value // 模型选择 | ||||||
|  |   // body.max_tokens = 1000 | ||||||
|  |   // body.top_p = 1 | ||||||
|  |   // body.presence_penalty = 0 | ||||||
|  |   // body.frequency_penalty = 0 | ||||||
|  |   // messages: deepClone(historyUserMsgs), // text ? [aiMsg] : historyUserMsgs, | ||||||
|  |   // body.stream = true | ||||||
|  |   // body.listUuid = listUuid.value | ||||||
|  |   // console.log('body: ',body); | ||||||
|  |   // body.messages = spliceMsg(body.messages, chatMode.value) | ||||||
|   try { |   try { | ||||||
|     // aiMsg.content = '' |     // aiMsg.content = '' | ||||||
|     // 发送问题到后端 |     // 发送问题到后端 | ||||||
| @ -1801,10 +1906,14 @@ async function sendText() { | |||||||
|     const signal = controller.signal |     const signal = controller.signal | ||||||
|     const resp = await fetch(baseUrl + '/chat/app/completion', { |     const resp = await fetch(baseUrl + '/chat/app/completion', { | ||||||
|       method: 'POST', |       method: 'POST', | ||||||
|       headers: { 'Content-Type': 'application/json', Authorization: token.value }, |       headers: { | ||||||
|  |         'Content-Type': 'application/json', | ||||||
|  |         Authorization: token.value, | ||||||
|  |       }, | ||||||
|       body: JSON.stringify(body), |       body: JSON.stringify(body), | ||||||
|       signal: signal, |       signal: signal, | ||||||
|     }) |     }) | ||||||
|  |     // console.log(resp) | ||||||
| 
 | 
 | ||||||
|     const reader = resp.body!.getReader() |     const reader = resp.body!.getReader() | ||||||
|     const decoder = new TextDecoder() |     const decoder = new TextDecoder() | ||||||
| @ -1832,20 +1941,23 @@ async function sendText() { | |||||||
| 
 | 
 | ||||||
|           if (chunk === '[DONE]') { |           if (chunk === '[DONE]') { | ||||||
|             done = true |             done = true | ||||||
|             console.log('sss') |  | ||||||
|             break |             break | ||||||
|           } |           } | ||||||
|           try { |           try { | ||||||
|             const json = JSON.parse(chunk) |             const json = JSON.parse(chunk) | ||||||
|             const delta = json.choices?.[0]?.delta?.content |             const delta = json.choices?.[0]?.delta?.content | ||||||
|             if (delta) { |             if (delta) { | ||||||
|               msgLoading.value = false |  | ||||||
|               aiMsg.content += delta |               aiMsg.content += delta | ||||||
|  |               // msgLoading.value = false | ||||||
|               //每次更新messages消息,实现流式输出 |               //每次更新messages消息,实现流式输出 | ||||||
|               messages[messages.length - 1] = { ...aiMsg } |               messages[messages.length - 1] = { ...aiMsg } | ||||||
|               scrollToBottom() |               scrollToBottom() | ||||||
|             } |             } | ||||||
|           } catch {} |           } catch (e) { | ||||||
|  |             console.log(e) | ||||||
|  |           } finally { | ||||||
|  |             console.log('over') | ||||||
|  |           } | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         //更新上下文消息 |         //更新上下文消息 | ||||||
| @ -1871,6 +1983,15 @@ async function sendText() { | |||||||
|         console.log('chunk------------------: ') |         console.log('chunk------------------: ') | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |     if (isJsonObject(buffer)) { | ||||||
|  |       const response = JSON.parse(buffer) | ||||||
|  |       if (response.code === 401) { | ||||||
|  |         showToastErr(response.msg) | ||||||
|  |       } else { | ||||||
|  |         showToastErr(response.msg) | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     scrollToBottom() |     scrollToBottom() | ||||||
|   } catch (err) { |   } catch (err) { | ||||||
|     // aiMsg.content = '请重新发送' |     // aiMsg.content = '请重新发送' | ||||||
| @ -1880,7 +2001,7 @@ async function sendText() { | |||||||
|   } finally { |   } finally { | ||||||
|     sendTextLoading.value = true |     sendTextLoading.value = true | ||||||
|     showActions.value = false |     showActions.value = false | ||||||
|     refreshSend.value = false // 重发已经结束 关闭重发 |     // refreshSend.value = false // 重发已经结束 关闭重发 | ||||||
|     msgLoading.value = false |     msgLoading.value = false | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @ -1892,14 +2013,22 @@ function copyText(msg: IMessage) { | |||||||
|         uni.showToast({ title: '已复制', icon: 'success' }) |         uni.showToast({ title: '已复制', icon: 'success' }) | ||||||
|       }, |       }, | ||||||
|       fail(err) { |       fail(err) { | ||||||
|         console.error('复制失败', err) |  | ||||||
|         uni.showToast({ title: '复制失败', icon: 'error' }) |         uni.showToast({ title: '复制失败', icon: 'error' }) | ||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| function refreshText() { | const msgType = (msg) => { | ||||||
|  |   return ( | ||||||
|  |     msg && | ||||||
|  |     msg.role === 'user' && | ||||||
|  |     (msg.type === 'text' || msg.type === 'image' || msg.type === 'video') | ||||||
|  |   ) | ||||||
|  | } | ||||||
|  | function refreshText(msg) { | ||||||
|  |   console.log('refresh msg', msg) | ||||||
|  | 
 | ||||||
|   if (!sendTextLoading.value) { |   if (!sendTextLoading.value) { | ||||||
|     // 正在接收消息,不可以重发 |     // 正在接收消息,不可以重发 | ||||||
|     return |     return | ||||||
| @ -1910,98 +2039,57 @@ function refreshText() { | |||||||
|   // 1. 找到最后两条用户消息(用于处理图文混合场景) |   // 1. 找到最后两条用户消息(用于处理图文混合场景) | ||||||
|   const userMessages = messages.filter((msg) => msg.role === 'user') |   const userMessages = messages.filter((msg) => msg.role === 'user') | ||||||
|   // const lastTwoUserMsgs = userMessages.slice(-2) |   // const lastTwoUserMsgs = userMessages.slice(-2) | ||||||
|   const [msg1, msg2] = deepClone(userMessages.slice(-2)) |   // const [msg1, msg2, msg3] = deepClone(userMessages.slice(-3)) | ||||||
|   // let text=lastTwoUserMsgs[0] // lastTwoUserMsgs.every((msg)=>msg.type==="text" || msg.type==="image" || msg.type==="video" || msg.type==="file") |   const newMsgArr = deepClone(userMessages.slice(-3)) | ||||||
|   // let file=lastTwoUserMsgs[1] // lastTwoUserMsgs.every((msg)=>msg.type==="text" || msg.type==="image" || msg.type==="video" || msg.type==="file") |  | ||||||
| 
 | 
 | ||||||
|   // 2. 提取文本内容和文件列表 |   // 2. 提取文本内容和文件列表 | ||||||
|   let refreshText = null |   let text = null | ||||||
|   const refreshFiles: UploadFile[] = [] |   const refreshFiles: UploadFile[] = [] | ||||||
|   if (msg1 && msg1.type === 'text' && msg2 && msg2.type !== 'text') { |   for (let i = newMsgArr.length - 1; i >= 0; i--) { | ||||||
|     msg1.mask = 'new' |     const msg = newMsgArr[i] | ||||||
|     msg2.mask = 'new' |     if (msg.type === 'text') { | ||||||
|     refreshFiles.push(msg1) |       refreshFiles.unshift(msg) | ||||||
|     refreshFiles.push(msg2) |       break | ||||||
|   } else if (msg1.type === 'text' && msg1.role === 'user' && !msg2) { |  | ||||||
|     msg1.mask = 'new' |  | ||||||
|     refreshFiles.push(msg1) |  | ||||||
|   } else if (msg2.type === 'text' && msg2.role === 'user' && !msg1) { |  | ||||||
|     msg2.mask = 'new' |  | ||||||
|     refreshFiles.push(msg2) |  | ||||||
|     } else { |     } else { | ||||||
|     msg2.mask = 'new' |       refreshFiles.unshift(msg) | ||||||
|     refreshFiles.push(msg2) |     } | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   // lastTwoUserMsgs.forEach((msg,i) => { |  | ||||||
|   // 	console.log('msg: ',msg); |  | ||||||
|   // 	if (msg.type === 'text' && msg.role==="user") { |  | ||||||
|   // 		refreshText = msg.content // 总是取最新的文本 |  | ||||||
|   // 		refreshFiles.push({ |  | ||||||
|   // 			content:msg.content, |  | ||||||
|   // 			type:"text", |  | ||||||
|   // 			role:"user", |  | ||||||
|   // 			timestamp:new Date(), |  | ||||||
|   // 			mask:"new" |  | ||||||
|   // 		}) |  | ||||||
|   // 	} else if(msg.type==="video"){ |  | ||||||
|   // 		msg.mask="new" |  | ||||||
|   // 		refreshFiles.push(msg) |  | ||||||
|   // 		// msg.content.forEach((file : any) => { |  | ||||||
|   // 		// 	console.log('lastTwoUserMsgs file: ',file); |  | ||||||
| 
 |  | ||||||
|   // 		// 	refreshFiles.push({ |  | ||||||
|   // 		// 		id: guid.getGuid(), |  | ||||||
|   // 		// 		url: file.video_url.url, |  | ||||||
|   // 		// 		status: 'success', |  | ||||||
|   // 		// 		name: file.name || '未命名文件', |  | ||||||
|   // 		// 		size: file.size || 0, |  | ||||||
|   // 		// 		uploadFileType: file.uploadFileType || detectFileType(file.video_url.url), |  | ||||||
|   // 		// 	}) |  | ||||||
|   // 		// }) |  | ||||||
|   // 	}else if(msg.type==="image"){ |  | ||||||
|   // 		msg.mask="new" |  | ||||||
|   // 		refreshFiles.push(msg) |  | ||||||
|   // 		// msg.content.forEach((file : any) => { |  | ||||||
|   // 		// 	console.log('lastTwoUserMsgs file: ',file); |  | ||||||
|   // 		// 	refreshFiles.push({ |  | ||||||
|   // 		// 		id: guid.getGuid(), |  | ||||||
|   // 		// 		url: file.image_url.url, |  | ||||||
|   // 		// 		status: 'success', |  | ||||||
|   // 		// 		name: file.name || '未命名文件', |  | ||||||
|   // 		// 		size: file.size || 0, |  | ||||||
|   // 		// 		uploadFileType: file.uploadFileType || detectFileType(file.image_url.url), |  | ||||||
|   // 		// 	}) |  | ||||||
|   // 		// }) |  | ||||||
|   // 	}else{ |  | ||||||
|   // 		msg.mask="new" |  | ||||||
|   // 		refreshFiles.push(msg) |  | ||||||
|   // 		// msg.content.forEach((file : any) => { |  | ||||||
|   // 		// 	console.log('lastTwoUserMsgs file: ',file); |  | ||||||
|   // 		// 	refreshFiles.push({ |  | ||||||
|   // 		// 		id: guid.getGuid(), |  | ||||||
|   // 		// 		url: file.content, |  | ||||||
|   // 		// 		status: 'success', |  | ||||||
|   // 		// 		name: file.name || '未命名文件', |  | ||||||
|   // 		// 		size: file.size || 0, |  | ||||||
|   // 		// 		uploadFileType: file.uploadFileType || detectFileType(file.content), |  | ||||||
|   // 		// 	}) |  | ||||||
|   // 		// }) |  | ||||||
|   // 	} |  | ||||||
|   // }) |  | ||||||
| 
 |  | ||||||
|   // 3. 更新输入框和上传列表 |   // 3. 更新输入框和上传列表 | ||||||
|   // inputText.value = refreshText |   // inputText.value = text | ||||||
|   // uploadList.splice(0, uploadList.length, ...refreshFiles) |   // uploadList.splice(0, uploadList.length, ...refreshFiles) | ||||||
|   refreshFiles.forEach((ele) => { |   refreshFiles.forEach((ele) => { | ||||||
|     messages.push(ele) |     messages.push(ele) | ||||||
|   }) |   }) | ||||||
|   refreshSend.value = true | 
 | ||||||
|   // inputText.value = refreshText |   // return | ||||||
|  | 
 | ||||||
|  |   // refreshSend.value = true | ||||||
|  |   // inputText.value = text | ||||||
|   // 4. 自动触发发送(模拟用户点击发送按钮) |   // 4. 自动触发发送(模拟用户点击发送按钮) | ||||||
|   setTimeout(() => { |   // setTimeout(() => { | ||||||
|     sendText() |   //   sendText() | ||||||
|   }, 100) |   // }, 100) | ||||||
|  |   let list = formatData(messages) | ||||||
|  |   const aiMsg = { | ||||||
|  |     role: 'assistant', | ||||||
|  |     type: 'text', | ||||||
|  |     content: '', | ||||||
|  |     timestamp: new Date(), | ||||||
|  |   } | ||||||
|  |   messages.push(aiMsg) | ||||||
|  |   const body = { | ||||||
|  |     model: chatMode.value, // 模型选择 | ||||||
|  |     max_tokens: 1000, | ||||||
|  |     top_p: 1, | ||||||
|  |     presence_penalty: 0, | ||||||
|  |     frequency_penalty: 0, | ||||||
|  |     messages: list, // text ? [aiMsg] : historyUserMsgs, | ||||||
|  |     stream: true, | ||||||
|  |     listUuid: listUuid.value, | ||||||
|  |     // listUuid:"eff18a10-1719-4528-ad63-ee5c01d0a412" | ||||||
|  |   } | ||||||
|  |   send(body) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| // 文件类型检测函数(根据URL后缀) | // 文件类型检测函数(根据URL后缀) | ||||||
| @ -2075,6 +2163,9 @@ const isFileType = (name: string) => { | |||||||
|   let match = name.match(reg) |   let match = name.match(reg) | ||||||
|   return match ? match[0] : null |   return match ? match[0] : null | ||||||
| } | } | ||||||
|  | const isVideo = (video) => { | ||||||
|  |   return videoFileType.value.includes(isFileType(video)) | ||||||
|  | } | ||||||
| </script> | </script> | ||||||
| 
 | 
 | ||||||
| <style lang="scss" scoped> | <style lang="scss" scoped> | ||||||
|  | |||||||
| @ -2,21 +2,19 @@ import { getEnvBaseUrl } from '@/utils' | |||||||
| import { httpPost } from '@/utils/http' | import { httpPost } from '@/utils/http' | ||||||
| // import { TOKEN } from './test';
 | // import { TOKEN } from './test';
 | ||||||
| import { apis, uploadFile } from '@/utils/tools' | import { apis, uploadFile } from '@/utils/tools' | ||||||
| const baseUrl = getEnvBaseUrl() | const baseUrl = 'https://erpapi.fontree.cn' // 临时使用接口
 | ||||||
|  | // const baseUrl = getEnvBaseUrl()
 | ||||||
| 
 | 
 | ||||||
| // /artwork/get-chunk-list 获取文件分断数据
 | // /artwork/get-chunk-list 获取文件分断数据
 | ||||||
| // /artwork/upload-chunk 分断上传画作图片
 | // /artwork/upload-chunk 分断上传画作图片
 | ||||||
| export const uploadFiles = (url, params) => { | export const uploadFiles = (url, params) => { | ||||||
|   let token = uni.getStorageSync('authorization') |  | ||||||
|   return new Promise((resolve, reject) => { |   return new Promise((resolve, reject) => { | ||||||
|     uni.uploadFile({ |     uni.uploadFile({ | ||||||
|       url: baseUrl + url, |       url: baseUrl + url, | ||||||
|       filePath: params.tempFilePath, |       filePath: params.tempFilePath, | ||||||
|       name: 'Chunk', |       name: 'Chunk', | ||||||
|       formData: params.formData, |       formData: params.formData, | ||||||
|       header: { |       header: {}, | ||||||
|         authorization: token, |  | ||||||
|       }, |  | ||||||
|       complete: (res) => { |       complete: (res) => { | ||||||
|         // console.log('res: ',res);
 |         // console.log('res: ',res);
 | ||||||
|         if (res.statusCode == 200) { |         if (res.statusCode == 200) { | ||||||
| @ -48,5 +46,5 @@ const getChunkList = (file) => { | |||||||
| 
 | 
 | ||||||
| export const uploadFileChunk = (data) => { | export const uploadFileChunk = (data) => { | ||||||
|   let token = uni.getStorageSync('authorization') |   let token = uni.getStorageSync('authorization') | ||||||
|   return uploadFiles('/artwork/upload-chunk', data) |   return uploadFiles('/upload/upload-chunk', data) | ||||||
| } | } | ||||||
|  | |||||||
| @ -203,3 +203,22 @@ async function sendText1(msgData = '') { | |||||||
| 		showActions.value = false | 		showActions.value = false | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | export const TOKEN="79b5c732d96d2b27a48a99dfd4a5566c43aaa5796242e854ebe3ffc198d6876b9628e7b764d9af65ab5dbb2d517ced88170491b74b048c0ba827c0d3741462cb89dc59ed46653a449af837a8262941ca1430937103230a1e32a1715f569f3efdbe6f8cb8b7b8642bd679668081b9b08f693d1b5be6002d936ec51e1e3e0c4927de9e32ac99a109b326e5d2bda27ec87624bb416ec70d2a95a2e190feeba9f0d6bae8571b3dfe89c824712344759a8f2bff9d70747c52525cf6a5614f9c770bca461a9b9c247b6dca97bcf83bbaf99bb726752c4fe1e9a4aa7de5c4cf3e88a3e480801280d45cdc124f9d8221105d852945dc6ce10bc1647e4f09dff4d52ffdfcde73053dc1f269841c964c3b0779ceae38fcd1ac41220de5941cafd00664ae15bb706dfecc00972d1cf3c94b3ddec7758e514d8c0b32e2195e3bcb802d58861ca93e98cf322b9824623cfba4820be34e" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  |         //  if (errors.length === 0) {
 | ||||||
|  |         //         if (success.FullFileUrl || success.CoverUrl) {
 | ||||||
|  |         //           file.status = 'success'
 | ||||||
|  |         //           console.log('success', success)
 | ||||||
|  | 
 | ||||||
|  |         //           resolve(success)
 | ||||||
|  |         //         } else {
 | ||||||
|  |         //           file.status = 'error'
 | ||||||
|  |         //         }
 | ||||||
|  |         //         console.log(success)
 | ||||||
|  |         //       } else {
 | ||||||
|  |         //         file.status = 'error'
 | ||||||
|  |         //         reject(errors)
 | ||||||
|  |         //       }
 | ||||||
| @ -5,6 +5,10 @@ export const fileSuffix = (str) => { | |||||||
|   let reg = /\.\w*$/ |   let reg = /\.\w*$/ | ||||||
|   return str.match(reg)[0] |   return str.match(reg)[0] | ||||||
| } | } | ||||||
|  | export const isJsonObject = (json) => { | ||||||
|  |   const str = json.trim() | ||||||
|  |   return str.startsWith('{') && str.endsWith('}') | ||||||
|  | } | ||||||
| export const officeFileTypeList = ['.docx', '.doc', '.xls', '.xlsx', '.pdf', '.txt'] | export const officeFileTypeList = ['.docx', '.doc', '.xls', '.xlsx', '.pdf', '.txt'] | ||||||
| export const videoFileType = ['.mp4', '.mov', '.wmv'] | export const videoFileType = ['.mp4', '.mov', '.wmv'] | ||||||
| export const picFileType = ['.jpg', '.png', '.jpeg'] | export const picFileType = ['.jpg', '.png', '.jpeg'] | ||||||
| @ -35,10 +39,16 @@ export const formatParams = (uploadList) => { | |||||||
|       } |       } | ||||||
|       videoList.push(media) |       videoList.push(media) | ||||||
|     } else if (officeFileTypeList.includes(fileSuffix(item.ori_url))) { |     } else if (officeFileTypeList.includes(fileSuffix(item.ori_url))) { | ||||||
|  |       const config = { | ||||||
|  |         content: item.url, | ||||||
|  |         filename: item.name, | ||||||
|  |         title: item.name, | ||||||
|  |       } | ||||||
|       let file = { |       let file = { | ||||||
|         role: 'system', |         role: 'system', | ||||||
|         name: item.name, |         name: item.name, | ||||||
|         content: item.url, |         // content: item.url,
 | ||||||
|  |         content: item.url, // item.url,
 | ||||||
|         size: item.size, |         size: item.size, | ||||||
|         mask: 'new', |         mask: 'new', | ||||||
|       } |       } | ||||||
| @ -83,6 +93,8 @@ export function formatData(list) { | |||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } else if (item.type === 'image' || item.type === 'video') { |     } else if (item.type === 'image' || item.type === 'video') { | ||||||
|  |       console.log('media', item) | ||||||
|  | 
 | ||||||
|       // 图片与视频混合在一起
 |       // 图片与视频混合在一起
 | ||||||
|       const content = [] |       const content = [] | ||||||
|       item.content.forEach((child) => { |       item.content.forEach((child) => { | ||||||
| @ -112,10 +124,12 @@ export function formatData(list) { | |||||||
|       result.push({ |       result.push({ | ||||||
|         role: 'user', |         role: 'user', | ||||||
|         content: content, |         content: content, | ||||||
|         type: 'image', |         type: item.type, | ||||||
|         mask: item.mask, |         mask: item.mask, | ||||||
|       }) |       }) | ||||||
|     } else if (item.type === 'file') { |     } else if (item.type === 'file') { | ||||||
|  |       console.log('file', item) | ||||||
|  | 
 | ||||||
|       let content = [] |       let content = [] | ||||||
|       item.content.forEach((child) => { |       item.content.forEach((child) => { | ||||||
|         if (child.role === 'system') { |         if (child.role === 'system') { | ||||||
| @ -171,15 +185,3 @@ export async function readFile(file, chunkSize = 10 * 1024 * 1024) { | |||||||
|   const buffer = await blob.blob() |   const buffer = await blob.blob() | ||||||
|   return sliceFile(buffer, chunkSize) |   return sliceFile(buffer, chunkSize) | ||||||
| } | } | ||||||
| 
 |  | ||||||
| function uploadChunkFile({ chunk, fileName }, index, total, fileId) { |  | ||||||
|   const formData = new FormData() |  | ||||||
|   formData.append('Chunk', chunk) |  | ||||||
|   formData.append('ChunkFileName', `${fileName}_${index}`) |  | ||||||
|   formData.append('total', total) |  | ||||||
|   formData.append('UseType', 100) |  | ||||||
|   formData.append('FileName', fileName) |  | ||||||
|   formData.append('Source', 'aiChat') |  | ||||||
|   formData.append('UseType', 100) |  | ||||||
|   return |  | ||||||
| } |  | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user