Compare commits
	
		
			No commits in common. "06bbec8d62bd7aa5e4bd7ae768b40fb600aef1c1" and "a0cab6f7631ce8d26e8fb180bbd12b3055af7eb5" have entirely different histories.
		
	
	
		
			06bbec8d62
			...
			a0cab6f763
		
	
		
| @ -52,7 +52,6 @@ | |||||||
|     "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", | ||||||
|  | |||||||
| @ -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 = `${token}` |       options.header.Authorization = `Bearer ${token}` | ||||||
|     } |     } | ||||||
|   }, |   }, | ||||||
| } | } | ||||||
|  | |||||||
| @ -160,7 +160,7 @@ | |||||||
|                       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 rounded-md overflow-hidden mr-1 c-black" |                       class="relative text-xs h-32 w-80 rounded-md overflow-hidden mr-1 c-black" | ||||||
|                     > |                     > | ||||||
|                       <!-- @click="previewVideo(msg.content)"--> |                       <!-- @click="previewVideo(msg.content)"--> | ||||||
|                       <video |                       <video | ||||||
| @ -184,17 +184,11 @@ | |||||||
|                   </view> |                   </view> | ||||||
|                 </view> |                 </view> | ||||||
|                 <view |                 <view | ||||||
|                   v-if=" |                   v-if="msg.role === 'assistant' && msg.type === 'text'" | ||||||
|                     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 |                   <image src="/static/aichat/resect.png" class="w-4.3 h-4" @click="refreshText()" /> | ||||||
|                     src="/static/aichat/resect.png" |  | ||||||
|                     class="w-4.3 h-4" |  | ||||||
|                     @click="refreshText(msg)" |  | ||||||
|                   /> |  | ||||||
|                 </view> |                 </view> | ||||||
|               </view> |               </view> | ||||||
|               <image |               <image | ||||||
| @ -338,7 +332,6 @@ | |||||||
|           @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 --> | ||||||
| @ -356,6 +349,7 @@ | |||||||
|           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']" | ||||||
|         /> |         /> | ||||||
| @ -483,7 +477,6 @@ | |||||||
| 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' | ||||||
| @ -496,16 +489,15 @@ 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' | ||||||
| 
 | 
 | ||||||
| dayjs.locale('zh-cn') | dayjs.locale('zh-cn') | ||||||
| const store = useUserStore() |  | ||||||
| 
 | 
 | ||||||
| interface UploadFile { | interface UploadFile { | ||||||
|   id: string |   id: string | ||||||
| @ -529,7 +521,7 @@ interface IMessage { | |||||||
|   timestamp: Date |   timestamp: Date | ||||||
| } | } | ||||||
| const MAX_FILE_SIZE = 10 * 1024 * 1024 // 5mb属于大文件 | const MAX_FILE_SIZE = 10 * 1024 * 1024 // 5mb属于大文件 | ||||||
| const FILE_SLICE_SIZE = 5 * 1024 * 1024 // 分片大小 | const FILE_SLICE_SIZE = 10 * 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 | ||||||
| @ -613,7 +605,7 @@ async function createChatSession() { | |||||||
|         gptModel: chatMode.value, |         gptModel: chatMode.value, | ||||||
|       }, |       }, | ||||||
|       header: { |       header: { | ||||||
|         // Authorization: token.value, |         Authorization: token.value, | ||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|     // 如果后台返回新的会话信息,可以在这里处理,比如拿到 listUuid 等 |     // 如果后台返回新的会话信息,可以在这里处理,比如拿到 listUuid 等 | ||||||
| @ -635,9 +627,9 @@ const state = reactive({ | |||||||
| }) | }) | ||||||
| const scrollTop = ref(0) | const scrollTop = ref(0) | ||||||
| async function fetchHistoryList() { | async function fetchHistoryList() { | ||||||
|   if (state.page >= state.total / state.pageSize + 1 && state.total !== null) { |   // if(state.page*state.pageSize>state.total && state.total!==null){ | ||||||
|     return |   // 	return | ||||||
|   } |   // } | ||||||
|   if (state.loading) { |   if (state.loading) { | ||||||
|     return |     return | ||||||
|   } |   } | ||||||
| @ -651,18 +643,12 @@ 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 && resp.data.data) { | ||||||
|       if (state.total === null) { |       rawList.value = rawList.value.concat(resp.data.data.data) | ||||||
|         rawList.value = resp.data.data.data |       state.total = resp.data.data.count //Math.ceil(resp.data.count/state.page) | ||||||
|         state.total = resp.data.data.count |  | ||||||
|       } else { |  | ||||||
|         rawList.value = rawList.value.concat(resp.data.data.data) |  | ||||||
|         state.total = resp.data.data.count //Math.ceil(resp.data.count/state.page) |  | ||||||
|       } |  | ||||||
| 
 |  | ||||||
|       // scrollTop.value+=60; |       // scrollTop.value+=60; | ||||||
|     } |     } | ||||||
|   } catch (err) { |   } catch (err) { | ||||||
| @ -676,10 +662,7 @@ const scrolltolowerLoadData = (e) => { | |||||||
| } | } | ||||||
| watch( | watch( | ||||||
|   () => state.page, |   () => state.page, | ||||||
|   async (newval) => { |   async () => { | ||||||
|     if (newval <= 0) { |  | ||||||
|       return |  | ||||||
|     } |  | ||||||
|     await fetchHistoryList() |     await fetchHistoryList() | ||||||
|   }, |   }, | ||||||
|   { deep: true }, |   { deep: true }, | ||||||
| @ -687,12 +670,9 @@ watch( | |||||||
| async function openPopup() { | async function openPopup() { | ||||||
|   state.page++ |   state.page++ | ||||||
|   showPopup.value = true |   showPopup.value = true | ||||||
|   rawList.value = [] |  | ||||||
| } | } | ||||||
| function closePopup() { | function closePopup() { | ||||||
|   showPopup.value = false |   showPopup.value = false | ||||||
|   state.page = 0 |  | ||||||
|   state.total = null |  | ||||||
| } | } | ||||||
| function toggleFullscreen() { | function toggleFullscreen() { | ||||||
|   fullscreen.value = !fullscreen.value |   fullscreen.value = !fullscreen.value | ||||||
| @ -709,12 +689,12 @@ 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 && 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)) | ||||||
| @ -991,30 +971,18 @@ onMounted(async () => { | |||||||
|   try { |   try { | ||||||
|     const init = async () => { |     const init = async () => { | ||||||
|       const wv = plus.webview.currentWebview() // 获取当前页面所属的 Webview 对象。 |       const wv = plus.webview.currentWebview() // 获取当前页面所属的 Webview 对象。 | ||||||
|       token.value = |       token.value = wv.token || uni.getStorageSync('token') || import.meta.env.VITE_DEV_TOKEN | ||||||
|         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, |  | ||||||
|       }) |  | ||||||
|       await createChatSession() |       await createChatSession() | ||||||
|     } |     } | ||||||
|     init() |     init() | ||||||
|   } catch (e) { |   } catch (e) { | ||||||
|     console.error('onMounted e: ', e) |     console.error('onMounted e: ', e) | ||||||
|   } finally { |   } finally { | ||||||
|     token.value = store.userInfo.token |  | ||||||
|     await createChatSession() |  | ||||||
|   } |   } | ||||||
| }) | }) | ||||||
| 
 | 
 | ||||||
| @ -1443,60 +1411,12 @@ const onPickImage = () => { | |||||||
|     }, |     }, | ||||||
|   }) |   }) | ||||||
| } | } | ||||||
| // 调用原生 Android API 拍摄视频 |  | ||||||
| 
 |  | ||||||
| const onPickVideo3 = () => { |  | ||||||
|   var cmr = plus.camera.getCamera() |  | ||||||
| 
 |  | ||||||
|   try { |  | ||||||
|     cmr.startVideoCapture( |  | ||||||
|       () => { |  | ||||||
|         alert('ok') |  | ||||||
|       }, |  | ||||||
|       () => { |  | ||||||
|         alert('err') |  | ||||||
|       }, |  | ||||||
|       {}, |  | ||||||
|     ) |  | ||||||
|   } catch (e) { |  | ||||||
|   } finally { |  | ||||||
|     cmr.stopVideoCapture() |  | ||||||
|   } |  | ||||||
| } |  | ||||||
| 
 | 
 | ||||||
|  | // 视频 | ||||||
| const onPickVideo = () => { | const onPickVideo = () => { | ||||||
|   uni.chooseVideo({ |  | ||||||
|     // sourceType: ['album', 'camera'], |  | ||||||
|     sourceType: ['album'], |  | ||||||
|     maxDuration: 60, |  | ||||||
|     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({ |   uni.chooseVideo({ | ||||||
|     sourceType: ['album', 'camera'], |     sourceType: ['album', 'camera'], | ||||||
|     maxDuration: 60, |  | ||||||
|     compressed: true, |     compressed: true, | ||||||
|     camera: 'back', |  | ||||||
|     // extension: uploadConfig.video.supportType, |     // extension: uploadConfig.video.supportType, | ||||||
|     success: (res: any) => { |     success: (res: any) => { | ||||||
|       console.log(res) |       console.log(res) | ||||||
| @ -1641,6 +1561,7 @@ 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" | ||||||
| @ -1657,16 +1578,18 @@ async function sendText() { | |||||||
|     } |     } | ||||||
|   } |   } | ||||||
|   const msg = inputText.value.trim() |   const msg = inputText.value.trim() | ||||||
|   if (!msg) { |   if (!msg && !refreshSend.value) { | ||||||
|     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 | ||||||
|   // 文本消息 |   // 文本消息 | ||||||
| 
 | 
 | ||||||
|   // 先判断是否上传文件,若有文件在判断文件类型,视频+图片不能与文档类文件同时上传 |   // 先判断是否上传文件,若有文件在判断文件类型,视频+图片不能与文档类文件同时上传 | ||||||
| @ -1818,16 +1741,16 @@ async function sendText() { | |||||||
|       } |       } | ||||||
|     } |     } | ||||||
|   } else { |   } else { | ||||||
|     // if (!refreshSend.value) { |     if (!refreshSend.value) { | ||||||
|     //不重发时触发 |       //不重发时触发 | ||||||
|     addMessage({ |       addMessage({ | ||||||
|       role: 'user', |         role: 'user', | ||||||
|       type: 'text', |         type: 'text', | ||||||
|       content: msg, |         content: msg, | ||||||
|       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 | ||||||
| 
 | 
 | ||||||
| @ -1859,6 +1782,7 @@ 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) | ||||||
| 
 | 
 | ||||||
| @ -1866,39 +1790,9 @@ async function sendText() { | |||||||
|   body.detail = JSON.stringify(messages) |   body.detail = JSON.stringify(messages) | ||||||
|   aiMsg.content = '' |   aiMsg.content = '' | ||||||
|   addMessage(aiMsg) |   addMessage(aiMsg) | ||||||
|   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) |  | ||||||
|   // console.log('body: ',body); |  | ||||||
|   // body.messages = spliceMsg(body.messages, chatMode.value) |  | ||||||
|   try { |   try { | ||||||
|     // aiMsg.content = '' |     // aiMsg.content = '' | ||||||
|     // 发送问题到后端 |     // 发送问题到后端 | ||||||
| @ -1907,14 +1801,10 @@ const send = async (body) => { | |||||||
|     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: { |       headers: { 'Content-Type': 'application/json', Authorization: token.value }, | ||||||
|         '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() | ||||||
| @ -1942,23 +1832,20 @@ const send = async (body) => { | |||||||
| 
 | 
 | ||||||
|           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 (e) { |           } catch {} | ||||||
|             console.log(e) |  | ||||||
|           } finally { |  | ||||||
|             console.log('over') |  | ||||||
|           } |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         //更新上下文消息 |         //更新上下文消息 | ||||||
| @ -1984,13 +1871,6 @@ const send = async (body) => { | |||||||
|         console.log('chunk------------------: ') |         console.log('chunk------------------: ') | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     if (isJsonObject(buffer)) { |  | ||||||
|       const response = JSON.parse(buffer) |  | ||||||
|       if (response.code === 401) { |  | ||||||
|         showToastErr('请重新登录') |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     scrollToBottom() |     scrollToBottom() | ||||||
|   } catch (err) { |   } catch (err) { | ||||||
|     // aiMsg.content = '请重新发送' |     // aiMsg.content = '请重新发送' | ||||||
| @ -2000,7 +1880,7 @@ const send = async (body) => { | |||||||
|   } 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 | ||||||
|   } |   } | ||||||
| } | } | ||||||
| @ -2012,22 +1892,14 @@ 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' }) | ||||||
|       }, |       }, | ||||||
|     }) |     }) | ||||||
|   } |   } | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| const msgType = (msg) => { | function refreshText() { | ||||||
|   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 | ||||||
| @ -2038,57 +1910,98 @@ function refreshText(msg) { | |||||||
|   // 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, msg3] = deepClone(userMessages.slice(-3)) |   const [msg1, msg2] = deepClone(userMessages.slice(-2)) | ||||||
|   const newMsgArr = deepClone(userMessages.slice(-3)) |   // let text=lastTwoUserMsgs[0] // lastTwoUserMsgs.every((msg)=>msg.type==="text" || msg.type==="image" || msg.type==="video" || msg.type==="file") | ||||||
|  |   // let file=lastTwoUserMsgs[1] // lastTwoUserMsgs.every((msg)=>msg.type==="text" || msg.type==="image" || msg.type==="video" || msg.type==="file") | ||||||
| 
 | 
 | ||||||
|   // 2. 提取文本内容和文件列表 |   // 2. 提取文本内容和文件列表 | ||||||
|   let text = null |   let refreshText = null | ||||||
|   const refreshFiles: UploadFile[] = [] |   const refreshFiles: UploadFile[] = [] | ||||||
|   for (let i = newMsgArr.length - 1; i >= 0; i--) { |   if (msg1 && msg1.type === 'text' && msg2 && msg2.type !== 'text') { | ||||||
|     const msg = newMsgArr[i] |     msg1.mask = 'new' | ||||||
|     if (msg.type === 'text') { |     msg2.mask = 'new' | ||||||
|       refreshFiles.unshift(msg) |     refreshFiles.push(msg1) | ||||||
|       break |     refreshFiles.push(msg2) | ||||||
|     } else { |   } else if (msg1.type === 'text' && msg1.role === 'user' && !msg2) { | ||||||
|       refreshFiles.unshift(msg) |     msg1.mask = 'new' | ||||||
|     } |     refreshFiles.push(msg1) | ||||||
|  |   } else if (msg2.type === 'text' && msg2.role === 'user' && !msg1) { | ||||||
|  |     msg2.mask = 'new' | ||||||
|  |     refreshFiles.push(msg2) | ||||||
|  |   } else { | ||||||
|  |     msg2.mask = 'new' | ||||||
|  |     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 = text |   // inputText.value = refreshText | ||||||
|   // 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 | ||||||
|   // return |   // inputText.value = refreshText | ||||||
| 
 |  | ||||||
|   // 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后缀) | ||||||
|  | |||||||
| @ -7,6 +7,7 @@ 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, | ||||||
| @ -14,6 +15,7 @@ export const uploadFiles = (url, params) => { | |||||||
|       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);
 | ||||||
|  | |||||||
| @ -202,6 +202,4 @@ async function sendText1(msgData = '') { | |||||||
| 		loading.value = false | 		loading.value = false | ||||||
| 		showActions.value = false | 		showActions.value = false | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 |  | ||||||
| export const TOKEN="79b5c732d96d2b27a48a99dfd4a5566c43aaa5796242e854ebe3ffc198d6876b9628e7b764d9af65ab5dbb2d517ced88170491b74b048c0ba827c0d3741462cb89dc59ed46653a449af837a8262941ca1430937103230a1e32a1715f569f3efdbe6f8cb8b7b8642bd679668081b9b08f693d1b5be6002d936ec51e1e3e0c4927de9e32ac99a109b326e5d2bda27ec87624bb416ec70d2a95a2e190feeba9f0d6bae8571b3dfe89c824712344759a8f2bff9d70747c52525cf6a5614f9c770bca461a9b9c247b6dca97bcf83bbaf99bb726752c4fe1e9a4aa7de5c4cf3e88a3e480801280d45cdc124f9d8221105d852945dc6ce10bc1647e4f09dff4d52ffdfcde73053dc1f269841c964c3b0779ceae38fcd1ac41220de5941cafd00664ae15bb706dfecc00972d1cf3c94b3ddec7758e514d8c0b32e2195e3bcb802d58861ca93e98cf322b9824623cfba4820be34e" |  | ||||||
| @ -5,10 +5,6 @@ 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'] | ||||||
| @ -39,16 +35,10 @@ 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,
 |  | ||||||
|         size: item.size, |         size: item.size, | ||||||
|         mask: 'new', |         mask: 'new', | ||||||
|       } |       } | ||||||
| @ -122,7 +112,7 @@ export function formatData(list) { | |||||||
|       result.push({ |       result.push({ | ||||||
|         role: 'user', |         role: 'user', | ||||||
|         content: content, |         content: content, | ||||||
|         type: item.type, |         type: 'image', | ||||||
|         mask: item.mask, |         mask: item.mask, | ||||||
|       }) |       }) | ||||||
|     } else if (item.type === 'file') { |     } else if (item.type === 'file') { | ||||||
| @ -181,3 +171,15 @@ 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