diff --git a/pages/index/index.vue b/pages/index/index.vue index 74a8861..26d4cef 100644 --- a/pages/index/index.vue +++ b/pages/index/index.vue @@ -222,7 +222,9 @@ const uploadRes = await new Promise((resolve, reject) => { uni.uploadFile({ // url: 'http://192.168.133.83:8000/recognize_speech', - url: 'http://192.168.10.44:8000/recognize_speech', + // url: 'http://192.168.10.44:8000/recognize_speech', + // url: 'http://192.168.1.18:8000/recognize_speech', + url: 'http://106.227.91.181:8000/recognize_speech', filePath: tempFilePath, name: 'speech', // 对应后端的 UploadFile 参数名 formData: { @@ -356,8 +358,8 @@ async getAIResponse(message){ try { // const url = 'http://192.168.133.83:9020/api/chat' - const url = 'http://192.168.10.44:9020/api/chat' - // const url = 'http://106.227.91.181:9020/api/chat' // 如需切换线上,改这里即可 + // const url = 'http://192.168.10.44:9020/api/chat' + const url = 'http://106.227.91.181:9020/api/chat' // 如需切换线上,改这里即可 const headers = { 'Content-Type': 'application/json' } const data = { message } @@ -637,17 +639,20 @@ this.typewriterTimers[messageId] = timer }, scrollToBottom() { - this.$nextTick(() => { - uni.createSelectorQuery().select('.content').boundingClientRect((rect) => { - uni.pageScrollTo({ - scrollTop: rect.height, - duration: 300, - class: '.content' - }); - }).exec(); - }) - - }, + let self = this; + this.$nextTick(() => { + uni.createSelectorQuery().select('.content').boundingClientRect((rect) => { + if(self.height !== rect.height){ + self.height = rect.height; + uni.pageScrollTo({ + scrollTop: rect.height, + duration: 300, + class: '.content' + }); + } + }).exec(); + }) + }, mockReply(text) { const candidates = [ '好的,我已经为您处理。', diff --git a/unpackage/dist/dev/app-plus/app-service.js b/unpackage/dist/dev/app-plus/app-service.js index 087bf1e..390382f 100644 --- a/unpackage/dist/dev/app-plus/app-service.js +++ b/unpackage/dist/dev/app-plus/app-service.js @@ -2210,7 +2210,7 @@ eval("__webpack_require__.r(__webpack_exports__);\n/* harmony import */ var _D_P /***/ (function(module, exports, __webpack_require__) { "use strict"; -eval("/* WEBPACK VAR INJECTION */(function(__f__) {\n\nvar _interopRequireDefault = __webpack_require__(/*! @babel/runtime/helpers/interopRequireDefault */ 1);\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\nvar _regenerator = _interopRequireDefault(__webpack_require__(/*! @babel/runtime/regenerator */ 52));\nvar _asyncToGenerator2 = _interopRequireDefault(__webpack_require__(/*! @babel/runtime/helpers/asyncToGenerator */ 54));\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n\nvar HISTORY_KEY = 'chat_history_groups';\nvar _default = {\n data: function data() {\n return {\n inputText: '',\n messages: [\n // {\n // \tid: 1,\n // \trole: 'user',\n // \ttype: 'text',\n // \tcontent: '帮我统计一下今日的销售数据',\n // \tinputType: 'text'\n // },\n // {\n // \tid: 2,\n // \trole: 'assistant',\n // \ttype: 'card',\n // \ttitle: '今日销售数据统计结果如下:',\n // \tcontent: '内容内容........................'\n // }\n ],\n scrollInto: '',\n drawerOpen: false,\n historyGroups: [],\n isRecording: false,\n isLoading: false,\n willCancel: false,\n recorder: null,\n recordStartY: 0,\n recordStartTs: 0,\n recordSimTimer: null,\n // show: false,\n innerAudioContext: null,\n popupVisible: false,\n typewriterTimers: {}\n };\n },\n computed: {\n timeOfDayText: function timeOfDayText() {\n var h = new Date().getHours();\n if (h < 6) return '凌晨好';\n if (h < 12) return '上午好';\n if (h < 18) return '下午好';\n return '晚上好';\n }\n },\n mounted: function mounted() {\n this.loadChatHistory();\n this.scrollToBottom();\n },\n beforeDestroy: function beforeDestroy() {\n // 清理所有打字机定时器\n Object.values(this.typewriterTimers).forEach(function (timer) {\n if (timer) clearInterval(timer);\n });\n this.typewriterTimers = {};\n // 清理录音定时器\n if (this.recordSimTimer) {\n clearTimeout(this.recordSimTimer);\n }\n },\n methods: {\n // 新增方法:上传音频并识别\n recognizeAudio: function recognizeAudio(tempFilePath) {\n return (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee() {\n var fileInfo, uploadRes, result;\n return _regenerator.default.wrap(function _callee$(_context) {\n while (1) {\n switch (_context.prev = _context.next) {\n case 0:\n _context.prev = 0;\n __f__(\"log\", '开始语音识别,文件路径:', tempFilePath, \" at pages/index/index.vue:208\");\n\n // 获取文件信息\n _context.next = 4;\n return new Promise(function (resolve, reject) {\n uni.getFileInfo({\n filePath: tempFilePath,\n success: resolve,\n fail: reject\n });\n });\n case 4:\n fileInfo = _context.sent;\n __f__(\"log\", '文件大小:', fileInfo.size, \" at pages/index/index.vue:219\");\n\n // 使用 UniApp 的上传文件 API\n _context.next = 8;\n return new Promise(function (resolve, reject) {\n uni.uploadFile({\n // url: 'http://192.168.133.83:8000/recognize_speech',\n url: 'http://192.168.10.44:8000/recognize_speech',\n filePath: tempFilePath,\n name: 'speech',\n // 对应后端的 UploadFile 参数名\n formData: {\n 'format': 'amr',\n 'rate': 16000,\n 'channel': 1,\n 'cuid': 'uniapp_user',\n 'audio_len': fileInfo.size\n },\n success: function success(res) {\n __f__(\"log\", '上传响应:', res, \" at pages/index/index.vue:236\");\n if (res.statusCode === 200) {\n try {\n // 尝试解析返回的 JSON 数据\n var data = JSON.parse(res.data);\n resolve({\n statusCode: 200,\n data: data\n });\n } catch (e) {\n reject(new Error('响应解析失败: ' + e.message));\n }\n } else {\n reject(new Error(\"\\u4E0A\\u4F20\\u5931\\u8D25: \".concat(res.statusCode)));\n }\n },\n fail: function fail(err) {\n reject(new Error('上传请求失败: ' + err.errMsg));\n }\n });\n });\n case 8:\n uploadRes = _context.sent;\n __f__(\"log\", '语音识别响应:', uploadRes, \" at pages/index/index.vue:255\");\n result = uploadRes.data;\n if (!(result.status === 'success')) {\n _context.next = 15;\n break;\n }\n return _context.abrupt(\"return\", result.result);\n case 15:\n throw new Error(result.error || '识别失败');\n case 16:\n _context.next = 23;\n break;\n case 18:\n _context.prev = 18;\n _context.t0 = _context[\"catch\"](0);\n __f__(\"error\", '语音识别错误:', _context.t0, \" at pages/index/index.vue:264\");\n uni.showToast({\n title: '识别失败: ' + (_context.t0.message || '网络错误'),\n icon: 'none'\n });\n return _context.abrupt(\"return\", null);\n case 23:\n case \"end\":\n return _context.stop();\n }\n }\n }, _callee, null, [[0, 18]]);\n }))();\n },\n // ==================== 历史记录管理 ====================\n formatDate: function formatDate(date) {\n var y = date.getFullYear();\n var m = String(date.getMonth() + 1).padStart(2, '0');\n var d = String(date.getDate()).padStart(2, '0');\n return \"\".concat(y, \"\\u5E74\").concat(m, \"\\u6708\").concat(d, \"\\u65E5\");\n },\n loadChatHistory: function loadChatHistory() {\n try {\n var data = uni.getStorageSync(HISTORY_KEY);\n if (data && Array.isArray(data.groups)) {\n this.historyGroups = data.groups;\n } else {\n this.historyGroups = [];\n }\n } catch (e) {\n this.historyGroups = [];\n }\n },\n addToHistory: function addToHistory(text) {\n var _uni$getStorageSync;\n var groups = ((_uni$getStorageSync = uni.getStorageSync(HISTORY_KEY)) === null || _uni$getStorageSync === void 0 ? void 0 : _uni$getStorageSync.groups) || [];\n var today = this.formatDate(new Date());\n var todayGroup = groups.find(function (g) {\n return g.date === today;\n });\n if (!todayGroup) {\n todayGroup = {\n date: today,\n items: []\n };\n groups.unshift(todayGroup);\n }\n if (!todayGroup.items.includes(text)) {\n todayGroup.items.unshift(text);\n }\n\n // 限制大小\n if (todayGroup.items.length > 50) todayGroup.items = todayGroup.items.slice(0, 50);\n if (groups.length > 30) groups = groups.slice(0, 30);\n this.historyGroups = groups;\n uni.setStorageSync(HISTORY_KEY, {\n groups: groups,\n updatedAt: Date.now()\n });\n },\n removeFromHistory: function removeFromHistory(text) {\n var _uni$getStorageSync2;\n var groups = ((_uni$getStorageSync2 = uni.getStorageSync(HISTORY_KEY)) === null || _uni$getStorageSync2 === void 0 ? void 0 : _uni$getStorageSync2.groups) || [];\n groups.forEach(function (group) {\n group.items = group.items.filter(function (item) {\n return item !== text;\n });\n });\n groups = groups.filter(function (g) {\n return g.items.length > 0;\n });\n this.historyGroups = groups;\n uni.setStorageSync(HISTORY_KEY, {\n groups: groups,\n updatedAt: Date.now()\n });\n },\n clearAllHistory: function clearAllHistory() {\n var _this = this;\n uni.showModal({\n title: '清除全部',\n content: '将删除所有对话记录,此操作不可恢复',\n success: function success(res) {\n if (res.confirm) {\n uni.removeStorageSync(HISTORY_KEY);\n _this.historyGroups = [];\n uni.showToast({\n title: '已清除',\n icon: 'success'\n });\n }\n }\n });\n },\n onLongPressHistory: function onLongPressHistory(text) {\n var _this2 = this;\n uni.showModal({\n title: '删除记录',\n content: '确定删除这条对话记录?',\n success: function success(res) {\n if (res.confirm) {\n _this2.removeFromHistory(text);\n }\n }\n });\n },\n // 工具\n removeMessage: function removeMessage(id) {\n var idx = this.messages.findIndex(function (m) {\n return m.id === id;\n });\n if (idx > -1) this.messages.splice(idx, 1);\n },\n addAssistantMessage: function addAssistantMessage(id, content) {\n this.messages.push({\n id: id,\n role: 'assistant',\n type: 'text',\n content: content,\n displayText: ''\n });\n },\n getAIResponse: function getAIResponse(message) {\n return (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee2() {\n var _res$data, _res$data$result, url, headers, data, res;\n return _regenerator.default.wrap(function _callee2$(_context2) {\n while (1) {\n switch (_context2.prev = _context2.next) {\n case 0:\n _context2.prev = 0;\n // const url = 'http://192.168.133.83:9020/api/chat'\n url = 'http://192.168.10.44:9020/api/chat'; // const url = 'http://106.227.91.181:9020/api/chat' // 如需切换线上,改这里即可\n headers = {\n 'Content-Type': 'application/json'\n };\n data = {\n message: message\n };\n __f__(\"log\", data, \" at pages/index/index.vue:364\");\n\n // const [error, res] = await uni.request({\n // url,\n // method: 'POST',\n // header: headers,\n // data\n // })\n\n // console.log(res)\n // 使用 Promise 风格\n _context2.next = 7;\n return new Promise(function (resolve, reject) {\n uni.request({\n url: url,\n method: 'POST',\n header: headers,\n data: data,\n success: function success(res) {\n return resolve(res);\n },\n fail: function fail(err) {\n return reject(err);\n }\n });\n });\n case 7:\n res = _context2.sent;\n __f__(\"log\", '请求响应:', res, \" at pages/index/index.vue:386\");\n if (!(res.statusCode !== 200)) {\n _context2.next = 11;\n break;\n }\n throw new Error(\"HTTP\\u9519\\u8BEF! \\u72B6\\u6001\\u7801: \".concat(res.statusCode));\n case 11:\n return _context2.abrupt(\"return\", ((_res$data = res.data) === null || _res$data === void 0 ? void 0 : (_res$data$result = _res$data.result) === null || _res$data$result === void 0 ? void 0 : _res$data$result.data) || '未获取到有效回复');\n case 14:\n _context2.prev = 14;\n _context2.t0 = _context2[\"catch\"](0);\n __f__(\"error\", 'AI请求错误:', _context2.t0, \" at pages/index/index.vue:395\");\n return _context2.abrupt(\"return\", \"\\u62B1\\u6B49\\uFF0C\\u51FA\\u4E86\\u70B9\\u95EE\\u9898: \".concat(_context2.t0.errMsg || _context2.t0.message));\n case 18:\n case \"end\":\n return _context2.stop();\n }\n }\n }, _callee2, null, [[0, 14]]);\n }))();\n },\n playVoice: function playVoice(voicePath) {\n if (!voicePath) {\n uni.showToast({\n title: '无可播放的语音',\n icon: 'none'\n });\n return;\n }\n if (!this.innerAudioContext) {\n this.innerAudioContext = uni.createInnerAudioContext();\n this.innerAudioContext.autoplay = false;\n this.innerAudioContext.onError(function () {\n uni.showToast({\n title: '播放失败',\n icon: 'none'\n });\n });\n }\n try {\n this.innerAudioContext.stop();\n } catch (e) {}\n __f__(\"log\", voicePath, \" at pages/index/index.vue:413\");\n this.innerAudioContext.src = voicePath;\n this.innerAudioContext.play();\n },\n onSettingTap: function onSettingTap() {\n uni.navigateTo({\n url: '/pages/setting/index'\n });\n },\n onSuggestionTap: function onSuggestionTap(text) {\n this.inputText = text;\n this.onSend();\n },\n onQuickAsk: function onQuickAsk(text) {\n this.inputText = text;\n this.onSend();\n },\n onSwitchModel: function onSwitchModel() {\n uni.showToast({\n title: '已切换为通用模型',\n icon: 'none'\n });\n },\n onInput: function onInput(e) {\n this.inputText = e.detail.value;\n },\n openDrawer: function openDrawer() {\n this.$refs.popup.open();\n },\n onPopupChange: function onPopupChange(e) {\n // e.show: true when opened, false when closed\n this.popupVisible = !!(e && e.show === true);\n },\n // ===== Voice input (WeChat-like) =====\n ensureRecorder: function ensureRecorder() {\n var _this3 = this;\n if (this.recorder) return;\n try {\n this.recorder = uni.getRecorderManager && uni.getRecorderManager();\n } catch (e) {\n this.recorder = null;\n }\n if (this.recorder) {\n this.recorder.onStart();\n this.recorder.onStop( /*#__PURE__*/function () {\n var _ref = (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee3(res) {\n var duration, tempFilePath, recognizedText;\n return _regenerator.default.wrap(function _callee3$(_context3) {\n while (1) {\n switch (_context3.prev = _context3.next) {\n case 0:\n duration = Date.now() - _this3.recordStartTs;\n tempFilePath = res.tempFilePath; // 添加这行,从res中获取文件路径\n if (!(_this3.willCancel || duration < 700)) {\n _context3.next = 5;\n break;\n }\n uni.showToast({\n title: duration < 700 ? '说话时间太短' : '已取消',\n icon: 'none'\n });\n return _context3.abrupt(\"return\");\n case 5:\n // 显示加载\n uni.showLoading({\n title: '识别中...'\n });\n\n // TODO: 上传 res.tempFilePath 做识别;现用 mock\n // this.handleRecognizedText(this.mockSpeechToText(duration), res.tempFilePath, Math.ceil(\n // \tduration / 100) / 10)\n // 真实识别\n _context3.next = 8;\n return _this3.recognizeAudio(tempFilePath);\n case 8:\n recognizedText = _context3.sent;\n uni.hideLoading();\n if (recognizedText) {\n // 成功:填入输入框\n _this3.inputText = recognizedText;\n _this3.$nextTick(function () {\n // 可选:自动发送\n // this.onSend('voice', tempFilePath, Math.ceil(duration / 100) / 10);\n });\n }\n case 11:\n case \"end\":\n return _context3.stop();\n }\n }\n }, _callee3);\n }));\n return function (_x) {\n return _ref.apply(this, arguments);\n };\n }());\n }\n },\n onPressMic: function onPressMic(e) {\n this.ensureRecorder();\n this.isRecording = true;\n // this.show = true\n this.willCancel = false;\n this.recordStartTs = Date.now();\n this.recordStartY = e.changedTouches && e.changedTouches[0] ? e.changedTouches[0].clientY : 0;\n if (this.recorder) {\n try {\n this.recorder.start({\n format: 'amr',\n sampleRate: 16000,\n encodeBitRate: 16000,\n // 编码比特率\n frameSize: 4,\n // 帧大小\n numberOfChannels: 1,\n duration: 60000\n });\n } catch (err) {}\n } else {\n if (this.recordSimTimer) clearTimeout(this.recordSimTimer);\n this.recordSimTimer = setTimeout(function () {}, 60000);\n }\n },\n onMoveMic: function onMoveMic(e) {\n if (!this.isRecording) return;\n var y = e.changedTouches && e.changedTouches[0] ? e.changedTouches[0].clientY : 0;\n this.willCancel = this.recordStartY - y > 60;\n },\n onReleaseMic: function onReleaseMic() {\n __f__(\"log\", 'onReleaseMic', \" at pages/index/index.vue:515\");\n if (!this.isRecording) return;\n var cancel = this.willCancel;\n this.isRecording = false;\n this.show = false;\n if (this.recorder) {\n try {\n this.recorder.stop();\n } catch (err) {\n __f__(\"log\", 'err', err, \" at pages/index/index.vue:524\");\n }\n }\n },\n handleRecognizedText: function handleRecognizedText(text, tempFilePath, duration) {\n if (!text) return;\n this.inputText = text;\n this.onSend('voice', tempFilePath, duration); // 传 'voice'\n },\n mockSpeechToText: function mockSpeechToText(ms) {\n var sec = Math.ceil(ms / 100) / 10;\n var pool = [\"\\u8BED\\u97F3\\u8F93\\u5165 \".concat(sec, \"s\\uFF0C\\u6A21\\u62DF\\u8BC6\\u522B\\uFF1A\\u5E2E\\u6211\\u7EDF\\u8BA1\\u4ECA\\u5929\\u9500\\u552E\\u989D\"), \"\\u8BED\\u97F3\\u8F93\\u5165 \".concat(sec, \"s\\uFF0C\\u6A21\\u62DF\\u8BC6\\u522B\\uFF1A\\u67E5\\u8BE2\\u8BA2\\u535520388993483\"), \"\\u8BED\\u97F3\\u8F93\\u5165 \".concat(sec, \"s\\uFF0C\\u6A21\\u62DF\\u8BC6\\u522B\\uFF1A\\u751F\\u6210\\u65E5\\u62A5\")];\n return pool[Math.floor(Math.random() * pool.length)];\n },\n onHistoryItemTap: function onHistoryItemTap(text) {\n this.inputText = text;\n this.onSend();\n this.$refs.popup.close();\n },\n onSend: function onSend() {\n var _arguments = arguments,\n _this4 = this;\n return (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee4() {\n var inputType, inputContent, duration, text, baseId, loadingId, reply, loadingIdx, replyId, _loadingIdx;\n return _regenerator.default.wrap(function _callee4$(_context4) {\n while (1) {\n switch (_context4.prev = _context4.next) {\n case 0:\n inputType = _arguments.length > 0 && _arguments[0] !== undefined ? _arguments[0] : 'text';\n inputContent = _arguments.length > 1 && _arguments[1] !== undefined ? _arguments[1] : '';\n duration = _arguments.length > 2 && _arguments[2] !== undefined ? _arguments[2] : undefined;\n text = (_this4.inputText || '').trim();\n if (!(!text || _this4.isLoading)) {\n _context4.next = 6;\n break;\n }\n return _context4.abrupt(\"return\");\n case 6:\n baseId = Date.now(); // 1. 用户消息\n _this4.messages.push({\n id: baseId,\n role: 'user',\n type: 'text',\n content: text,\n inputType: inputType,\n inputContent: inputContent,\n duration: duration\n });\n\n // 2. loading 消息\n loadingId = baseId + 0.5;\n _this4.messages.push({\n id: loadingId,\n role: 'assistant',\n loading: true\n });\n _this4.scrollToBottom();\n _this4.inputText = '';\n _this4.isLoading = true;\n _this4.addToHistory(text);\n _context4.prev = 14;\n _context4.next = 17;\n return _this4.getAIResponse(text);\n case 17:\n reply = _context4.sent;\n // 4. 移除 loading\n loadingIdx = _this4.messages.findIndex(function (m) {\n return m.id === loadingId;\n });\n if (loadingIdx > -1) _this4.messages.splice(loadingIdx, 1);\n\n // 5. 添加回复 + 打字机\n replyId = baseId + 1;\n _this4.messages.push({\n id: replyId,\n role: 'assistant',\n type: 'text',\n content: reply,\n displayText: ''\n });\n _this4.typewriter(replyId, reply);\n _context4.next = 30;\n break;\n case 25:\n _context4.prev = 25;\n _context4.t0 = _context4[\"catch\"](14);\n // 出错时也展示\n _loadingIdx = _this4.messages.findIndex(function (m) {\n return m.id === loadingId;\n });\n if (_loadingIdx > -1) _this4.messages.splice(_loadingIdx, 1);\n _this4.messages.push({\n id: baseId + 1,\n role: 'assistant',\n content: \"\\u8BF7\\u6C42\\u51FA\\u9519\\uFF1A\".concat(_context4.t0.message || _context4.t0)\n });\n case 30:\n _context4.prev = 30;\n _this4.isLoading = false;\n _this4.$nextTick(function () {\n return _this4.scrollToBottom();\n });\n return _context4.finish(30);\n case 34:\n case \"end\":\n return _context4.stop();\n }\n }\n }, _callee4, null, [[14, 25, 30, 34]]);\n }))();\n },\n typewriter: function typewriter(messageId, fullText) {\n var _this5 = this;\n var msg = this.messages.find(function (m) {\n return m.id === messageId;\n });\n if (!msg) return;\n // 清理之前的定时器(如果存在)\n if (this.typewriterTimers[messageId]) {\n clearInterval(this.typewriterTimers[messageId]);\n }\n var index = 0;\n msg.displayText = fullText.substring(0, index + 1);\n index += 1;\n var speed = 50; // 每个字符间隔50ms\n var timer = setInterval(function () {\n if (index < fullText.length) {\n msg.displayText = fullText.substring(0, index + 1);\n index++;\n // 打字过程中自动滚动到底部\n _this5.$nextTick(function () {\n _this5.scrollToBottom();\n });\n } else {\n clearInterval(timer);\n delete _this5.typewriterTimers[messageId];\n // 完成后使用完整文本\n msg.displayText = fullText;\n }\n }, speed);\n this.typewriterTimers[messageId] = timer;\n },\n scrollToBottom: function scrollToBottom() {\n this.$nextTick(function () {\n uni.createSelectorQuery().select('.content').boundingClientRect(function (rect) {\n uni.pageScrollTo({\n scrollTop: rect.height,\n duration: 300,\n class: '.content'\n });\n }).exec();\n });\n },\n mockReply: function mockReply(text) {\n var candidates = ['好的,我已经为您处理。', '收到请求,以下是结果的概览。', '我理解了,这是一个示例回复。', '已记录,稍后将完善报表。'];\n var pick = candidates[Math.floor(Math.random() * candidates.length)];\n return pick + '(已收到:“' + text + '”)';\n },\n onListen: function onListen(text) {\n try {\n // H5: Web Speech API\n if (typeof window !== 'undefined' && window.speechSynthesis) {\n var u = new SpeechSynthesisUtterance(String(text));\n u.lang = 'zh-CN';\n u.rate = 1;\n u.pitch = 1;\n window.speechSynthesis.cancel();\n window.speechSynthesis.speak(u);\n return;\n }\n } catch (e) {}\n uni.showToast({\n title: '当前端不支持语音播放',\n icon: 'none'\n });\n }\n }\n};\nexports.default = _default;\n/* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! ./node_modules/@dcloudio/vue-cli-plugin-uni/lib/format-log.js */ 35)[\"default\"]))//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["uni-app:///pages/index/index.vue"],"names":["data","inputText","messages","scrollInto","drawerOpen","historyGroups","isRecording","isLoading","willCancel","recorder","recordStartY","recordStartTs","recordSimTimer","innerAudioContext","popupVisible","typewriterTimers","computed","timeOfDayText","mounted","beforeDestroy","Object","clearTimeout","methods","recognizeAudio","uni","filePath","success","fail","fileInfo","url","name","formData","resolve","statusCode","reject","uploadRes","result","title","icon","formatDate","loadChatHistory","addToHistory","todayGroup","date","items","groups","updatedAt","removeFromHistory","group","clearAllHistory","content","onLongPressHistory","removeMessage","addAssistantMessage","id","role","type","displayText","getAIResponse","headers","message","method","header","res","playVoice","onSettingTap","onSuggestionTap","onQuickAsk","onSwitchModel","onInput","openDrawer","onPopupChange","ensureRecorder","duration","tempFilePath","recognizedText","onPressMic","format","sampleRate","encodeBitRate","frameSize","numberOfChannels","onMoveMic","onReleaseMic","handleRecognizedText","mockSpeechToText","sec","onHistoryItemTap","onSend","inputType","inputContent","text","baseId","loadingId","loading","reply","loadingIdx","replyId","typewriter","clearInterval","msg","index","scrollToBottom","scrollTop","class","mockReply","onListen","u","window"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6IA;AAAA,eACA;EACAA;IACA;MACAC;MACAC;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;MAAA,CACA;MACAC;MACAC;MACAC,iBACA;MACAC;MACAC;MACAC;MACAC;MACAC;MACAC;MACAC;MACA;MACAC;MACAC;MACAC;IACA;EACA;EACAC;IACAC;MACA;MACA;MACA;MACA;MACA;IACA;EACA;EACAC;IACA;IACA;EACA;EACAC;IACA;IACAC;MACA;IACA;IACA;IACA;IACA;MACAC;IACA;EACA;EACAC;IACA;IACAC;MAAA;QAAA;QAAA;UAAA;YAAA;cAAA;gBAAA;gBAEA;;gBAEA;gBAAA;gBAAA,OACA;kBACAC;oBACAC;oBACAC;oBACAC;kBACA;gBACA;cAAA;gBANAC;gBAQA;;gBAEA;gBAAA;gBAAA,OACA;kBACAJ;oBACA;oBACAK;oBACAJ;oBACAK;oBAAA;oBACAC;sBACA;sBACA;sBACA;sBACA;sBACA;oBACA;oBACAL;sBACA;sBACA;wBACA;0BACA;0BACA;0BACAM;4BAAAC;4BAAAjC;0BAAA;wBACA;0BACAkC;wBACA;sBACA;wBACAA;sBACA;oBACA;oBACAP;sBACAO;oBACA;kBACA;gBACA;cAAA;gBA/BAC;gBAiCA;gBAEAC;gBAAA,MACAA;kBAAA;kBAAA;gBAAA;gBAAA,iCACAA;cAAA;gBAAA,MAEA;cAAA;gBAAA;gBAAA;cAAA;gBAAA;gBAAA;gBAGA;gBACAZ;kBACAa;kBACAC;gBACA;gBAAA,iCACA;cAAA;cAAA;gBAAA;YAAA;UAAA;QAAA;MAAA;IAEA;IACA;IACAC;MACA;MACA;MACA;MACA;IACA;IACAC;MACA;QACA;QACA;UACA;QACA;UACA;QACA;MACA;QACA;MACA;IACA;IACAC;MAAA;MACA;MACA;MACA;QAAA;MAAA;MAEA;QACAC;UAAAC;UAAAC;QAAA;QACAC;MACA;MAEA;QACAH;MACA;;MAEA;MACA;MACA;MAEA;MACAlB;QAAAqB;QAAAC;MAAA;IACA;IAEAC;MAAA;MACA;MACAF;QACAG;UAAA;QAAA;MACA;MACAH;QAAA;MAAA;MACA;MACArB;QAAAqB;QAAAC;MAAA;IACA;IAEAG;MAAA;MACAzB;QACAa;QACAa;QACAxB;UACA;YACAF;YACA;YACAA;cAAAa;cAAAC;YAAA;UACA;QACA;MACA;IACA;IACAa;MAAA;MACA3B;QACAa;QACAa;QACAxB;UACA;YACA;UACA;QACA;MACA;IACA;IACA;IACA0B;MACA;QAAA;MAAA;MACA;IACA;IACAC;MACA;QAAAC;QAAAC;QAAAC;QAAAN;QAAAO;MAAA;IACA;IAEAC;MAAA;QAAA;QAAA;UAAA;YAAA;cAAA;gBAAA;gBAEA;gBACA7B,4CACA;gBACA8B;kBAAA;gBAAA;gBACA3D;kBAAA4D;gBAAA;gBAEA;;gBAEA;gBACA;gBACA;gBACA;gBACA;gBACA;;gBAEA;gBACA;gBAAA;gBAAA,OACA;kBACApC;oBACAK;oBACAgC;oBACAC;oBACA9D;oBACA0B;sBAAA;oBAAA;oBACAC;sBAAA;oBAAA;kBACA;gBACA;cAAA;gBATAoC;gBAWA;gBAAA,MAEAA;kBAAA;kBAAA;gBAAA;gBAAA,MACA;cAAA;gBAAA,kCAGA;cAAA;gBAAA;gBAAA;gBAGA;gBAAA,8FACA;cAAA;cAAA;gBAAA;YAAA;UAAA;QAAA;MAAA;IAEA;IAEAC;MACA;QACAxC;UAAAa;UAAAC;QAAA;QACA;MACA;MACA;QACA;QACA;QACA;UACAd;YAAAa;YAAAC;UAAA;QACA;MACA;MACA;QAAA;MAAA;MACA;MACA;MACA;IACA;IACA2B;MACAzC;QACAK;MACA;IACA;IACAqC;MACA;MACA;IACA;IACAC;MACA;MACA;IACA;IACAC;MACA5C;QACAa;QACAC;MACA;IACA;IACA+B;MACA;IACA;IACAC;MACA;IACA;IACAC;MACA;MACA;IACA;IACA;IACAC;MAAA;MACA;MACA;QACA;MACA;QACA;MACA;MACA;QACA;QACA;UAAA;YAAA;YAAA;cAAA;gBAAA;kBAAA;oBACAC;oBACAC;oBAAA,MACA;sBAAA;sBAAA;oBAAA;oBACAlD;sBACAa;sBACAC;oBACA;oBAAA;kBAAA;oBAGA;oBACAd;sBAAAa;oBAAA;;oBAEA;oBACA;oBACA;oBACA;oBAAA;oBAAA,OACA;kBAAA;oBAAAsC;oBACAnD;oBACA;sBACA;sBACA;sBACA;wBACA;wBACA;sBAAA,CACA;oBACA;kBAAA;kBAAA;oBAAA;gBAAA;cAAA;YAAA;UAAA,CACA;UAAA;YAAA;UAAA;QAAA;MACA;IACA;IACAoD;MACA;MACA;MACA;MACA;MACA;MACA;MACA;QACA;UACA;YACAC;YACAC;YACAC;YAAA;YACAC;YAAA;YACAC;YACAR;UACA;QACA;MACA;QACA;QACA;MACA;IACA;IACAS;MACA;MACA;MACA;IACA;IACAC;MACA;MACA;MACA;MACA;MACA;MACA;QACA;UACA;QACA;UACA;QACA;MACA;IACA;IACAC;MACA;MACA;MACA;IACA;IACAC;MACA;MACA,+CACAC,wIACAA,qHACAA,sEACA;MACA;IACA;IACAC;MACA;MACA;MACA;IACA;IACAC;MAAA;QAAA;MAAA;QAAA;QAAA;UAAA;YAAA;cAAA;gBAAAC;gBAAAC;gBAAAjB;gBACAkB;gBAAA,MACA;kBAAA;kBAAA;gBAAA;gBAAA;cAAA;gBAEAC,qBAEA;gBACA;kBACAtC;kBACAC;kBACAC;kBACAN;kBACAuC;kBACAC;kBACAjB;gBACA;;gBAEA;gBACAoB;gBACA;kBACAvC;kBACAC;kBACAuC;gBACA;gBAEA;gBACA;gBACA;gBAEA;gBAAA;gBAAA;gBAAA,OAIA;cAAA;gBAAAC;gBAEA;gBACAC;kBAAA;gBAAA;gBACA;;gBAEA;gBACAC;gBACA;kBACA3C;kBACAC;kBACAC;kBACAN;kBACAO;gBACA;gBAEA;gBAAA;gBAAA;cAAA;gBAAA;gBAAA;gBAEA;gBACAuC;kBAAA;gBAAA;gBACA;gBACA;kBACA1C;kBACAC;kBACAL;gBACA;cAAA;gBAAA;gBAEA;gBACA;kBAAA;gBAAA;gBAAA;cAAA;cAAA;gBAAA;YAAA;UAAA;QAAA;MAAA;IAEA;IACAgD;MAAA;MACA;QAAA;MAAA;MACA;MACA;MACA;QACAC;MACA;MACA;MACAC;MACAC;MACA;MACA;QACA;UACAD;UACAC;UACA;UACA;YACA;UACA;QACA;UACAF;UACA;UACA;UACAC;QACA;MACA;MACA;IACA;IACAE;MACA;QACA9E;UACAA;YACA+E;YACA9B;YACA+B;UACA;QACA;MACA;IAEA;IACAC;MACA,kBACA,eACA,kBACA,kBACA,eACA;MACA;MACA;IACA;IACAC;MACA;QACA;QACA;UACA;UACAC;UACAA;UACAA;UACAC;UACAA;UACA;QACA;MACA;MACApF;QACAa;QACAC;MACA;IACA;EACA;AACA;AAAA,2B","file":"51.js","sourcesContent":["\t<template>\r\n\t\t<view class=\"ai-page\">\r\n\t\t\t<uni-nav-bar  left-icon=\"left\" @clickLeft=\"openDrawer\" title=\"AI对话\">\r\n\t\t\t\t<template v-slot:left>\r\n\t\t\t\t\t<view class=\"hamburger\">\r\n\t\t\t\t\t\t<view class=\"line\" />\r\n\t\t\t\t\t\t<view class=\"line\" />\r\n\t\t\t\t\t\t<view class=\"line\" />\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</template>\r\n\t\t\t\t<template v-slot:right>\r\n\t\t\t\t\t<view class=\"nav-right\">\r\n\t\t\t\t\t\t<!-- <view class=\"gear\" @tap=\"onSettingTap\">⚙️</view> -->\r\n\t\t\t\t\t\t<image src=\"../../static/set.png\" mode=\"widthFix\" @tap=\"onSettingTap\" style=\"width: 18px;\"></image>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</template>\r\n\t\t\t</uni-nav-bar>\r\n\t\t\t<!-- scrollable content -->\r\n\t\t\t<scroll-view class=\"content\" :scroll-y=\"true\" show-scrollbar=\"false\" \r\n\t\t\t\tscroll-with-animation ref=\"scrollView\">\r\n\t\t\t\t<!-- greeting card -->\t\r\n\t\t\t\t<view class=\"greet-card\">\r\n\t\t\t\t\t\t<image src=\"../../static/ai.webp\" mode=\"widthFix\" style=\"width: 60px;margin-right: 10px;\"></image>\r\n\t\t\t\t\t<view class=\"greet-text\">\r\n\t\t\t\t\t\t<view class=\"hi\">HI，{{ timeOfDayText }}</view>\r\n\t\t\t\t\t\t<view class=\"sub\">我是萃星科技智能体</view>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</view>\r\n\r\n\t\t\t\t<!-- welcome sentence -->\r\n\t\t\t\t<view class=\"welcome\">\r\n\t\t\t\t\t您好！非常高兴与您交流，今天有什么可以帮到您？\r\n\t\t\t\t</view>\r\n\r\n\t\t\t\t<!-- suggestions -->\r\n\t\t\t\t<view class=\"guess-panel\">\r\n\t\t\t\t\t<view class=\"guess-title\">猜你想问</view>\r\n\t\t\t\t\t<view class=\"guess-list\">\r\n\t\t\t\t\t\t<view class=\"guess-item\" @tap=\"onSuggestionTap('今日出入库数据')\">\r\n\t\t\t\t\t\t\t<text>今日出入库数据</text>\r\n\t\t\t\t\t\t\t<text class=\"arrow\">›</text>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t<view class=\"guess-item\" @tap=\"onSuggestionTap('今日销售数据')\">\r\n\t\t\t\t\t\t\t<text>今日销售数据</text>\r\n\t\t\t\t\t\t\t<text class=\"arrow\">›</text>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t<view class=\"guess-item\" @tap=\"onSuggestionTap('今日生产数据')\">\r\n\t\t\t\t\t\t\t<text>今日生产数据</text>\r\n\t\t\t\t\t\t\t<text class=\"arrow\">›</text>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</view>\r\n\r\n\t\t\t\t<!-- conversation -->\r\n\t\t\t\t<view class=\"chat\">\r\n\t\t\t\t\t<view v-for=\"m in messages\" :key=\"m.id\" :id=\"'msg-' + m.id\" :class=\"['msg', m.role]\">\r\n\t\t\t\t\t\t<view v-if=\"m.role === 'user'\" class=\"bubble user-bubble\">\r\n\t\t\t\t\t\t\t<text v-if=\"m.inputType === 'text'\">{{ m.content }}</text>\r\n\t\t\t\t\t\t\t<view class=\"text-voice\" v-else @tap=\"playVoice(m.inputContent,m.id)\">\r\n\t\t\t\t\t\t\t\t<text>{{m.duration }}</text>\r\n\t\t\t\t\t\t\t\t<image class=\"voice-play\" src=\"../../static/voice-play.png\" mode=\"widthFix\"></image>\r\n\t\t\t\t\t\t\t</view>\r\n\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t<view v-else class=\"bubble ai-bubble\">\r\n\t\t\t\t\t\t\t<view v-if=\"m.type === 'card'\" class=\"ai-card\">\r\n\t\t\t\t\t\t\t\t<view class=\"ai-card-title\">{{ m.title }}</view>\r\n\t\t\t\t\t\t\t\t<view class=\"ai-card-body\">{{ m.content }}</view>\r\n\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t\t<view v-else-if=\"m.loading\" class=\"ai-loading\">\r\n\t\t\t\t\t\t\t\t<view class=\"loading-dot\"></view>\r\n\t\t\t\t\t\t\t\t<view class=\"loading-dot\"></view>\r\n\t\t\t\t\t\t\t\t<view class=\"loading-dot\"></view>\r\n\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t\t<view v-else>\r\n\t\t\t\t\t\t\t\t<text>{{ m.displayText !== undefined ? m.displayText : m.content }}</text>\r\n\t\t\t\t\t\t\t\t<!-- <text class=\"listen-btn\" @tap=\"onListen(m.content)\">🔊</text> -->\r\n\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</view>\r\n\r\n\t\t\t\t<view style=\"height: 12px;\" />\r\n\t\t\t</scroll-view>\r\n\r\n\t\t\t<!-- bottom dock: quick actions + input bar -->\r\n\t\t\t<view class=\"dock\">\r\n\t\t\t\t<scroll-view class=\"quick-actions horizontal\" scroll-x show-scrollbar=\"false\">\r\n\t\t\t\t\t<view class=\"qa-btn minor\" @tap=\"onSwitchModel\">切换模型</view>\r\n\t\t\t\t\t<view class=\"qa-btn\" @tap=\"onQuickAsk('你是谁？')\">自我介绍</view>\r\n\t\t\t\t\t<view class=\"qa-btn\" @tap=\"onQuickAsk('今日任务有哪些？')\">快捷提问</view>\r\n\t\t\t\t\t<view class=\"qa-btn\" @tap=\"onQuickAsk('展示一份报表示例')\">快捷提问</view>\r\n\t\t\t\t\t<view class=\"qa-btn\" @tap=\"onQuickAsk('生成日报模版')\">快捷提问</view>\r\n\t\t\t\t</scroll-view>\r\n\t\t\t\t<view class=\"input-bar\">\r\n\t\t\t\t\t<input class=\"input\" confirm-type=\"send\" :value=\"inputText\" @input=\"onInput\" @confirm=\"onSend()\"\r\n\t\t\t\t\t\tplaceholder=\"你可以说…\" placeholder-class=\"ph\" />\r\n\t\t\t\t\t<view :class=\"['mic', { recording: isRecording }]\" @touchstart.stop=\"onPressMic\"\r\n\t\t\t\t\t\t@touchmove.stop=\"onMoveMic\" @touchend.stop=\"onReleaseMic\">🎙️</view>\r\n\t\t\t\t\t<button class=\"send\" type=\"primary\" @tap=\"onSend\">发送</button>\r\n\t\t\t\t</view>\r\n\t\t\t</view>\r\n\r\n\t\t\t<!-- left drawer -->、\r\n\t\t\t<uni-popup ref=\"popup\" background-color=\"#fff\" type=\"left\" :z-index=\"10090\" @change=\"onPopupChange\" style=\"z-index: 99999;width: 100vw\" >\r\n\t\t\t\t<view class=\"drawer-mask\">\r\n\t\t\t\t\t<view class=\"drawer\">\r\n\t\t\t\t\t\t<scroll-view class=\"drawer-scroll\" scroll-y show-scrollbar=\"false\">\r\n\t\t\t\t\t\t\t<view v-for=\"g in historyGroups\" :key=\"g.date\" class=\"drawer-group\">\r\n\t\t\t\t\t\t\t\t<view class=\"drawer-date\">{{ g.date }}</view>\r\n\t\t\t\t\t\t\t\t<view v-for=\"(t, idx) in g.items\" :key=\"idx\" class=\"drawer-item\" @tap=\"onHistoryItemTap(t)\" @longpress=\"onLongPressHistory(t)\">\r\n\t\t\t\t\t\t\t\t\t{{ t }}\r\n\t\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t\t\t<view class=\"drawer-divider\" />\r\n\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t</scroll-view>\r\n\t\t\t\t\t\t<view class=\"drawer-footer\">\r\n\t\t\t\t\t\t\t<view class=\"user-icon\">👤</view>\r\n\t\t\t\t\t\t\t<text class=\"user-name\">用户名</text>\r\n\t\t\t\t\t\t\t<view class=\"footer-gear\" @tap=\"clearAllHistory\">⚙️</view>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</view>\r\n\t\t\t</uni-popup>\r\n\r\n\t\t\t<!-- Voice recording overlay -->\r\n\t\t\t<view v-if=\"isRecording\" class=\"record-mask\">\r\n\t\t\t\t<view class=\"record-box\" :class=\"{ cancel: willCancel }\">\r\n\t\t\t\t\t<view class=\"record-icon\">🎙️</view>\r\n\t\t\t\t\t<view class=\"record-text\">{{ willCancel ? '松开手指，取消发送' : '手指上滑，取消发送' }}</view>\r\n\t\t\t\t</view>\r\n\t\t\t</view>\r\n\t\t\t\r\n\t\t\t<view v-if=\"isRecording\" class=\"mask-layer\" @touchmove.stop.prevent>\r\n\t\t\t\t\r\n\t\t\t</view>\r\n\r\n\t\t</view>\r\n\t</template>\r\n\r\n\t<script>\r\n\t\tconst HISTORY_KEY = 'chat_history_groups'\r\n\t\texport default {\r\n\t\t\tdata() {\r\n\t\t\t\treturn {\r\n\t\t\t\t\tinputText: '',\r\n\t\t\t\t\tmessages: [\r\n\t\t\t\t\t\t// {\r\n\t\t\t\t\t\t// \tid: 1,\r\n\t\t\t\t\t\t// \trole: 'user',\r\n\t\t\t\t\t\t// \ttype: 'text',\r\n\t\t\t\t\t\t// \tcontent: '帮我统计一下今日的销售数据',\r\n\t\t\t\t\t\t// \tinputType: 'text'\r\n\t\t\t\t\t\t// },\r\n\t\t\t\t\t\t// {\r\n\t\t\t\t\t\t// \tid: 2,\r\n\t\t\t\t\t\t// \trole: 'assistant',\r\n\t\t\t\t\t\t// \ttype: 'card',\r\n\t\t\t\t\t\t// \ttitle: '今日销售数据统计结果如下：',\r\n\t\t\t\t\t\t// \tcontent: '内容内容........................'\r\n\t\t\t\t\t\t// }\r\n\t\t\t\t\t],\r\n\t\t\t\t\tscrollInto: '',\r\n\t\t\t\t\tdrawerOpen: false,\r\n\t\t\t\t\thistoryGroups: [\r\n\t\t\t\t\t],\r\n\t\t\t\t\tisRecording: false,\r\n\t\t\t\t\tisLoading:false,\r\n\t\t\t\t\twillCancel: false,\r\n\t\t\t\t\trecorder: null,\r\n\t\t\t\t\trecordStartY: 0,\r\n\t\t\t\t\trecordStartTs: 0,\r\n\t\t\t\t\trecordSimTimer: null,\r\n\t\t\t\t\t// show: false,\r\n\t\t\t\t\tinnerAudioContext: null,\r\n\t\t\t\t\tpopupVisible: false,\r\n\t\t\t\t\ttypewriterTimers: {},\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tcomputed: {\r\n\t\t\t\ttimeOfDayText() {\r\n\t\t\t\t\tconst h = new Date().getHours()\r\n\t\t\t\t\tif (h < 6) return '凌晨好'\r\n\t\t\t\t\tif (h < 12) return '上午好'\r\n\t\t\t\t\tif (h < 18) return '下午好'\r\n\t\t\t\t\treturn '晚上好'\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tmounted() {\r\n\t\t\t\tthis.loadChatHistory()\r\n\t\t\t\tthis.scrollToBottom();\r\n\t\t\t},\r\n\t\t\tbeforeDestroy() {\r\n\t\t\t\t// 清理所有打字机定时器\r\n\t\t\t\tObject.values(this.typewriterTimers).forEach(timer => {\r\n\t\t\t\t\tif (timer) clearInterval(timer)\r\n\t\t\t\t})\r\n\t\t\t\tthis.typewriterTimers = {}\r\n\t\t\t\t// 清理录音定时器\r\n\t\t\t\tif (this.recordSimTimer) {\r\n\t\t\t\t\tclearTimeout(this.recordSimTimer)\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tmethods: {\r\n\t\t\t\t// 新增方法：上传音频并识别\r\n\t\t\tasync recognizeAudio(tempFilePath) {\r\n\t\t\t  try {\r\n\t\t\t\tconsole.log('开始语音识别，文件路径:', tempFilePath)\r\n\t\t\t\t\r\n\t\t\t\t// 获取文件信息\r\n\t\t\t\tconst fileInfo = await new Promise((resolve, reject) => {\r\n\t\t\t\t  uni.getFileInfo({\r\n\t\t\t\t\tfilePath: tempFilePath,\r\n\t\t\t\t\tsuccess: resolve,\r\n\t\t\t\t\tfail: reject\r\n\t\t\t\t  })\r\n\t\t\t\t})\r\n\t\t\t\t\r\n\t\t\t\tconsole.log('文件大小:', fileInfo.size)\r\n\t\t\t\t\r\n\t\t\t\t// 使用 UniApp 的上传文件 API\r\n\t\t\t\tconst uploadRes = await new Promise((resolve, reject) => {\r\n\t\t\t\t  uni.uploadFile({\r\n\t\t\t\t\t// url: 'http://192.168.133.83:8000/recognize_speech',\r\n\t\t\t\t\turl: 'http://192.168.10.44:8000/recognize_speech',\r\n\t\t\t\t\tfilePath: tempFilePath,\r\n\t\t\t\t\tname: 'speech', // 对应后端的 UploadFile 参数名\r\n\t\t\t\t\tformData: {\r\n\t\t\t\t\t  'format': 'amr',\r\n\t\t\t\t\t  'rate': 16000,\r\n\t\t\t\t\t  'channel': 1,\r\n\t\t\t\t\t  'cuid': 'uniapp_user',\r\n\t\t\t\t\t  'audio_len': fileInfo.size\r\n\t\t\t\t\t},\r\n\t\t\t\t\tsuccess: (res) => {\r\n\t\t\t\t\t  console.log('上传响应:', res)\r\n\t\t\t\t\t  if (res.statusCode === 200) {\r\n\t\t\t\t\t\ttry {\r\n\t\t\t\t\t\t  // 尝试解析返回的 JSON 数据\r\n\t\t\t\t\t\t  const data = JSON.parse(res.data)\r\n\t\t\t\t\t\t  resolve({ statusCode: 200, data })\r\n\t\t\t\t\t\t} catch (e) {\r\n\t\t\t\t\t\t  reject(new Error('响应解析失败: ' + e.message))\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t  } else {\r\n\t\t\t\t\t\treject(new Error(`上传失败: ${res.statusCode}`))\r\n\t\t\t\t\t  }\r\n\t\t\t\t\t},\r\n\t\t\t\t\tfail: (err) => {\r\n\t\t\t\t\t  reject(new Error('上传请求失败: ' + err.errMsg))\r\n\t\t\t\t\t}\r\n\t\t\t\t  })\r\n\t\t\t\t})\r\n\t\t\t\r\n\t\t\t\tconsole.log('语音识别响应:', uploadRes)\r\n\t\t\t\r\n\t\t\t\tconst result = uploadRes.data\r\n\t\t\t\tif (result.status === 'success') {\r\n\t\t\t\t  return result.result\r\n\t\t\t\t} else {\r\n\t\t\t\t  throw new Error(result.error || '识别失败')\r\n\t\t\t\t}\r\n\t\t\t  } catch (error) {\r\n\t\t\t\tconsole.error('语音识别错误:', error)\r\n\t\t\t\tuni.showToast({\r\n\t\t\t\t  title: '识别失败: ' + (error.message || '网络错误'),\r\n\t\t\t\t  icon: 'none'\r\n\t\t\t\t})\r\n\t\t\t\treturn null\r\n\t\t\t  }\r\n\t\t\t},\r\n\t\t\t\t// ==================== 历史记录管理 ====================\r\n\t\t\t\t\tformatDate(date) {\r\n\t\t\t\t\t  const y = date.getFullYear()\r\n\t\t\t\t\t  const m = String(date.getMonth() + 1).padStart(2, '0')\r\n\t\t\t\t\t  const d = String(date.getDate()).padStart(2, '0')\r\n\t\t\t\t\t  return `${y}年${m}月${d}日`\r\n\t\t\t\t\t},\r\n\t\t\t\t\tloadChatHistory() {\r\n\t\t\t\t\t\t  try {\r\n\t\t\t\t\t\t\tconst data = uni.getStorageSync(HISTORY_KEY)\r\n\t\t\t\t\t\t\tif (data && Array.isArray(data.groups)) {\r\n\t\t\t\t\t\t\t  this.historyGroups = data.groups\r\n\t\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\t  this.historyGroups = []\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t  } catch (e) {\r\n\t\t\t\t\t\t\tthis.historyGroups = []\r\n\t\t\t\t\t\t  }\r\n\t\t\t\t\t\t},\r\n\t\t\t\t\taddToHistory(text) {\r\n\t\t\t\t\t\t  let groups = uni.getStorageSync(HISTORY_KEY)?.groups || []\r\n\t\t\t\t\t\t  const today = this.formatDate(new Date())\r\n\t\t\t\t\t\t  let todayGroup = groups.find(g => g.date === today)\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t  if (!todayGroup) {\r\n\t\t\t\t\t\t\ttodayGroup = { date: today, items: [] }\r\n\t\t\t\t\t\t\tgroups.unshift(todayGroup)\r\n\t\t\t\t\t\t  }\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t  if (!todayGroup.items.includes(text)) {\r\n\t\t\t\t\t\t\ttodayGroup.items.unshift(text)\r\n\t\t\t\t\t\t  }\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t  // 限制大小\r\n\t\t\t\t\t\t  if (todayGroup.items.length > 50) todayGroup.items = todayGroup.items.slice(0, 50)\r\n\t\t\t\t\t\t  if (groups.length > 30) groups = groups.slice(0, 30)\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t  this.historyGroups = groups\r\n\t\t\t\t\t\t  uni.setStorageSync(HISTORY_KEY, { groups, updatedAt: Date.now() })\r\n\t\t\t\t\t\t},\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\tremoveFromHistory(text) {\r\n\t\t\t\t\t\t\t  let groups = uni.getStorageSync(HISTORY_KEY)?.groups || []\r\n\t\t\t\t\t\t\t  groups.forEach(group => {\r\n\t\t\t\t\t\t\t\tgroup.items = group.items.filter(item => item !== text)\r\n\t\t\t\t\t\t\t  })\r\n\t\t\t\t\t\t\t  groups = groups.filter(g => g.items.length > 0)\r\n\t\t\t\t\t\t\t  this.historyGroups = groups\r\n\t\t\t\t\t\t\t  uni.setStorageSync(HISTORY_KEY, { groups, updatedAt: Date.now() })\r\n\t\t\t\t\t\t\t},\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\tclearAllHistory() {\r\n\t\t\t\t\t\t\t  uni.showModal({\r\n\t\t\t\t\t\t\t\ttitle: '清除全部',\r\n\t\t\t\t\t\t\t\tcontent: '将删除所有对话记录，此操作不可恢复',\r\n\t\t\t\t\t\t\t\tsuccess: (res) => {\r\n\t\t\t\t\t\t\t\t  if (res.confirm) {\r\n\t\t\t\t\t\t\t\t\tuni.removeStorageSync(HISTORY_KEY)\r\n\t\t\t\t\t\t\t\t\tthis.historyGroups = []\r\n\t\t\t\t\t\t\t\t\tuni.showToast({ title: '已清除', icon: 'success' })\r\n\t\t\t\t\t\t\t\t  }\r\n\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t  })\r\n\t\t\t\t\t\t\t},\r\n\t\t\t\t\t\tonLongPressHistory(text) {\r\n\t\t\t\t\t\t\t  uni.showModal({\r\n\t\t\t\t\t\t\t\ttitle: '删除记录',\r\n\t\t\t\t\t\t\t\tcontent: '确定删除这条对话记录？',\r\n\t\t\t\t\t\t\t\tsuccess: (res) => {\r\n\t\t\t\t\t\t\t\t  if (res.confirm) {\r\n\t\t\t\t\t\t\t\t\tthis.removeFromHistory(text)\r\n\t\t\t\t\t\t\t\t  }\r\n\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t  })\r\n\t\t\t\t\t\t\t},\t\t\r\n\t\t\t\t\t// 工具\r\n\t\t\t\t\t\tremoveMessage(id) {\r\n\t\t\t\t\t\t  const idx = this.messages.findIndex(m => m.id === id)\r\n\t\t\t\t\t\t  if (idx > -1) this.messages.splice(idx, 1)\r\n\t\t\t\t\t\t},\r\n\t\t\t\t\t\taddAssistantMessage(id, content) {\r\n\t\t\t\t\t\t\t  this.messages.push({ id, role: 'assistant', type: 'text', content, displayText: '' })\r\n\t\t\t\t\t\t\t},\r\n\t\t\t\t\t\t\t\t\r\n\t\t\t\tasync getAIResponse(message){\r\n\t\t\t\t\ttry {\r\n\t\t\t\t\t\t\t// const url = 'http://192.168.133.83:9020/api/chat'\r\n\t\t\t\t\t\t\tconst url = 'http://192.168.10.44:9020/api/chat'\r\n\t\t\t\t\t\t\t// const url = 'http://106.227.91.181:9020/api/chat'  // 如需切换线上，改这里即可\r\n\t\t\t\t\t\t\tconst headers = { 'Content-Type': 'application/json' }\r\n\t\t\t\t\t\t\tconst data = { message }\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\t\tconsole.log(data)\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t  //       const [error, res] = await uni.request({\r\n\t\t\t\t\t  //         url,\r\n\t\t\t\t\t  //         method: 'POST',\r\n\t\t\t\t\t  //         header: headers,\r\n\t\t\t\t\t  //         data\r\n\t\t\t\t\t  //       })\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\t\t// console.log(res)\r\n\t\t\t\t\t\t\t// 使用 Promise 风格\r\n\t\t\t\t\t\t\tconst res = await new Promise((resolve, reject) => {\r\n\t\t\t\t\t\t\t  uni.request({\r\n\t\t\t\t\t\t\t\turl,\r\n\t\t\t\t\t\t\t\tmethod: 'POST',\r\n\t\t\t\t\t\t\t\theader: headers,\r\n\t\t\t\t\t\t\t\tdata,\r\n\t\t\t\t\t\t\t\tsuccess: (res) => resolve(res),\r\n\t\t\t\t\t\t\t\tfail: (err) => reject(err)\r\n\t\t\t\t\t\t\t  })\r\n\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t\r\n\t\t\t\t\t\t\tconsole.log('请求响应:', res)\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t\tif (res.statusCode !== 200) {\r\n\t\t\t\t\t\t\t\t  throw new Error(`HTTP错误! 状态码: ${res.statusCode}`)\r\n\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\t\t\treturn res.data?.result?.data || '未获取到有效回复'\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\t\t  } catch (error) {\r\n\t\t\t\t\t\t\t\tconsole.error('AI请求错误:', error)\r\n\t\t\t\t\t\t\t\treturn `抱歉，出了点问题: ${error.errMsg || error.message}`\r\n\t\t\t\t\t\t\t  }\r\n\t\t\t\t},\r\n\t\t\t\t\r\n\t\t\t\tplayVoice(voicePath) {\r\n\t\t\t\t\tif (!voicePath) {\r\n\t\t\t\t\t\tuni.showToast({ title: '无可播放的语音', icon: 'none' })\r\n\t\t\t\t\t\treturn\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif (!this.innerAudioContext) {\r\n\t\t\t\t\t\tthis.innerAudioContext = uni.createInnerAudioContext()\r\n\t\t\t\t\t\tthis.innerAudioContext.autoplay = false\r\n\t\t\t\t\t\tthis.innerAudioContext.onError(() => {\r\n\t\t\t\t\t\t\tuni.showToast({ title: '播放失败', icon: 'none' })\r\n\t\t\t\t\t\t});\r\n\t\t\t\t\t}\r\n\t\t\t\t\ttry { this.innerAudioContext.stop() } catch(e) {}\r\n\t\t\t\t\tconsole.log(voicePath)\r\n\t\t\t\t\tthis.innerAudioContext.src = voicePath\r\n\t\t\t\t\tthis.innerAudioContext.play()\r\n\t\t\t\t},\r\n\t\t\t\tonSettingTap() {\r\n\t\t\t\t\tuni.navigateTo({\r\n\t\t\t\t\t\turl: '/pages/setting/index'\r\n\t\t\t\t\t})\r\n\t\t\t\t},\r\n\t\t\t\tonSuggestionTap(text) {\r\n\t\t\t\t\tthis.inputText = text\r\n\t\t\t\t\tthis.onSend();\r\n\t\t\t\t},\r\n\t\t\t\tonQuickAsk(text) {\r\n\t\t\t\t\tthis.inputText = text\r\n\t\t\t\t\tthis.onSend()\r\n\t\t\t\t},\r\n\t\t\t\tonSwitchModel() {\r\n\t\t\t\t\tuni.showToast({\r\n\t\t\t\t\t\ttitle: '已切换为通用模型',\r\n\t\t\t\t\t\ticon: 'none'\r\n\t\t\t\t\t})\r\n\t\t\t\t},\r\n\t\t\t\tonInput(e) {\r\n\t\t\t\t\tthis.inputText = e.detail.value\r\n\t\t\t\t},\r\n\t\t\t\topenDrawer() {\r\n\t\t\t\t\tthis.$refs.popup.open()\r\n\t\t\t\t},\r\n\t\t\t\tonPopupChange(e){\r\n\t\t\t\t\t// e.show: true when opened, false when closed\r\n\t\t\t\t\tthis.popupVisible = !!(e && (e.show === true))\r\n\t\t\t\t},\r\n\t\t\t\t// ===== Voice input (WeChat-like) =====\r\n\t\t\t\tensureRecorder() {\r\n\t\t\t\t\tif (this.recorder) return\r\n\t\t\t\t\ttry {\r\n\t\t\t\t\t\tthis.recorder = uni.getRecorderManager && uni.getRecorderManager()\r\n\t\t\t\t\t} catch (e) {\r\n\t\t\t\t\t\tthis.recorder = null\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif (this.recorder) {\r\n\t\t\t\t\t\tthis.recorder.onStart()\r\n\t\t\t\t\t\tthis.recorder.onStop( async(res) => {\r\n\t\t\t\t\t\t\tconst duration = Date.now() - this.recordStartTs;\r\n\t\t\t\t\t\t\tconst tempFilePath = res.tempFilePath; // 添加这行，从res中获取文件路径\r\n\t\t\t\t\t\t\tif (this.willCancel || duration < 700) {\r\n\t\t\t\t\t\t\t\tuni.showToast({\r\n\t\t\t\t\t\t\t\t\ttitle: duration < 700 ? '说话时间太短' : '已取消',\r\n\t\t\t\t\t\t\t\t\ticon: 'none'\r\n\t\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t\t\treturn\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t// 显示加载\r\n\t\t\t\t\t\t\tuni.showLoading({ title: '识别中...' });\r\n\t\t\t\t\t\t\t\t  \r\n\t\t\t\t\t\t\t// TODO: 上传 res.tempFilePath 做识别；现用 mock\r\n\t\t\t\t\t\t\t// this.handleRecognizedText(this.mockSpeechToText(duration), res.tempFilePath, Math.ceil(\r\n\t\t\t\t\t\t\t// \tduration / 100) / 10)\r\n\t\t\t\t\t\t\t// 真实识别\r\n\t\t\t\t\t\t\tconst recognizedText = await this.recognizeAudio(tempFilePath);\r\n\t\t\t\t\t\t\tuni.hideLoading();\t \r\n\t\t\t\t\t\t\t if (recognizedText) {\r\n\t\t\t\t\t\t\t\t\t // 成功：填入输入框\r\n\t\t\t\t\t\t\t\t\t this.inputText = recognizedText;\r\n\t\t\t\t\t\t\t\t\t this.$nextTick(() => {\r\n\t\t\t\t\t\t\t\t\t   // 可选：自动发送\r\n\t\t\t\t\t\t\t\t\t   // this.onSend('voice', tempFilePath, Math.ceil(duration / 100) / 10);\r\n\t\t\t\t\t\t\t\t\t });\r\n\t\t\t\t\t\t\t\t   }\r\n\t\t\t\t\t\t})\r\n\t\t\t\t\t}\r\n\t\t\t\t},\r\n\t\t\t\tonPressMic(e) {\r\n\t\t\t\t\tthis.ensureRecorder()\r\n\t\t\t\t\tthis.isRecording = true\r\n\t\t\t\t\t// this.show = true\r\n\t\t\t\t\tthis.willCancel = false\r\n\t\t\t\t\tthis.recordStartTs = Date.now()\r\n\t\t\t\t\tthis.recordStartY = (e.changedTouches && e.changedTouches[0]) ? e.changedTouches[0].clientY : 0\r\n\t\t\t\t\tif (this.recorder) {\r\n\t\t\t\t\t\ttry {\r\n\t\t\t\t\t\t\tthis.recorder.start({\r\n\t\t\t\t\t\t\t\tformat: 'amr',\r\n\t\t\t\t\t\t\t\tsampleRate: 16000,\r\n\t\t\t\t\t\t\t\tencodeBitRate: 16000, // 编码比特率\r\n\t\t\t\t\t\t\t\tframeSize: 4, // 帧大小\r\n\t\t\t\t\t\t\t\tnumberOfChannels: 1,\r\n\t\t\t\t\t\t\t\tduration: 60000\r\n\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t} catch (err) {}\r\n\t\t\t\t\t} else {\r\n\t\t\t\t\t\tif (this.recordSimTimer) clearTimeout(this.recordSimTimer)\r\n\t\t\t\t\t\tthis.recordSimTimer = setTimeout(() => {}, 60000)\r\n\t\t\t\t\t}\r\n\t\t\t\t},\r\n\t\t\t\tonMoveMic(e) {\r\n\t\t\t\t\tif (!this.isRecording) return;\r\n\t\t\t\t\tconst y = (e.changedTouches && e.changedTouches[0]) ? e.changedTouches[0].clientY : 0;\r\n\t\t\t\t\tthis.willCancel = (this.recordStartY - y) > 60\r\n\t\t\t\t},\r\n\t\t\t\tonReleaseMic() {\r\n\t\t\t\t\tconsole.log('onReleaseMic');\r\n\t\t\t\t\tif (!this.isRecording) return\r\n\t\t\t\t\tconst cancel = this.willCancel\r\n\t\t\t\t\tthis.isRecording = false;\r\n\t\t\t\t\tthis.show = false\r\n\t\t\t\t\tif (this.recorder) {\r\n\t\t\t\t\t\ttry {\r\n\t\t\t\t\t\t\tthis.recorder.stop()\r\n\t\t\t\t\t\t} catch (err) {\r\n\t\t\t\t\t\t\tconsole.log('err', err);\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t},\r\n\t\t\t\thandleRecognizedText(text, tempFilePath, duration) {\r\n\t\t\t\t\tif (!text) return\r\n\t\t\t\t\tthis.inputText = text\r\n\t\t\t\t\tthis.onSend('voice', tempFilePath, duration) // 传 'voice'\r\n\t\t\t\t},\r\n\t\t\t\tmockSpeechToText(ms) {\r\n\t\t\t\t\tconst sec = Math.ceil(ms / 100) / 10\r\n\t\t\t\t\tconst pool = [\r\n\t\t\t\t\t\t`语音输入 ${sec}s，模拟识别：帮我统计今天销售额`,\r\n\t\t\t\t\t\t`语音输入 ${sec}s，模拟识别：查询订单20388993483`,\r\n\t\t\t\t\t\t`语音输入 ${sec}s，模拟识别：生成日报`\r\n\t\t\t\t\t]\r\n\t\t\t\t\treturn pool[Math.floor(Math.random() * pool.length)]\r\n\t\t\t\t},\r\n\t\t\t\tonHistoryItemTap(text) {\r\n\t\t\t\t\tthis.inputText = text\r\n\t\t\t\t\tthis.onSend();\r\n\t\t\t\t\tthis.$refs.popup.close()\r\n\t\t\t\t},\r\n\t\t\t\tasync onSend(inputType = 'text', inputContent = '', duration = undefined) {\r\n\t\t\t\t  const text = (this.inputText || '').trim()\r\n\t\t\t\t  if (!text || this.isLoading) return\r\n\t\t\t\t\r\n\t\t\t\t  const baseId = Date.now()\r\n\t\t\t\t\r\n\t\t\t\t  // 1. 用户消息\r\n\t\t\t\t  this.messages.push({\r\n\t\t\t\t\tid: baseId,\r\n\t\t\t\t\trole: 'user',\r\n\t\t\t\t\ttype: 'text',\r\n\t\t\t\t\tcontent: text,\r\n\t\t\t\t\tinputType,\r\n\t\t\t\t\tinputContent,\r\n\t\t\t\t\tduration\r\n\t\t\t\t  })\r\n\t\t\t\t\r\n\t\t\t\t  // 2. loading 消息\r\n\t\t\t\t  const loadingId = baseId + 0.5\r\n\t\t\t\t  this.messages.push({\r\n\t\t\t\t\tid: loadingId,\r\n\t\t\t\t\trole: 'assistant',\r\n\t\t\t\t\tloading: true\r\n\t\t\t\t  })\r\n\t\t\t\t\r\n\t\t\t\t  this.scrollToBottom()\r\n\t\t\t\t  this.inputText = ''\r\n\t\t\t\t  this.isLoading = true\r\n\t\t\t\t\r\n\t\t\t\t  this.addToHistory(text)\r\n\t\t\t\t\r\n\t\t\t\t  try {\r\n\t\t\t\t\t// 3. 真正等待 AI 回复\r\n\t\t\t\t\tconst reply = await this.getAIResponse(text)\r\n\t\t\t\t\r\n\t\t\t\t\t// 4. 移除 loading\r\n\t\t\t\t\tconst loadingIdx = this.messages.findIndex(m => m.id === loadingId)\r\n\t\t\t\t\tif (loadingIdx > -1) this.messages.splice(loadingIdx, 1)\r\n\t\t\t\t\r\n\t\t\t\t\t// 5. 添加回复 + 打字机\r\n\t\t\t\t\tconst replyId = baseId + 1\r\n\t\t\t\t\tthis.messages.push({\r\n\t\t\t\t\t  id: replyId,\r\n\t\t\t\t\t  role: 'assistant',\r\n\t\t\t\t\t  type: 'text',\r\n\t\t\t\t\t  content: reply,\r\n\t\t\t\t\t  displayText: ''\r\n\t\t\t\t\t})\r\n\t\t\t\t\r\n\t\t\t\t\tthis.typewriter(replyId, reply)\r\n\t\t\t\t  } catch (e) {\r\n\t\t\t\t\t// 出错时也展示\r\n\t\t\t\t\tconst loadingIdx = this.messages.findIndex(m => m.id === loadingId)\r\n\t\t\t\t\tif (loadingIdx > -1) this.messages.splice(loadingIdx, 1)\r\n\t\t\t\t\tthis.messages.push({\r\n\t\t\t\t\t  id: baseId + 1,\r\n\t\t\t\t\t  role: 'assistant',\r\n\t\t\t\t\t  content: `请求出错：${e.message || e}`\r\n\t\t\t\t\t})\r\n\t\t\t\t  } finally {\r\n\t\t\t\t\tthis.isLoading = false\r\n\t\t\t\t\tthis.$nextTick(() => this.scrollToBottom())\r\n\t\t\t\t  }\r\n\t\t\t\t},\r\n\t\t\t\ttypewriter(messageId, fullText) {\r\n\t\t\t\t\tconst msg = this.messages.find(m => m.id === messageId)\r\n\t\t\t\t\tif (!msg) return\r\n\t\t\t\t\t// 清理之前的定时器（如果存在）\r\n\t\t\t\t\tif (this.typewriterTimers[messageId]) {\r\n\t\t\t\t\t\tclearInterval(this.typewriterTimers[messageId])\r\n\t\t\t\t\t}\r\n\t\t\t\t\tlet index = 0\r\n\t\t\t\t\tmsg.displayText = fullText.substring(0, index + 1);\r\n\t\t\t\t\tindex += 1;\r\n\t\t\t\t\tconst speed = 50 // 每个字符间隔50ms\r\n\t\t\t\t\tconst timer = setInterval(() => {\r\n\t\t\t\t\t\tif (index < fullText.length) {\r\n\t\t\t\t\t\t\tmsg.displayText = fullText.substring(0, index + 1)\r\n\t\t\t\t\t\t\tindex++\r\n\t\t\t\t\t\t\t// 打字过程中自动滚动到底部\r\n\t\t\t\t\t\t\tthis.$nextTick(() => {\r\n\t\t\t\t\t\t\t\tthis.scrollToBottom()\r\n\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\tclearInterval(timer)\r\n\t\t\t\t\t\t\tdelete this.typewriterTimers[messageId]\r\n\t\t\t\t\t\t\t// 完成后使用完整文本\r\n\t\t\t\t\t\t\tmsg.displayText = fullText\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}, speed)\r\n\t\t\t\t\tthis.typewriterTimers[messageId] = timer\r\n\t\t\t\t},\r\n\t\t\t\tscrollToBottom() {\r\n\t\t\t\t\tthis.$nextTick(() => {\r\n\t\t\t\t\t\tuni.createSelectorQuery().select('.content').boundingClientRect((rect) => {\r\n\t\t\t\t\t\t\tuni.pageScrollTo({\r\n\t\t\t\t\t\t\t\tscrollTop: rect.height,\r\n\t\t\t\t\t\t\t\tduration: 300,\r\n\t\t\t\t\t\t\t\tclass: '.content'\r\n\t\t\t\t\t\t\t});\r\n\t\t\t\t\t\t}).exec();\r\n\t\t\t\t\t})\r\n\r\n\t\t\t\t},\r\n\t\t\t\tmockReply(text) {\r\n\t\t\t\t\tconst candidates = [\r\n\t\t\t\t\t\t'好的，我已经为您处理。',\r\n\t\t\t\t\t\t'收到请求，以下是结果的概览。',\r\n\t\t\t\t\t\t'我理解了，这是一个示例回复。',\r\n\t\t\t\t\t\t'已记录，稍后将完善报表。'\r\n\t\t\t\t\t]\r\n\t\t\t\t\tconst pick = candidates[Math.floor(Math.random() * candidates.length)]\r\n\t\t\t\t\treturn pick + '（已收到：“' + text + '”）'\r\n\t\t\t\t},\r\n\t\t\t\tonListen(text) {\r\n\t\t\t\t\ttry {\r\n\t\t\t\t\t\t// H5: Web Speech API\r\n\t\t\t\t\t\tif (typeof window !== 'undefined' && window.speechSynthesis) {\r\n\t\t\t\t\t\t\tconst u = new SpeechSynthesisUtterance(String(text))\r\n\t\t\t\t\t\t\tu.lang = 'zh-CN'\r\n\t\t\t\t\t\t\tu.rate = 1\r\n\t\t\t\t\t\t\tu.pitch = 1\r\n\t\t\t\t\t\t\twindow.speechSynthesis.cancel()\r\n\t\t\t\t\t\t\twindow.speechSynthesis.speak(u)\r\n\t\t\t\t\t\t\treturn\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t} catch (e) {}\r\n\t\t\t\t\tuni.showToast({\r\n\t\t\t\t\t\ttitle: '当前端不支持语音播放',\r\n\t\t\t\t\t\ticon: 'none'\r\n\t\t\t\t\t})\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t</script>\r\n\r\n\t<style scoped>\r\n\t\t::v-deep .uni-nav-bar-text{\r\n\t\t\tfont-size: 18px !important;\r\n\t\t}\r\n\t\t\r\n\t\t::v-deeo .uni-navbar--border{\r\n\t\t\t/* border-bottom: 0px !important; */\r\n\t\t\tborder-bottom: 1px solid #fff !important;\r\n\t\t}\r\n\t\t\r\n\t\t.ai-page {\r\n\t\t\theight: 100vh;\r\n\t\t\tdisplay: flex;\r\n\t\t\tflex-direction: column;\r\n\t\t\tbackground: #f7f8fc;\r\n\t\t}\r\n\r\n\t\t.nav {\r\n\t\t\theight: 44px;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: space-between;\r\n\t\t\tpadding: 0 12px;\r\n\t\t\tbackground: #ffffff;\r\n\t\t\tbox-shadow: 0 2px 8px rgba(0, 0, 0, 0.04);\r\n\t\t\tposition: fixed;\r\n\t\t\tleft: 0;\r\n\t\t\tright: 0;\r\n\t\t\ttop: 0;\r\n\t\t\tz-index: 9;\r\n\t\t}\r\n\r\n\t\t.nav-title {\r\n\t\t\tfont-size: 16px;\r\n\t\t\tfont-weight: 600;\r\n\t\t}\r\n\r\n\t\t.hamburger {\r\n\t\t\twidth: 18px;\r\n\t\t}\r\n\r\n\t\t.hamburger .line {\r\n\t\t\theight: 2px;\r\n\t\t\tbackground: #333;\r\n\t\t\tmargin: 3px 0;\r\n\t\t\tborder-radius: 2px;\r\n\t\t}\r\n\r\n\t\t.gear {\r\n\t\t\twidth: 18px;\r\n\t\t\theight: 18px;\r\n\t\t\tposition: relative;\r\n\t\t\tcolor: #000;\r\n\t\t}\r\n\r\n\t\t.content {\r\n\t\t\tflex: 1;\r\n\t\t\tpadding: 16px 12px 68px 12px;\r\n\t\t\tbackground-color: #f7f8fc;\r\n\t\t\twidth: 100%;\r\n\t\t\tbox-sizing: border-box;\r\n\t\t}\r\n\r\n\t\t.greet-card {\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 14px;\r\n\t\t\tpadding: 12px;\r\n\t\t\tmargin-bottom: 10px;\r\n\t\t}\r\n\r\n\t\t.avatar-inner {\r\n\t\t\tfont-size: 26px;\r\n\t\t}\r\n\r\n\t\t.greet-text .hi {\r\n\t\t\tfont-size: 16px;\r\n\t\t\tfont-weight: 700;\r\n\t\t\tcolor: #0b56ff;\r\n\t\t}\r\n\r\n\t\t.greet-text .sub {\r\n\t\t\tfont-size: 12px;\r\n\t\t\tcolor: #4a76b1;\r\n\t\t\tmargin-top: 4px;\r\n\t\t}\r\n\r\n\t\t.welcome {\r\n\t\t\tfont-size: 13px;\r\n\t\t\tcolor: #333;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 12px;\r\n\t\t\tpadding: 10px 12px;\r\n\t\t\tmargin: 12px 0;\r\n\t\t}\r\n\r\n\t\t.guess-panel {\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 14px;\r\n\t\t\tpadding: 10px;\r\n\t\t\tmargin-bottom: 16px;\r\n\t\t}\r\n\r\n\t\t.guess-title {\r\n\t\t\tcolor: #5f6fff;\r\n\t\t\tfont-size: 14px;\r\n\t\t\tmargin-bottom: 8px;\r\n\t\t}\r\n\r\n\t\t.guess-list {\r\n\t\t\tdisplay: flex;\r\n\t\t\tflex-direction: column;\r\n\t\t}\r\n\r\n\t\t.guess-item {\r\n\t\t\tbackground: #f7f8fc;\r\n\t\t\tborder-radius: 10px;\r\n\t\t\tpadding: 12px;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: space-between;\r\n\t\t\tmargin-bottom: 10px;\r\n\t\t\tbox-sizing: border-box;\r\n\t\t}\r\n\r\n\t\t.guess-item:last-child {\r\n\t\t\tmargin-bottom: 0;\r\n\t\t}\r\n\r\n\t\t.guess-item .arrow {\r\n\t\t\tcolor: #9aa3b2;\r\n\t\t\tfont-size: 18px;\r\n\t\t}\r\n\r\n\t\t.chat {\r\n\t\t\tmargin: 6px 0 12px;\r\n\t\t}\r\n\r\n\t\t.msg {\r\n\t\t\tmargin: 10px 0;\r\n\t\t\tdisplay: flex;\r\n\t\t}\r\n\r\n\t\t.msg.user {\r\n\t\t\tjustify-content: flex-end;\r\n\t\t}\r\n\r\n\t\t.bubble {\r\n\t\t\tmax-width: 80%;\r\n\t\t\tpadding: 10px 12px;\r\n\t\t\tborder-radius: 14px;\r\n\t\t\tfont-size: 14px;\r\n\t\t\tline-height: 1.5;\r\n\t\t}\r\n\r\n\t\t.user-bubble {\r\n\t\t\tbackground: #4e7bff;\r\n\t\t\tcolor: #fff;\r\n\t\t\tborder-bottom-right-radius: 4px;\r\n\t\t\tmargin-right: 6px;\r\n\t\t}\r\n\r\n\t\t.ai-bubble {\r\n\t\t\tbackground: #fff;\r\n\t\t\tcolor: #333;\r\n\t\t\tborder-bottom-left-radius: 4px;\r\n\t\t\tbox-shadow: 0 2px 8px rgba(0, 0, 0, 0.06);\r\n\t\t}\r\n\r\n\t\t.listen-btn {\r\n\t\t\tmargin-left: 8px;\r\n\t\t\tcolor: #6b7280;\r\n\t\t\tfont-size: 14px;\r\n\t\t}\r\n\r\n\t\t.ai-card-title {\r\n\t\t\tcolor: #5f6fff;\r\n\t\t\tfont-weight: 600;\r\n\t\t\tmargin-bottom: 6px;\r\n\t\t}\r\n\r\n\t\t.ai-card-body {\r\n\t\t\tcolor: #666;\r\n\t\t}\r\n\r\n\t\t/* loading animation */\r\n\t\t.ai-loading {\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tgap: 6px;\r\n\t\t\tpadding: 4px 0;\r\n\t\t}\r\n\r\n\t\t.loading-dot {\r\n\t\t\twidth: 8px;\r\n\t\t\theight: 8px;\r\n\t\t\tborder-radius: 50%;\r\n\t\t\tbackground: #9ca3af;\r\n\t\t\tanimation: loading-bounce 1.4s ease-in-out infinite both;\r\n\t\t}\r\n\r\n\t\t.loading-dot:nth-child(1) {\r\n\t\t\tanimation-delay: -0.32s;\r\n\t\t}\r\n\r\n\t\t.loading-dot:nth-child(2) {\r\n\t\t\tanimation-delay: -0.16s;\r\n\t\t}\r\n\r\n\t\t@keyframes loading-bounce {\r\n\t\t\t0%, 80%, 100% {\r\n\t\t\t\ttransform: scale(0.8);\r\n\t\t\t\topacity: 0.5;\r\n\t\t\t}\r\n\t\t\t40% {\r\n\t\t\t\ttransform: scale(1.2);\r\n\t\t\t\topacity: 1;\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t/* bottom dock */\r\n\t\t.dock {\r\n\t\t\tposition: fixed;\r\n\t\t\tleft: 0;\r\n\t\t\tright: 0;\r\n\t\t\tbottom: 0;\r\n\t\t\tbackground: #f7f8fc;\r\n\t\t\tbox-shadow: 0 -4px 12px rgba(0, 0, 0, 0.06);\r\n\t\t\tpadding-bottom: constant(safe-area-inset-bottom);\r\n\t\t\tpadding-bottom: env(safe-area-inset-bottom);\r\n\t\t}\r\n\r\n\t\t.quick-actions {\r\n\t\t\tpadding: 6px 10px 4px;\r\n\t\t}\r\n\r\n\t\t.quick-actions.horizontal {\r\n\t\t\twhite-space: nowrap;\r\n\t\t\twidth: 95%;\r\n\t\t}\r\n\r\n\t\t.qa-btn {\r\n\t\t\tdisplay: inline-flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: center;\r\n\t\t\tmin-width: 96px;\r\n\t\t\ttext-align: center;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 10px;\r\n\t\t\tpadding: 8px 10px;\r\n\t\t\tfont-size: 12px;\r\n\t\t\tcolor: #3b3f45;\r\n\t\t\tbox-shadow: 0 2px 8px rgba(0, 0, 0, 0.04);\r\n\t\t\tmargin-right: 10px;\r\n\t\t}\r\n\r\n\t\t.qa-btn.minor {\r\n\t\t\tbackground: #eff1ff;\r\n\t\t\tcolor: #4e7bff;\r\n\t\t}\r\n\r\n\t\t.qa-btn:last-child {\r\n\t\t\tmargin-right: 0;\r\n\t\t}\r\n\r\n\t\t.input-bar {\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tpadding: 8px 10px 12px;\r\n\t\t\tgap: 8px;\r\n\t\t\tbackground: #f7f8fc;\r\n\t\t}\r\n\r\n\t\t.input {\r\n\t\t\tflex: 1;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 24px;\r\n\t\t\tpadding: 10px 14px;\r\n\t\t\tfont-size: 14px;\r\n\t\t}\r\n\r\n\t\t.ph {\r\n\t\t\tcolor: #9aa3b2;\r\n\t\t}\r\n\r\n\t\t.mic {\r\n\t\t\twidth: 36px;\r\n\t\t\theight: 36px;\r\n\t\t\tborder-radius: 18px;\r\n\t\t\tbackground: #fff;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: center;\r\n\t\t\tbox-shadow: 0 2px 8px rgba(0, 0, 0, 0.06);\r\n\t\t}\r\n\r\n\t\t.mic.recording {\r\n\t\t\tbackground: #fffbf0;\r\n\t\t\tbox-shadow: 0 0 0 2px rgba(255, 193, 7, .25) inset;\r\n\t\t}\r\n\r\n\t\t.send {\r\n\t\t\theight: 36px;\r\n\t\t\tline-height: 36px;\r\n\t\t\tpadding: 0 14px;\r\n\t\t\tborder-radius: 18px;\r\n\t\t\tbackground: #4e7bff;\r\n\t\t\tcolor: #fff;\r\n\t\t\tfont-size: 14px;\r\n\t\t}\r\n\r\n\t\t/* drawer */\r\n\t\t.drawer-mask {\r\n\t\t\twidth: 70vw;\r\n\t\t\theight: 100vh;\r\n\t\t}\r\n\r\n\t\t.drawer {\r\n\t\t\twidth: 100%;\r\n\t\t\theight: 100vh;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-top-right-radius: 8px;\r\n\t\t\tborder-bottom-right-radius: 8px;\r\n\t\t\tdisplay: flex;\r\n\t\t\tflex-direction: column;\r\n\t\t}\r\n\r\n\t\t.drawer.show {\r\n\t\t\ttransform: translateX(0);\r\n\t\t}\r\n\r\n\t\t.drawer-scroll {\r\n\t\t\theight: calc(100vh - 64px);\r\n\t\t\tpadding: 12px;\r\n\t\t\tbox-sizing: border-box;\r\n\t\t}\r\n\r\n\t\t.drawer-group {\r\n\t\t\tpadding: 10px 8px 0;\r\n\t\t}\r\n\r\n\t\t.drawer-date {\r\n\t\t\tcolor: #9aa3b2;\r\n\t\t\tfont-size: 12px;\r\n\t\t\tmargin-bottom: 8px;\r\n\t\t}\r\n\r\n\t\t.drawer-item {\r\n\t\t\tcolor: #333;\r\n\t\t\tfont-size: 13px;\r\n\t\t\tline-height: 20px;\r\n\t\t\tmargin: 6px 0;\r\n\t\t}\r\n\r\n\t\t.drawer-divider {\r\n\t\t\theight: 1px;\r\n\t\t\tbackground: #eeeeee;\r\n\t\t\tmargin: 12px 0;\r\n\t\t}\r\n\r\n\t\t.drawer-footer {\r\n\t\t\tpadding: 12px;\r\n\t\t\tborder-top: 1px solid #eeeeee;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t}\r\n\r\n\t\t.drawer-footer {\r\n\t\t\t/* fixed height for calc above */\r\n\t\t\theight: 64px;\r\n\t\t}\r\n\r\n\t\t.user-icon {\r\n\t\t\twidth: 24px;\r\n\t\t\ttext-align: center;\r\n\t\t}\r\n\r\n\t\t.user-name {\r\n\t\t\tflex: 1;\r\n\t\t\tfont-size: 14px;\r\n\t\t\tcolor: #333;\r\n\t\t}\r\n\r\n\t\t.footer-gear {\r\n\t\t\twidth: 24px;\r\n\t\t\ttext-align: center;\r\n\t\t}\r\n\r\n\t\t/* voice overlay */\r\n\t\t.record-mask {\r\n\t\t\tposition: fixed;\r\n\t\t\tleft: 0;\r\n\t\t\tright: 0;\r\n\t\t\ttop: 0;\r\n\t\t\tbottom: 0;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: center;\r\n\t\t\tz-index: 9999;\r\n\t\t}\r\n\r\n\t\t.record-box {\r\n\t\t\tbackground: rgba(0, 0, 0, .75);\r\n\t\t\tcolor: #fff;\r\n\t\t\tpadding: 16px 18px;\r\n\t\t\tborder-radius: 12px;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tgap: 10px;\r\n\t\t\tmin-width: 220rpx;\r\n\t\t}\r\n\r\n\t\t.record-box.cancel {\r\n\t\t\tbackground: rgba(221, 44, 0, .85);\r\n\t\t}\r\n\r\n\t\t.record-icon {\r\n\t\t\tfont-size: 20px;\r\n\t\t}\r\n\r\n\t\t.record-text {\r\n\t\t\tfont-size: 14px;\r\n\t\t}\r\n\r\n\t\t.text-voice {\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t}\r\n\r\n\t\t.voice-play {\r\n\t\t\twidth: 20px;\r\n\t\t\tmargin-left: 5px;\r\n\t\t}\r\n\t\t\r\n\t\t.mask-layer{\r\n\t\t\tposition: fixed;\r\n\t\t\tleft: 0;\r\n\t\t\tright: 0;\r\n\t\t\ttop: 0;\r\n\t\t\tbottom: 0;\r\n\t\t\tbackground-color: rgba(0, 0, 0, .1);\r\n\t\t}\r\n\t\t\r\n\t\t\r\n\t</style>"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///51\n"); +eval("/* WEBPACK VAR INJECTION */(function(__f__) {\n\nvar _interopRequireDefault = __webpack_require__(/*! @babel/runtime/helpers/interopRequireDefault */ 1);\nObject.defineProperty(exports, \"__esModule\", {\n value: true\n});\nexports.default = void 0;\nvar _regenerator = _interopRequireDefault(__webpack_require__(/*! @babel/runtime/regenerator */ 52));\nvar _asyncToGenerator2 = _interopRequireDefault(__webpack_require__(/*! @babel/runtime/helpers/asyncToGenerator */ 54));\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n//\n\nvar HISTORY_KEY = 'chat_history_groups';\nvar _default = {\n data: function data() {\n return {\n inputText: '',\n messages: [\n // {\n // \tid: 1,\n // \trole: 'user',\n // \ttype: 'text',\n // \tcontent: '帮我统计一下今日的销售数据',\n // \tinputType: 'text'\n // },\n // {\n // \tid: 2,\n // \trole: 'assistant',\n // \ttype: 'card',\n // \ttitle: '今日销售数据统计结果如下:',\n // \tcontent: '内容内容........................'\n // }\n ],\n scrollInto: '',\n drawerOpen: false,\n historyGroups: [],\n isRecording: false,\n isLoading: false,\n willCancel: false,\n recorder: null,\n recordStartY: 0,\n recordStartTs: 0,\n recordSimTimer: null,\n // show: false,\n innerAudioContext: null,\n popupVisible: false,\n typewriterTimers: {}\n };\n },\n computed: {\n timeOfDayText: function timeOfDayText() {\n var h = new Date().getHours();\n if (h < 6) return '凌晨好';\n if (h < 12) return '上午好';\n if (h < 18) return '下午好';\n return '晚上好';\n }\n },\n mounted: function mounted() {\n this.loadChatHistory();\n this.scrollToBottom();\n },\n beforeDestroy: function beforeDestroy() {\n // 清理所有打字机定时器\n Object.values(this.typewriterTimers).forEach(function (timer) {\n if (timer) clearInterval(timer);\n });\n this.typewriterTimers = {};\n // 清理录音定时器\n if (this.recordSimTimer) {\n clearTimeout(this.recordSimTimer);\n }\n },\n methods: {\n // 新增方法:上传音频并识别\n recognizeAudio: function recognizeAudio(tempFilePath) {\n return (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee() {\n var fileInfo, uploadRes, result;\n return _regenerator.default.wrap(function _callee$(_context) {\n while (1) {\n switch (_context.prev = _context.next) {\n case 0:\n _context.prev = 0;\n __f__(\"log\", '开始语音识别,文件路径:', tempFilePath, \" at pages/index/index.vue:208\");\n\n // 获取文件信息\n _context.next = 4;\n return new Promise(function (resolve, reject) {\n uni.getFileInfo({\n filePath: tempFilePath,\n success: resolve,\n fail: reject\n });\n });\n case 4:\n fileInfo = _context.sent;\n __f__(\"log\", '文件大小:', fileInfo.size, \" at pages/index/index.vue:219\");\n\n // 使用 UniApp 的上传文件 API\n _context.next = 8;\n return new Promise(function (resolve, reject) {\n uni.uploadFile({\n // url: 'http://192.168.133.83:8000/recognize_speech',\n // url: 'http://192.168.10.44:8000/recognize_speech',\n // url: 'http://192.168.1.18:8000/recognize_speech',\n url: 'http://106.227.91.181:8000/recognize_speech',\n filePath: tempFilePath,\n name: 'speech',\n // 对应后端的 UploadFile 参数名\n formData: {\n 'format': 'amr',\n 'rate': 16000,\n 'channel': 1,\n 'cuid': 'uniapp_user',\n 'audio_len': fileInfo.size\n },\n success: function success(res) {\n __f__(\"log\", '上传响应:', res, \" at pages/index/index.vue:238\");\n if (res.statusCode === 200) {\n try {\n // 尝试解析返回的 JSON 数据\n var data = JSON.parse(res.data);\n resolve({\n statusCode: 200,\n data: data\n });\n } catch (e) {\n reject(new Error('响应解析失败: ' + e.message));\n }\n } else {\n reject(new Error(\"\\u4E0A\\u4F20\\u5931\\u8D25: \".concat(res.statusCode)));\n }\n },\n fail: function fail(err) {\n reject(new Error('上传请求失败: ' + err.errMsg));\n }\n });\n });\n case 8:\n uploadRes = _context.sent;\n __f__(\"log\", '语音识别响应:', uploadRes, \" at pages/index/index.vue:257\");\n result = uploadRes.data;\n if (!(result.status === 'success')) {\n _context.next = 15;\n break;\n }\n return _context.abrupt(\"return\", result.result);\n case 15:\n throw new Error(result.error || '识别失败');\n case 16:\n _context.next = 23;\n break;\n case 18:\n _context.prev = 18;\n _context.t0 = _context[\"catch\"](0);\n __f__(\"error\", '语音识别错误:', _context.t0, \" at pages/index/index.vue:266\");\n uni.showToast({\n title: '识别失败: ' + (_context.t0.message || '网络错误'),\n icon: 'none'\n });\n return _context.abrupt(\"return\", null);\n case 23:\n case \"end\":\n return _context.stop();\n }\n }\n }, _callee, null, [[0, 18]]);\n }))();\n },\n // ==================== 历史记录管理 ====================\n formatDate: function formatDate(date) {\n var y = date.getFullYear();\n var m = String(date.getMonth() + 1).padStart(2, '0');\n var d = String(date.getDate()).padStart(2, '0');\n return \"\".concat(y, \"\\u5E74\").concat(m, \"\\u6708\").concat(d, \"\\u65E5\");\n },\n loadChatHistory: function loadChatHistory() {\n try {\n var data = uni.getStorageSync(HISTORY_KEY);\n if (data && Array.isArray(data.groups)) {\n this.historyGroups = data.groups;\n } else {\n this.historyGroups = [];\n }\n } catch (e) {\n this.historyGroups = [];\n }\n },\n addToHistory: function addToHistory(text) {\n var _uni$getStorageSync;\n var groups = ((_uni$getStorageSync = uni.getStorageSync(HISTORY_KEY)) === null || _uni$getStorageSync === void 0 ? void 0 : _uni$getStorageSync.groups) || [];\n var today = this.formatDate(new Date());\n var todayGroup = groups.find(function (g) {\n return g.date === today;\n });\n if (!todayGroup) {\n todayGroup = {\n date: today,\n items: []\n };\n groups.unshift(todayGroup);\n }\n if (!todayGroup.items.includes(text)) {\n todayGroup.items.unshift(text);\n }\n\n // 限制大小\n if (todayGroup.items.length > 50) todayGroup.items = todayGroup.items.slice(0, 50);\n if (groups.length > 30) groups = groups.slice(0, 30);\n this.historyGroups = groups;\n uni.setStorageSync(HISTORY_KEY, {\n groups: groups,\n updatedAt: Date.now()\n });\n },\n removeFromHistory: function removeFromHistory(text) {\n var _uni$getStorageSync2;\n var groups = ((_uni$getStorageSync2 = uni.getStorageSync(HISTORY_KEY)) === null || _uni$getStorageSync2 === void 0 ? void 0 : _uni$getStorageSync2.groups) || [];\n groups.forEach(function (group) {\n group.items = group.items.filter(function (item) {\n return item !== text;\n });\n });\n groups = groups.filter(function (g) {\n return g.items.length > 0;\n });\n this.historyGroups = groups;\n uni.setStorageSync(HISTORY_KEY, {\n groups: groups,\n updatedAt: Date.now()\n });\n },\n clearAllHistory: function clearAllHistory() {\n var _this = this;\n uni.showModal({\n title: '清除全部',\n content: '将删除所有对话记录,此操作不可恢复',\n success: function success(res) {\n if (res.confirm) {\n uni.removeStorageSync(HISTORY_KEY);\n _this.historyGroups = [];\n uni.showToast({\n title: '已清除',\n icon: 'success'\n });\n }\n }\n });\n },\n onLongPressHistory: function onLongPressHistory(text) {\n var _this2 = this;\n uni.showModal({\n title: '删除记录',\n content: '确定删除这条对话记录?',\n success: function success(res) {\n if (res.confirm) {\n _this2.removeFromHistory(text);\n }\n }\n });\n },\n // 工具\n removeMessage: function removeMessage(id) {\n var idx = this.messages.findIndex(function (m) {\n return m.id === id;\n });\n if (idx > -1) this.messages.splice(idx, 1);\n },\n addAssistantMessage: function addAssistantMessage(id, content) {\n this.messages.push({\n id: id,\n role: 'assistant',\n type: 'text',\n content: content,\n displayText: ''\n });\n },\n getAIResponse: function getAIResponse(message) {\n return (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee2() {\n var _res$data, _res$data$result, url, headers, data, res;\n return _regenerator.default.wrap(function _callee2$(_context2) {\n while (1) {\n switch (_context2.prev = _context2.next) {\n case 0:\n _context2.prev = 0;\n // const url = 'http://192.168.133.83:9020/api/chat'\n // const url = 'http://192.168.10.44:9020/api/chat'\n url = 'http://106.227.91.181:9020/api/chat'; // 如需切换线上,改这里即可\n headers = {\n 'Content-Type': 'application/json'\n };\n data = {\n message: message\n }; // console.log(data)\n // const [error, res] = await uni.request({\n // url,\n // method: 'POST',\n // header: headers,\n // data\n // })\n // console.log(res)\n // 使用 Promise 风格\n _context2.next = 6;\n return new Promise(function (resolve, reject) {\n uni.request({\n url: url,\n method: 'POST',\n header: headers,\n data: data,\n success: function success(res) {\n return resolve(res);\n },\n fail: function fail(err) {\n return reject(err);\n }\n });\n });\n case 6:\n res = _context2.sent;\n __f__(\"log\", '请求响应:', res, \" at pages/index/index.vue:388\");\n if (!(res.statusCode !== 200)) {\n _context2.next = 10;\n break;\n }\n throw new Error(\"HTTP\\u9519\\u8BEF! \\u72B6\\u6001\\u7801: \".concat(res.statusCode));\n case 10:\n return _context2.abrupt(\"return\", ((_res$data = res.data) === null || _res$data === void 0 ? void 0 : (_res$data$result = _res$data.result) === null || _res$data$result === void 0 ? void 0 : _res$data$result.data) || '未获取到有效回复');\n case 13:\n _context2.prev = 13;\n _context2.t0 = _context2[\"catch\"](0);\n __f__(\"error\", 'AI请求错误:', _context2.t0, \" at pages/index/index.vue:397\");\n return _context2.abrupt(\"return\", \"\\u62B1\\u6B49\\uFF0C\\u51FA\\u4E86\\u70B9\\u95EE\\u9898: \".concat(_context2.t0.errMsg || _context2.t0.message));\n case 17:\n case \"end\":\n return _context2.stop();\n }\n }\n }, _callee2, null, [[0, 13]]);\n }))();\n },\n playVoice: function playVoice(voicePath) {\n if (!voicePath) {\n uni.showToast({\n title: '无可播放的语音',\n icon: 'none'\n });\n return;\n }\n if (!this.innerAudioContext) {\n this.innerAudioContext = uni.createInnerAudioContext();\n this.innerAudioContext.autoplay = false;\n this.innerAudioContext.onError(function () {\n uni.showToast({\n title: '播放失败',\n icon: 'none'\n });\n });\n }\n try {\n this.innerAudioContext.stop();\n } catch (e) {}\n __f__(\"log\", voicePath, \" at pages/index/index.vue:415\");\n this.innerAudioContext.src = voicePath;\n this.innerAudioContext.play();\n },\n onSettingTap: function onSettingTap() {\n uni.navigateTo({\n url: '/pages/setting/index'\n });\n },\n onSuggestionTap: function onSuggestionTap(text) {\n this.inputText = text;\n this.onSend();\n },\n onQuickAsk: function onQuickAsk(text) {\n this.inputText = text;\n this.onSend();\n },\n onSwitchModel: function onSwitchModel() {\n uni.showToast({\n title: '已切换为通用模型',\n icon: 'none'\n });\n },\n onInput: function onInput(e) {\n this.inputText = e.detail.value;\n },\n openDrawer: function openDrawer() {\n this.$refs.popup.open();\n },\n onPopupChange: function onPopupChange(e) {\n // e.show: true when opened, false when closed\n this.popupVisible = !!(e && e.show === true);\n },\n // ===== Voice input (WeChat-like) =====\n ensureRecorder: function ensureRecorder() {\n var _this3 = this;\n if (this.recorder) return;\n try {\n this.recorder = uni.getRecorderManager && uni.getRecorderManager();\n } catch (e) {\n this.recorder = null;\n }\n if (this.recorder) {\n this.recorder.onStart();\n this.recorder.onStop( /*#__PURE__*/function () {\n var _ref = (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee3(res) {\n var duration, tempFilePath, recognizedText;\n return _regenerator.default.wrap(function _callee3$(_context3) {\n while (1) {\n switch (_context3.prev = _context3.next) {\n case 0:\n duration = Date.now() - _this3.recordStartTs;\n tempFilePath = res.tempFilePath; // 添加这行,从res中获取文件路径\n if (!(_this3.willCancel || duration < 700)) {\n _context3.next = 5;\n break;\n }\n uni.showToast({\n title: duration < 700 ? '说话时间太短' : '已取消',\n icon: 'none'\n });\n return _context3.abrupt(\"return\");\n case 5:\n // 显示加载\n uni.showLoading({\n title: '识别中...'\n });\n\n // TODO: 上传 res.tempFilePath 做识别;现用 mock\n // this.handleRecognizedText(this.mockSpeechToText(duration), res.tempFilePath, Math.ceil(\n // \tduration / 100) / 10)\n // 真实识别\n _context3.next = 8;\n return _this3.recognizeAudio(tempFilePath);\n case 8:\n recognizedText = _context3.sent;\n uni.hideLoading();\n if (recognizedText) {\n // 成功:填入输入框\n _this3.inputText = recognizedText;\n _this3.$nextTick(function () {\n // 可选:自动发送\n // this.onSend('voice', tempFilePath, Math.ceil(duration / 100) / 10);\n });\n }\n case 11:\n case \"end\":\n return _context3.stop();\n }\n }\n }, _callee3);\n }));\n return function (_x) {\n return _ref.apply(this, arguments);\n };\n }());\n }\n },\n onPressMic: function onPressMic(e) {\n this.ensureRecorder();\n this.isRecording = true;\n // this.show = true\n this.willCancel = false;\n this.recordStartTs = Date.now();\n this.recordStartY = e.changedTouches && e.changedTouches[0] ? e.changedTouches[0].clientY : 0;\n if (this.recorder) {\n try {\n this.recorder.start({\n format: 'amr',\n sampleRate: 16000,\n encodeBitRate: 16000,\n // 编码比特率\n frameSize: 4,\n // 帧大小\n numberOfChannels: 1,\n duration: 60000\n });\n } catch (err) {}\n } else {\n if (this.recordSimTimer) clearTimeout(this.recordSimTimer);\n this.recordSimTimer = setTimeout(function () {}, 60000);\n }\n },\n onMoveMic: function onMoveMic(e) {\n if (!this.isRecording) return;\n var y = e.changedTouches && e.changedTouches[0] ? e.changedTouches[0].clientY : 0;\n this.willCancel = this.recordStartY - y > 60;\n },\n onReleaseMic: function onReleaseMic() {\n __f__(\"log\", 'onReleaseMic', \" at pages/index/index.vue:517\");\n if (!this.isRecording) return;\n var cancel = this.willCancel;\n this.isRecording = false;\n this.show = false;\n if (this.recorder) {\n try {\n this.recorder.stop();\n } catch (err) {\n __f__(\"log\", 'err', err, \" at pages/index/index.vue:526\");\n }\n }\n },\n handleRecognizedText: function handleRecognizedText(text, tempFilePath, duration) {\n if (!text) return;\n this.inputText = text;\n this.onSend('voice', tempFilePath, duration); // 传 'voice'\n },\n mockSpeechToText: function mockSpeechToText(ms) {\n var sec = Math.ceil(ms / 100) / 10;\n var pool = [\"\\u8BED\\u97F3\\u8F93\\u5165 \".concat(sec, \"s\\uFF0C\\u6A21\\u62DF\\u8BC6\\u522B\\uFF1A\\u5E2E\\u6211\\u7EDF\\u8BA1\\u4ECA\\u5929\\u9500\\u552E\\u989D\"), \"\\u8BED\\u97F3\\u8F93\\u5165 \".concat(sec, \"s\\uFF0C\\u6A21\\u62DF\\u8BC6\\u522B\\uFF1A\\u67E5\\u8BE2\\u8BA2\\u535520388993483\"), \"\\u8BED\\u97F3\\u8F93\\u5165 \".concat(sec, \"s\\uFF0C\\u6A21\\u62DF\\u8BC6\\u522B\\uFF1A\\u751F\\u6210\\u65E5\\u62A5\")];\n return pool[Math.floor(Math.random() * pool.length)];\n },\n onHistoryItemTap: function onHistoryItemTap(text) {\n this.inputText = text;\n this.onSend();\n this.$refs.popup.close();\n },\n onSend: function onSend() {\n var _arguments = arguments,\n _this4 = this;\n return (0, _asyncToGenerator2.default)( /*#__PURE__*/_regenerator.default.mark(function _callee4() {\n var inputType, inputContent, duration, text, baseId, loadingId, reply, loadingIdx, replyId, _loadingIdx;\n return _regenerator.default.wrap(function _callee4$(_context4) {\n while (1) {\n switch (_context4.prev = _context4.next) {\n case 0:\n inputType = _arguments.length > 0 && _arguments[0] !== undefined ? _arguments[0] : 'text';\n inputContent = _arguments.length > 1 && _arguments[1] !== undefined ? _arguments[1] : '';\n duration = _arguments.length > 2 && _arguments[2] !== undefined ? _arguments[2] : undefined;\n text = (_this4.inputText || '').trim();\n if (!(!text || _this4.isLoading)) {\n _context4.next = 6;\n break;\n }\n return _context4.abrupt(\"return\");\n case 6:\n baseId = Date.now(); // 1. 用户消息\n _this4.messages.push({\n id: baseId,\n role: 'user',\n type: 'text',\n content: text,\n inputType: inputType,\n inputContent: inputContent,\n duration: duration\n });\n\n // 2. loading 消息\n loadingId = baseId + 0.5;\n _this4.messages.push({\n id: loadingId,\n role: 'assistant',\n loading: true\n });\n _this4.scrollToBottom();\n _this4.inputText = '';\n _this4.isLoading = true;\n _this4.addToHistory(text);\n _context4.prev = 14;\n _context4.next = 17;\n return _this4.getAIResponse(text);\n case 17:\n reply = _context4.sent;\n // 4. 移除 loading\n loadingIdx = _this4.messages.findIndex(function (m) {\n return m.id === loadingId;\n });\n if (loadingIdx > -1) _this4.messages.splice(loadingIdx, 1);\n\n // 5. 添加回复 + 打字机\n replyId = baseId + 1;\n _this4.messages.push({\n id: replyId,\n role: 'assistant',\n type: 'text',\n content: reply,\n displayText: ''\n });\n _this4.typewriter(replyId, reply);\n _context4.next = 30;\n break;\n case 25:\n _context4.prev = 25;\n _context4.t0 = _context4[\"catch\"](14);\n // 出错时也展示\n _loadingIdx = _this4.messages.findIndex(function (m) {\n return m.id === loadingId;\n });\n if (_loadingIdx > -1) _this4.messages.splice(_loadingIdx, 1);\n _this4.messages.push({\n id: baseId + 1,\n role: 'assistant',\n content: \"\\u8BF7\\u6C42\\u51FA\\u9519\\uFF1A\".concat(_context4.t0.message || _context4.t0)\n });\n case 30:\n _context4.prev = 30;\n _this4.isLoading = false;\n _this4.$nextTick(function () {\n return _this4.scrollToBottom();\n });\n return _context4.finish(30);\n case 34:\n case \"end\":\n return _context4.stop();\n }\n }\n }, _callee4, null, [[14, 25, 30, 34]]);\n }))();\n },\n typewriter: function typewriter(messageId, fullText) {\n var _this5 = this;\n var msg = this.messages.find(function (m) {\n return m.id === messageId;\n });\n if (!msg) return;\n // 清理之前的定时器(如果存在)\n if (this.typewriterTimers[messageId]) {\n clearInterval(this.typewriterTimers[messageId]);\n }\n var index = 0;\n msg.displayText = fullText.substring(0, index + 1);\n index += 1;\n var speed = 50; // 每个字符间隔50ms\n var timer = setInterval(function () {\n if (index < fullText.length) {\n msg.displayText = fullText.substring(0, index + 1);\n index++;\n // 打字过程中自动滚动到底部\n _this5.$nextTick(function () {\n _this5.scrollToBottom();\n });\n } else {\n clearInterval(timer);\n delete _this5.typewriterTimers[messageId];\n // 完成后使用完整文本\n msg.displayText = fullText;\n }\n }, speed);\n this.typewriterTimers[messageId] = timer;\n },\n scrollToBottom: function scrollToBottom() {\n var self = this;\n this.$nextTick(function () {\n uni.createSelectorQuery().select('.content').boundingClientRect(function (rect) {\n if (self.height !== rect.height) {\n self.height = rect.height;\n uni.pageScrollTo({\n scrollTop: rect.height,\n duration: 300,\n class: '.content'\n });\n }\n }).exec();\n });\n },\n mockReply: function mockReply(text) {\n var candidates = ['好的,我已经为您处理。', '收到请求,以下是结果的概览。', '我理解了,这是一个示例回复。', '已记录,稍后将完善报表。'];\n var pick = candidates[Math.floor(Math.random() * candidates.length)];\n return pick + '(已收到:“' + text + '”)';\n },\n onListen: function onListen(text) {\n try {\n // H5: Web Speech API\n if (typeof window !== 'undefined' && window.speechSynthesis) {\n var u = new SpeechSynthesisUtterance(String(text));\n u.lang = 'zh-CN';\n u.rate = 1;\n u.pitch = 1;\n window.speechSynthesis.cancel();\n window.speechSynthesis.speak(u);\n return;\n }\n } catch (e) {}\n uni.showToast({\n title: '当前端不支持语音播放',\n icon: 'none'\n });\n }\n }\n};\nexports.default = _default;\n/* WEBPACK VAR INJECTION */}.call(this, __webpack_require__(/*! ./node_modules/@dcloudio/vue-cli-plugin-uni/lib/format-log.js */ 35)[\"default\"]))//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["uni-app:///pages/index/index.vue"],"names":["data","inputText","messages","scrollInto","drawerOpen","historyGroups","isRecording","isLoading","willCancel","recorder","recordStartY","recordStartTs","recordSimTimer","innerAudioContext","popupVisible","typewriterTimers","computed","timeOfDayText","mounted","beforeDestroy","Object","clearTimeout","methods","recognizeAudio","uni","filePath","success","fail","fileInfo","url","name","formData","resolve","statusCode","reject","uploadRes","result","title","icon","formatDate","loadChatHistory","addToHistory","todayGroup","date","items","groups","updatedAt","removeFromHistory","group","clearAllHistory","content","onLongPressHistory","removeMessage","addAssistantMessage","id","role","type","displayText","getAIResponse","headers","message","method","header","res","playVoice","onSettingTap","onSuggestionTap","onQuickAsk","onSwitchModel","onInput","openDrawer","onPopupChange","ensureRecorder","duration","tempFilePath","recognizedText","onPressMic","format","sampleRate","encodeBitRate","frameSize","numberOfChannels","onMoveMic","onReleaseMic","handleRecognizedText","mockSpeechToText","sec","onHistoryItemTap","onSend","inputType","inputContent","text","baseId","loadingId","loading","reply","loadingIdx","replyId","typewriter","clearInterval","msg","index","scrollToBottom","self","scrollTop","class","mockReply","onListen","u","window"],"mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AA6IA;AAAA,eACA;EACAA;IACA;MACAC;MACAC;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;QACA;MAAA,CACA;MACAC;MACAC;MACAC,iBACA;MACAC;MACAC;MACAC;MACAC;MACAC;MACAC;MACAC;MACA;MACAC;MACAC;MACAC;IACA;EACA;EACAC;IACAC;MACA;MACA;MACA;MACA;MACA;IACA;EACA;EACAC;IACA;IACA;EACA;EACAC;IACA;IACAC;MACA;IACA;IACA;IACA;IACA;MACAC;IACA;EACA;EACAC;IACA;IACAC;MAAA;QAAA;QAAA;UAAA;YAAA;cAAA;gBAAA;gBAEA;;gBAEA;gBAAA;gBAAA,OACA;kBACAC;oBACAC;oBACAC;oBACAC;kBACA;gBACA;cAAA;gBANAC;gBAQA;;gBAEA;gBAAA;gBAAA,OACA;kBACAJ;oBACA;oBACA;oBACA;oBACAK;oBACAJ;oBACAK;oBAAA;oBACAC;sBACA;sBACA;sBACA;sBACA;sBACA;oBACA;oBACAL;sBACA;sBACA;wBACA;0BACA;0BACA;0BACAM;4BAAAC;4BAAAjC;0BAAA;wBACA;0BACAkC;wBACA;sBACA;wBACAA;sBACA;oBACA;oBACAP;sBACAO;oBACA;kBACA;gBACA;cAAA;gBAjCAC;gBAmCA;gBAEAC;gBAAA,MACAA;kBAAA;kBAAA;gBAAA;gBAAA,iCACAA;cAAA;gBAAA,MAEA;cAAA;gBAAA;gBAAA;cAAA;gBAAA;gBAAA;gBAGA;gBACAZ;kBACAa;kBACAC;gBACA;gBAAA,iCACA;cAAA;cAAA;gBAAA;YAAA;UAAA;QAAA;MAAA;IAEA;IACA;IACAC;MACA;MACA;MACA;MACA;IACA;IACAC;MACA;QACA;QACA;UACA;QACA;UACA;QACA;MACA;QACA;MACA;IACA;IACAC;MAAA;MACA;MACA;MACA;QAAA;MAAA;MAEA;QACAC;UAAAC;UAAAC;QAAA;QACAC;MACA;MAEA;QACAH;MACA;;MAEA;MACA;MACA;MAEA;MACAlB;QAAAqB;QAAAC;MAAA;IACA;IAEAC;MAAA;MACA;MACAF;QACAG;UAAA;QAAA;MACA;MACAH;QAAA;MAAA;MACA;MACArB;QAAAqB;QAAAC;MAAA;IACA;IAEAG;MAAA;MACAzB;QACAa;QACAa;QACAxB;UACA;YACAF;YACA;YACAA;cAAAa;cAAAC;YAAA;UACA;QACA;MACA;IACA;IACAa;MAAA;MACA3B;QACAa;QACAa;QACAxB;UACA;YACA;UACA;QACA;MACA;IACA;IACA;IACA0B;MACA;QAAA;MAAA;MACA;IACA;IACAC;MACA;QAAAC;QAAAC;QAAAC;QAAAN;QAAAO;MAAA;IACA;IAEAC;MAAA;QAAA;QAAA;UAAA;YAAA;cAAA;gBAAA;gBAEA;gBACA;gBACA7B;gBACA8B;kBAAA;gBAAA;gBACA3D;kBAAA4D;gBAAA,GAEA;gBAEA;gBACA;gBACA;gBACA;gBACA;gBACA;gBAEA;gBACA;gBAAA;gBAAA,OACA;kBACApC;oBACAK;oBACAgC;oBACAC;oBACA9D;oBACA0B;sBAAA;oBAAA;oBACAC;sBAAA;oBAAA;kBACA;gBACA;cAAA;gBATAoC;gBAWA;gBAAA,MAEAA;kBAAA;kBAAA;gBAAA;gBAAA,MACA;cAAA;gBAAA,kCAGA;cAAA;gBAAA;gBAAA;gBAGA;gBAAA,8FACA;cAAA;cAAA;gBAAA;YAAA;UAAA;QAAA;MAAA;IAEA;IAEAC;MACA;QACAxC;UAAAa;UAAAC;QAAA;QACA;MACA;MACA;QACA;QACA;QACA;UACAd;YAAAa;YAAAC;UAAA;QACA;MACA;MACA;QAAA;MAAA;MACA;MACA;MACA;IACA;IACA2B;MACAzC;QACAK;MACA;IACA;IACAqC;MACA;MACA;IACA;IACAC;MACA;MACA;IACA;IACAC;MACA5C;QACAa;QACAC;MACA;IACA;IACA+B;MACA;IACA;IACAC;MACA;IACA;IACAC;MACA;MACA;IACA;IACA;IACAC;MAAA;MACA;MACA;QACA;MACA;QACA;MACA;MACA;QACA;QACA;UAAA;YAAA;YAAA;cAAA;gBAAA;kBAAA;oBACAC;oBACAC;oBAAA,MACA;sBAAA;sBAAA;oBAAA;oBACAlD;sBACAa;sBACAC;oBACA;oBAAA;kBAAA;oBAGA;oBACAd;sBAAAa;oBAAA;;oBAEA;oBACA;oBACA;oBACA;oBAAA;oBAAA,OACA;kBAAA;oBAAAsC;oBACAnD;oBACA;sBACA;sBACA;sBACA;wBACA;wBACA;sBAAA,CACA;oBACA;kBAAA;kBAAA;oBAAA;gBAAA;cAAA;YAAA;UAAA,CACA;UAAA;YAAA;UAAA;QAAA;MACA;IACA;IACAoD;MACA;MACA;MACA;MACA;MACA;MACA;MACA;QACA;UACA;YACAC;YACAC;YACAC;YAAA;YACAC;YAAA;YACAC;YACAR;UACA;QACA;MACA;QACA;QACA;MACA;IACA;IACAS;MACA;MACA;MACA;IACA;IACAC;MACA;MACA;MACA;MACA;MACA;MACA;QACA;UACA;QACA;UACA;QACA;MACA;IACA;IACAC;MACA;MACA;MACA;IACA;IACAC;MACA;MACA,+CACAC,wIACAA,qHACAA,sEACA;MACA;IACA;IACAC;MACA;MACA;MACA;IACA;IACAC;MAAA;QAAA;MAAA;QAAA;QAAA;UAAA;YAAA;cAAA;gBAAAC;gBAAAC;gBAAAjB;gBACAkB;gBAAA,MACA;kBAAA;kBAAA;gBAAA;gBAAA;cAAA;gBAEAC,qBAEA;gBACA;kBACAtC;kBACAC;kBACAC;kBACAN;kBACAuC;kBACAC;kBACAjB;gBACA;;gBAEA;gBACAoB;gBACA;kBACAvC;kBACAC;kBACAuC;gBACA;gBAEA;gBACA;gBACA;gBAEA;gBAAA;gBAAA;gBAAA,OAIA;cAAA;gBAAAC;gBAEA;gBACAC;kBAAA;gBAAA;gBACA;;gBAEA;gBACAC;gBACA;kBACA3C;kBACAC;kBACAC;kBACAN;kBACAO;gBACA;gBAEA;gBAAA;gBAAA;cAAA;gBAAA;gBAAA;gBAEA;gBACAuC;kBAAA;gBAAA;gBACA;gBACA;kBACA1C;kBACAC;kBACAL;gBACA;cAAA;gBAAA;gBAEA;gBACA;kBAAA;gBAAA;gBAAA;cAAA;cAAA;gBAAA;YAAA;UAAA;QAAA;MAAA;IAEA;IACAgD;MAAA;MACA;QAAA;MAAA;MACA;MACA;MACA;QACAC;MACA;MACA;MACAC;MACAC;MACA;MACA;QACA;UACAD;UACAC;UACA;UACA;YACA;UACA;QACA;UACAF;UACA;UACA;UACAC;QACA;MACA;MACA;IACA;IACAE;MACA;MACA;QACA9E;UACA;YACA+E;YACA/E;cACAgF;cACA/B;cACAgC;YACA;UACA;QACA;MACA;IACA;IACAC;MACA,kBACA,eACA,kBACA,kBACA,eACA;MACA;MACA;IACA;IACAC;MACA;QACA;QACA;UACA;UACAC;UACAA;UACAA;UACAC;UACAA;UACA;QACA;MACA;MACArF;QACAa;QACAC;MACA;IACA;EACA;AACA;AAAA,2B","file":"51.js","sourcesContent":["\t<template>\r\n\t\t<view class=\"ai-page\">\r\n\t\t\t<uni-nav-bar  left-icon=\"left\" @clickLeft=\"openDrawer\" title=\"AI对话\">\r\n\t\t\t\t<template v-slot:left>\r\n\t\t\t\t\t<view class=\"hamburger\">\r\n\t\t\t\t\t\t<view class=\"line\" />\r\n\t\t\t\t\t\t<view class=\"line\" />\r\n\t\t\t\t\t\t<view class=\"line\" />\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</template>\r\n\t\t\t\t<template v-slot:right>\r\n\t\t\t\t\t<view class=\"nav-right\">\r\n\t\t\t\t\t\t<!-- <view class=\"gear\" @tap=\"onSettingTap\">⚙️</view> -->\r\n\t\t\t\t\t\t<image src=\"../../static/set.png\" mode=\"widthFix\" @tap=\"onSettingTap\" style=\"width: 18px;\"></image>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</template>\r\n\t\t\t</uni-nav-bar>\r\n\t\t\t<!-- scrollable content -->\r\n\t\t\t<scroll-view class=\"content\" :scroll-y=\"true\" show-scrollbar=\"false\" \r\n\t\t\t\tscroll-with-animation ref=\"scrollView\">\r\n\t\t\t\t<!-- greeting card -->\t\r\n\t\t\t\t<view class=\"greet-card\">\r\n\t\t\t\t\t\t<image src=\"../../static/ai.webp\" mode=\"widthFix\" style=\"width: 60px;margin-right: 10px;\"></image>\r\n\t\t\t\t\t<view class=\"greet-text\">\r\n\t\t\t\t\t\t<view class=\"hi\">HI，{{ timeOfDayText }}</view>\r\n\t\t\t\t\t\t<view class=\"sub\">我是萃星科技智能体</view>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</view>\r\n\r\n\t\t\t\t<!-- welcome sentence -->\r\n\t\t\t\t<view class=\"welcome\">\r\n\t\t\t\t\t您好！非常高兴与您交流，今天有什么可以帮到您？\r\n\t\t\t\t</view>\r\n\r\n\t\t\t\t<!-- suggestions -->\r\n\t\t\t\t<view class=\"guess-panel\">\r\n\t\t\t\t\t<view class=\"guess-title\">猜你想问</view>\r\n\t\t\t\t\t<view class=\"guess-list\">\r\n\t\t\t\t\t\t<view class=\"guess-item\" @tap=\"onSuggestionTap('今日出入库数据')\">\r\n\t\t\t\t\t\t\t<text>今日出入库数据</text>\r\n\t\t\t\t\t\t\t<text class=\"arrow\">›</text>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t<view class=\"guess-item\" @tap=\"onSuggestionTap('今日销售数据')\">\r\n\t\t\t\t\t\t\t<text>今日销售数据</text>\r\n\t\t\t\t\t\t\t<text class=\"arrow\">›</text>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t<view class=\"guess-item\" @tap=\"onSuggestionTap('今日生产数据')\">\r\n\t\t\t\t\t\t\t<text>今日生产数据</text>\r\n\t\t\t\t\t\t\t<text class=\"arrow\">›</text>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</view>\r\n\r\n\t\t\t\t<!-- conversation -->\r\n\t\t\t\t<view class=\"chat\">\r\n\t\t\t\t\t<view v-for=\"m in messages\" :key=\"m.id\" :id=\"'msg-' + m.id\" :class=\"['msg', m.role]\">\r\n\t\t\t\t\t\t<view v-if=\"m.role === 'user'\" class=\"bubble user-bubble\">\r\n\t\t\t\t\t\t\t<text v-if=\"m.inputType === 'text'\">{{ m.content }}</text>\r\n\t\t\t\t\t\t\t<view class=\"text-voice\" v-else @tap=\"playVoice(m.inputContent,m.id)\">\r\n\t\t\t\t\t\t\t\t<text>{{m.duration }}</text>\r\n\t\t\t\t\t\t\t\t<image class=\"voice-play\" src=\"../../static/voice-play.png\" mode=\"widthFix\"></image>\r\n\t\t\t\t\t\t\t</view>\r\n\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t<view v-else class=\"bubble ai-bubble\">\r\n\t\t\t\t\t\t\t<view v-if=\"m.type === 'card'\" class=\"ai-card\">\r\n\t\t\t\t\t\t\t\t<view class=\"ai-card-title\">{{ m.title }}</view>\r\n\t\t\t\t\t\t\t\t<view class=\"ai-card-body\">{{ m.content }}</view>\r\n\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t\t<view v-else-if=\"m.loading\" class=\"ai-loading\">\r\n\t\t\t\t\t\t\t\t<view class=\"loading-dot\"></view>\r\n\t\t\t\t\t\t\t\t<view class=\"loading-dot\"></view>\r\n\t\t\t\t\t\t\t\t<view class=\"loading-dot\"></view>\r\n\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t\t<view v-else>\r\n\t\t\t\t\t\t\t\t<text>{{ m.displayText !== undefined ? m.displayText : m.content }}</text>\r\n\t\t\t\t\t\t\t\t<!-- <text class=\"listen-btn\" @tap=\"onListen(m.content)\">🔊</text> -->\r\n\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</view>\r\n\r\n\t\t\t\t<view style=\"height: 12px;\" />\r\n\t\t\t</scroll-view>\r\n\r\n\t\t\t<!-- bottom dock: quick actions + input bar -->\r\n\t\t\t<view class=\"dock\">\r\n\t\t\t\t<scroll-view class=\"quick-actions horizontal\" scroll-x show-scrollbar=\"false\">\r\n\t\t\t\t\t<view class=\"qa-btn minor\" @tap=\"onSwitchModel\">切换模型</view>\r\n\t\t\t\t\t<view class=\"qa-btn\" @tap=\"onQuickAsk('你是谁？')\">自我介绍</view>\r\n\t\t\t\t\t<view class=\"qa-btn\" @tap=\"onQuickAsk('今日任务有哪些？')\">快捷提问</view>\r\n\t\t\t\t\t<view class=\"qa-btn\" @tap=\"onQuickAsk('展示一份报表示例')\">快捷提问</view>\r\n\t\t\t\t\t<view class=\"qa-btn\" @tap=\"onQuickAsk('生成日报模版')\">快捷提问</view>\r\n\t\t\t\t</scroll-view>\r\n\t\t\t\t<view class=\"input-bar\">\r\n\t\t\t\t\t<input class=\"input\" confirm-type=\"send\" :value=\"inputText\" @input=\"onInput\" @confirm=\"onSend()\"\r\n\t\t\t\t\t\tplaceholder=\"你可以说…\" placeholder-class=\"ph\" />\r\n\t\t\t\t\t<view :class=\"['mic', { recording: isRecording }]\" @touchstart.stop=\"onPressMic\"\r\n\t\t\t\t\t\t@touchmove.stop=\"onMoveMic\" @touchend.stop=\"onReleaseMic\">🎙️</view>\r\n\t\t\t\t\t<button class=\"send\" type=\"primary\" @tap=\"onSend\">发送</button>\r\n\t\t\t\t</view>\r\n\t\t\t</view>\r\n\r\n\t\t\t<!-- left drawer -->、\r\n\t\t\t<uni-popup ref=\"popup\" background-color=\"#fff\" type=\"left\" :z-index=\"10090\" @change=\"onPopupChange\" style=\"z-index: 99999;width: 100vw\" >\r\n\t\t\t\t<view class=\"drawer-mask\">\r\n\t\t\t\t\t<view class=\"drawer\">\r\n\t\t\t\t\t\t<scroll-view class=\"drawer-scroll\" scroll-y show-scrollbar=\"false\">\r\n\t\t\t\t\t\t\t<view v-for=\"g in historyGroups\" :key=\"g.date\" class=\"drawer-group\">\r\n\t\t\t\t\t\t\t\t<view class=\"drawer-date\">{{ g.date }}</view>\r\n\t\t\t\t\t\t\t\t<view v-for=\"(t, idx) in g.items\" :key=\"idx\" class=\"drawer-item\" @tap=\"onHistoryItemTap(t)\" @longpress=\"onLongPressHistory(t)\">\r\n\t\t\t\t\t\t\t\t\t{{ t }}\r\n\t\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t\t\t<view class=\"drawer-divider\" />\r\n\t\t\t\t\t\t\t</view>\r\n\t\t\t\t\t\t</scroll-view>\r\n\t\t\t\t\t\t<view class=\"drawer-footer\">\r\n\t\t\t\t\t\t\t<view class=\"user-icon\">👤</view>\r\n\t\t\t\t\t\t\t<text class=\"user-name\">用户名</text>\r\n\t\t\t\t\t\t\t<view class=\"footer-gear\" @tap=\"clearAllHistory\">⚙️</view>\r\n\t\t\t\t\t\t</view>\r\n\t\t\t\t\t</view>\r\n\t\t\t\t</view>\r\n\t\t\t</uni-popup>\r\n\r\n\t\t\t<!-- Voice recording overlay -->\r\n\t\t\t<view v-if=\"isRecording\" class=\"record-mask\">\r\n\t\t\t\t<view class=\"record-box\" :class=\"{ cancel: willCancel }\">\r\n\t\t\t\t\t<view class=\"record-icon\">🎙️</view>\r\n\t\t\t\t\t<view class=\"record-text\">{{ willCancel ? '松开手指，取消发送' : '手指上滑，取消发送' }}</view>\r\n\t\t\t\t</view>\r\n\t\t\t</view>\r\n\t\t\t\r\n\t\t\t<view v-if=\"isRecording\" class=\"mask-layer\" @touchmove.stop.prevent>\r\n\t\t\t\t\r\n\t\t\t</view>\r\n\r\n\t\t</view>\r\n\t</template>\r\n\r\n\t<script>\r\n\t\tconst HISTORY_KEY = 'chat_history_groups'\r\n\t\texport default {\r\n\t\t\tdata() {\r\n\t\t\t\treturn {\r\n\t\t\t\t\tinputText: '',\r\n\t\t\t\t\tmessages: [\r\n\t\t\t\t\t\t// {\r\n\t\t\t\t\t\t// \tid: 1,\r\n\t\t\t\t\t\t// \trole: 'user',\r\n\t\t\t\t\t\t// \ttype: 'text',\r\n\t\t\t\t\t\t// \tcontent: '帮我统计一下今日的销售数据',\r\n\t\t\t\t\t\t// \tinputType: 'text'\r\n\t\t\t\t\t\t// },\r\n\t\t\t\t\t\t// {\r\n\t\t\t\t\t\t// \tid: 2,\r\n\t\t\t\t\t\t// \trole: 'assistant',\r\n\t\t\t\t\t\t// \ttype: 'card',\r\n\t\t\t\t\t\t// \ttitle: '今日销售数据统计结果如下：',\r\n\t\t\t\t\t\t// \tcontent: '内容内容........................'\r\n\t\t\t\t\t\t// }\r\n\t\t\t\t\t],\r\n\t\t\t\t\tscrollInto: '',\r\n\t\t\t\t\tdrawerOpen: false,\r\n\t\t\t\t\thistoryGroups: [\r\n\t\t\t\t\t],\r\n\t\t\t\t\tisRecording: false,\r\n\t\t\t\t\tisLoading:false,\r\n\t\t\t\t\twillCancel: false,\r\n\t\t\t\t\trecorder: null,\r\n\t\t\t\t\trecordStartY: 0,\r\n\t\t\t\t\trecordStartTs: 0,\r\n\t\t\t\t\trecordSimTimer: null,\r\n\t\t\t\t\t// show: false,\r\n\t\t\t\t\tinnerAudioContext: null,\r\n\t\t\t\t\tpopupVisible: false,\r\n\t\t\t\t\ttypewriterTimers: {},\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tcomputed: {\r\n\t\t\t\ttimeOfDayText() {\r\n\t\t\t\t\tconst h = new Date().getHours()\r\n\t\t\t\t\tif (h < 6) return '凌晨好'\r\n\t\t\t\t\tif (h < 12) return '上午好'\r\n\t\t\t\t\tif (h < 18) return '下午好'\r\n\t\t\t\t\treturn '晚上好'\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tmounted() {\r\n\t\t\t\tthis.loadChatHistory()\r\n\t\t\t\tthis.scrollToBottom();\r\n\t\t\t},\r\n\t\t\tbeforeDestroy() {\r\n\t\t\t\t// 清理所有打字机定时器\r\n\t\t\t\tObject.values(this.typewriterTimers).forEach(timer => {\r\n\t\t\t\t\tif (timer) clearInterval(timer)\r\n\t\t\t\t})\r\n\t\t\t\tthis.typewriterTimers = {}\r\n\t\t\t\t// 清理录音定时器\r\n\t\t\t\tif (this.recordSimTimer) {\r\n\t\t\t\t\tclearTimeout(this.recordSimTimer)\r\n\t\t\t\t}\r\n\t\t\t},\r\n\t\t\tmethods: {\r\n\t\t\t\t// 新增方法：上传音频并识别\r\n\t\t\tasync recognizeAudio(tempFilePath) {\r\n\t\t\t  try {\r\n\t\t\t\tconsole.log('开始语音识别，文件路径:', tempFilePath)\r\n\t\t\t\t\r\n\t\t\t\t// 获取文件信息\r\n\t\t\t\tconst fileInfo = await new Promise((resolve, reject) => {\r\n\t\t\t\t  uni.getFileInfo({\r\n\t\t\t\t\tfilePath: tempFilePath,\r\n\t\t\t\t\tsuccess: resolve,\r\n\t\t\t\t\tfail: reject\r\n\t\t\t\t  })\r\n\t\t\t\t})\r\n\t\t\t\t\r\n\t\t\t\tconsole.log('文件大小:', fileInfo.size)\r\n\t\t\t\t\r\n\t\t\t\t// 使用 UniApp 的上传文件 API\r\n\t\t\t\tconst uploadRes = await new Promise((resolve, reject) => {\r\n\t\t\t\t  uni.uploadFile({\r\n\t\t\t\t\t// url: 'http://192.168.133.83:8000/recognize_speech',\r\n\t\t\t\t\t// url: 'http://192.168.10.44:8000/recognize_speech',\r\n\t\t\t\t\t// url: 'http://192.168.1.18:8000/recognize_speech',\r\n\t\t\t\t\turl: 'http://106.227.91.181:8000/recognize_speech',\r\n\t\t\t\t\tfilePath: tempFilePath,\r\n\t\t\t\t\tname: 'speech', // 对应后端的 UploadFile 参数名\r\n\t\t\t\t\tformData: {\r\n\t\t\t\t\t  'format': 'amr',\r\n\t\t\t\t\t  'rate': 16000,\r\n\t\t\t\t\t  'channel': 1,\r\n\t\t\t\t\t  'cuid': 'uniapp_user',\r\n\t\t\t\t\t  'audio_len': fileInfo.size\r\n\t\t\t\t\t},\r\n\t\t\t\t\tsuccess: (res) => {\r\n\t\t\t\t\t  console.log('上传响应:', res)\r\n\t\t\t\t\t  if (res.statusCode === 200) {\r\n\t\t\t\t\t\ttry {\r\n\t\t\t\t\t\t  // 尝试解析返回的 JSON 数据\r\n\t\t\t\t\t\t  const data = JSON.parse(res.data)\r\n\t\t\t\t\t\t  resolve({ statusCode: 200, data })\r\n\t\t\t\t\t\t} catch (e) {\r\n\t\t\t\t\t\t  reject(new Error('响应解析失败: ' + e.message))\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t  } else {\r\n\t\t\t\t\t\treject(new Error(`上传失败: ${res.statusCode}`))\r\n\t\t\t\t\t  }\r\n\t\t\t\t\t},\r\n\t\t\t\t\tfail: (err) => {\r\n\t\t\t\t\t  reject(new Error('上传请求失败: ' + err.errMsg))\r\n\t\t\t\t\t}\r\n\t\t\t\t  })\r\n\t\t\t\t})\r\n\t\t\t\r\n\t\t\t\tconsole.log('语音识别响应:', uploadRes)\r\n\t\t\t\r\n\t\t\t\tconst result = uploadRes.data\r\n\t\t\t\tif (result.status === 'success') {\r\n\t\t\t\t  return result.result\r\n\t\t\t\t} else {\r\n\t\t\t\t  throw new Error(result.error || '识别失败')\r\n\t\t\t\t}\r\n\t\t\t  } catch (error) {\r\n\t\t\t\tconsole.error('语音识别错误:', error)\r\n\t\t\t\tuni.showToast({\r\n\t\t\t\t  title: '识别失败: ' + (error.message || '网络错误'),\r\n\t\t\t\t  icon: 'none'\r\n\t\t\t\t})\r\n\t\t\t\treturn null\r\n\t\t\t  }\r\n\t\t\t},\r\n\t\t\t\t// ==================== 历史记录管理 ====================\r\n\t\t\t\t\tformatDate(date) {\r\n\t\t\t\t\t  const y = date.getFullYear()\r\n\t\t\t\t\t  const m = String(date.getMonth() + 1).padStart(2, '0')\r\n\t\t\t\t\t  const d = String(date.getDate()).padStart(2, '0')\r\n\t\t\t\t\t  return `${y}年${m}月${d}日`\r\n\t\t\t\t\t},\r\n\t\t\t\t\tloadChatHistory() {\r\n\t\t\t\t\t\t  try {\r\n\t\t\t\t\t\t\tconst data = uni.getStorageSync(HISTORY_KEY)\r\n\t\t\t\t\t\t\tif (data && Array.isArray(data.groups)) {\r\n\t\t\t\t\t\t\t  this.historyGroups = data.groups\r\n\t\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\t  this.historyGroups = []\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t  } catch (e) {\r\n\t\t\t\t\t\t\tthis.historyGroups = []\r\n\t\t\t\t\t\t  }\r\n\t\t\t\t\t\t},\r\n\t\t\t\t\taddToHistory(text) {\r\n\t\t\t\t\t\t  let groups = uni.getStorageSync(HISTORY_KEY)?.groups || []\r\n\t\t\t\t\t\t  const today = this.formatDate(new Date())\r\n\t\t\t\t\t\t  let todayGroup = groups.find(g => g.date === today)\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t  if (!todayGroup) {\r\n\t\t\t\t\t\t\ttodayGroup = { date: today, items: [] }\r\n\t\t\t\t\t\t\tgroups.unshift(todayGroup)\r\n\t\t\t\t\t\t  }\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t  if (!todayGroup.items.includes(text)) {\r\n\t\t\t\t\t\t\ttodayGroup.items.unshift(text)\r\n\t\t\t\t\t\t  }\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t  // 限制大小\r\n\t\t\t\t\t\t  if (todayGroup.items.length > 50) todayGroup.items = todayGroup.items.slice(0, 50)\r\n\t\t\t\t\t\t  if (groups.length > 30) groups = groups.slice(0, 30)\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t  this.historyGroups = groups\r\n\t\t\t\t\t\t  uni.setStorageSync(HISTORY_KEY, { groups, updatedAt: Date.now() })\r\n\t\t\t\t\t\t},\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\tremoveFromHistory(text) {\r\n\t\t\t\t\t\t\t  let groups = uni.getStorageSync(HISTORY_KEY)?.groups || []\r\n\t\t\t\t\t\t\t  groups.forEach(group => {\r\n\t\t\t\t\t\t\t\tgroup.items = group.items.filter(item => item !== text)\r\n\t\t\t\t\t\t\t  })\r\n\t\t\t\t\t\t\t  groups = groups.filter(g => g.items.length > 0)\r\n\t\t\t\t\t\t\t  this.historyGroups = groups\r\n\t\t\t\t\t\t\t  uni.setStorageSync(HISTORY_KEY, { groups, updatedAt: Date.now() })\r\n\t\t\t\t\t\t\t},\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\tclearAllHistory() {\r\n\t\t\t\t\t\t\t  uni.showModal({\r\n\t\t\t\t\t\t\t\ttitle: '清除全部',\r\n\t\t\t\t\t\t\t\tcontent: '将删除所有对话记录，此操作不可恢复',\r\n\t\t\t\t\t\t\t\tsuccess: (res) => {\r\n\t\t\t\t\t\t\t\t  if (res.confirm) {\r\n\t\t\t\t\t\t\t\t\tuni.removeStorageSync(HISTORY_KEY)\r\n\t\t\t\t\t\t\t\t\tthis.historyGroups = []\r\n\t\t\t\t\t\t\t\t\tuni.showToast({ title: '已清除', icon: 'success' })\r\n\t\t\t\t\t\t\t\t  }\r\n\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t  })\r\n\t\t\t\t\t\t\t},\r\n\t\t\t\t\t\tonLongPressHistory(text) {\r\n\t\t\t\t\t\t\t  uni.showModal({\r\n\t\t\t\t\t\t\t\ttitle: '删除记录',\r\n\t\t\t\t\t\t\t\tcontent: '确定删除这条对话记录？',\r\n\t\t\t\t\t\t\t\tsuccess: (res) => {\r\n\t\t\t\t\t\t\t\t  if (res.confirm) {\r\n\t\t\t\t\t\t\t\t\tthis.removeFromHistory(text)\r\n\t\t\t\t\t\t\t\t  }\r\n\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t  })\r\n\t\t\t\t\t\t\t},\t\t\r\n\t\t\t\t\t// 工具\r\n\t\t\t\t\t\tremoveMessage(id) {\r\n\t\t\t\t\t\t  const idx = this.messages.findIndex(m => m.id === id)\r\n\t\t\t\t\t\t  if (idx > -1) this.messages.splice(idx, 1)\r\n\t\t\t\t\t\t},\r\n\t\t\t\t\t\taddAssistantMessage(id, content) {\r\n\t\t\t\t\t\t\t  this.messages.push({ id, role: 'assistant', type: 'text', content, displayText: '' })\r\n\t\t\t\t\t\t\t},\r\n\t\t\t\t\t\t\t\t\r\n\t\t\t\tasync getAIResponse(message){\r\n\t\t\t\t\ttry {\r\n\t\t\t\t\t\t\t// const url = 'http://192.168.133.83:9020/api/chat'\r\n\t\t\t\t\t\t\t// const url = 'http://192.168.10.44:9020/api/chat'\r\n\t\t\t\t\t\t\tconst url = 'http://106.227.91.181:9020/api/chat'  // 如需切换线上，改这里即可\r\n\t\t\t\t\t\t\tconst headers = { 'Content-Type': 'application/json' }\r\n\t\t\t\t\t\t\tconst data = { message }\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\t\t// console.log(data)\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t  //       const [error, res] = await uni.request({\r\n\t\t\t\t\t  //         url,\r\n\t\t\t\t\t  //         method: 'POST',\r\n\t\t\t\t\t  //         header: headers,\r\n\t\t\t\t\t  //         data\r\n\t\t\t\t\t  //       })\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\t\t// console.log(res)\r\n\t\t\t\t\t\t\t// 使用 Promise 风格\r\n\t\t\t\t\t\t\tconst res = await new Promise((resolve, reject) => {\r\n\t\t\t\t\t\t\t  uni.request({\r\n\t\t\t\t\t\t\t\turl,\r\n\t\t\t\t\t\t\t\tmethod: 'POST',\r\n\t\t\t\t\t\t\t\theader: headers,\r\n\t\t\t\t\t\t\t\tdata,\r\n\t\t\t\t\t\t\t\tsuccess: (res) => resolve(res),\r\n\t\t\t\t\t\t\t\tfail: (err) => reject(err)\r\n\t\t\t\t\t\t\t  })\r\n\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t\r\n\t\t\t\t\t\t\tconsole.log('请求响应:', res)\r\n\t\t\t\t\t\r\n\t\t\t\t\t\t\tif (res.statusCode !== 200) {\r\n\t\t\t\t\t\t\t\t  throw new Error(`HTTP错误! 状态码: ${res.statusCode}`)\r\n\t\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\t\t\treturn res.data?.result?.data || '未获取到有效回复'\r\n\t\t\t\t\t\t\t\r\n\t\t\t\t\t\t\t  } catch (error) {\r\n\t\t\t\t\t\t\t\tconsole.error('AI请求错误:', error)\r\n\t\t\t\t\t\t\t\treturn `抱歉，出了点问题: ${error.errMsg || error.message}`\r\n\t\t\t\t\t\t\t  }\r\n\t\t\t\t},\r\n\t\t\t\t\r\n\t\t\t\tplayVoice(voicePath) {\r\n\t\t\t\t\tif (!voicePath) {\r\n\t\t\t\t\t\tuni.showToast({ title: '无可播放的语音', icon: 'none' })\r\n\t\t\t\t\t\treturn\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif (!this.innerAudioContext) {\r\n\t\t\t\t\t\tthis.innerAudioContext = uni.createInnerAudioContext()\r\n\t\t\t\t\t\tthis.innerAudioContext.autoplay = false\r\n\t\t\t\t\t\tthis.innerAudioContext.onError(() => {\r\n\t\t\t\t\t\t\tuni.showToast({ title: '播放失败', icon: 'none' })\r\n\t\t\t\t\t\t});\r\n\t\t\t\t\t}\r\n\t\t\t\t\ttry { this.innerAudioContext.stop() } catch(e) {}\r\n\t\t\t\t\tconsole.log(voicePath)\r\n\t\t\t\t\tthis.innerAudioContext.src = voicePath\r\n\t\t\t\t\tthis.innerAudioContext.play()\r\n\t\t\t\t},\r\n\t\t\t\tonSettingTap() {\r\n\t\t\t\t\tuni.navigateTo({\r\n\t\t\t\t\t\turl: '/pages/setting/index'\r\n\t\t\t\t\t})\r\n\t\t\t\t},\r\n\t\t\t\tonSuggestionTap(text) {\r\n\t\t\t\t\tthis.inputText = text\r\n\t\t\t\t\tthis.onSend();\r\n\t\t\t\t},\r\n\t\t\t\tonQuickAsk(text) {\r\n\t\t\t\t\tthis.inputText = text\r\n\t\t\t\t\tthis.onSend()\r\n\t\t\t\t},\r\n\t\t\t\tonSwitchModel() {\r\n\t\t\t\t\tuni.showToast({\r\n\t\t\t\t\t\ttitle: '已切换为通用模型',\r\n\t\t\t\t\t\ticon: 'none'\r\n\t\t\t\t\t})\r\n\t\t\t\t},\r\n\t\t\t\tonInput(e) {\r\n\t\t\t\t\tthis.inputText = e.detail.value\r\n\t\t\t\t},\r\n\t\t\t\topenDrawer() {\r\n\t\t\t\t\tthis.$refs.popup.open()\r\n\t\t\t\t},\r\n\t\t\t\tonPopupChange(e){\r\n\t\t\t\t\t// e.show: true when opened, false when closed\r\n\t\t\t\t\tthis.popupVisible = !!(e && (e.show === true))\r\n\t\t\t\t},\r\n\t\t\t\t// ===== Voice input (WeChat-like) =====\r\n\t\t\t\tensureRecorder() {\r\n\t\t\t\t\tif (this.recorder) return\r\n\t\t\t\t\ttry {\r\n\t\t\t\t\t\tthis.recorder = uni.getRecorderManager && uni.getRecorderManager()\r\n\t\t\t\t\t} catch (e) {\r\n\t\t\t\t\t\tthis.recorder = null\r\n\t\t\t\t\t}\r\n\t\t\t\t\tif (this.recorder) {\r\n\t\t\t\t\t\tthis.recorder.onStart()\r\n\t\t\t\t\t\tthis.recorder.onStop( async(res) => {\r\n\t\t\t\t\t\t\tconst duration = Date.now() - this.recordStartTs;\r\n\t\t\t\t\t\t\tconst tempFilePath = res.tempFilePath; // 添加这行，从res中获取文件路径\r\n\t\t\t\t\t\t\tif (this.willCancel || duration < 700) {\r\n\t\t\t\t\t\t\t\tuni.showToast({\r\n\t\t\t\t\t\t\t\t\ttitle: duration < 700 ? '说话时间太短' : '已取消',\r\n\t\t\t\t\t\t\t\t\ticon: 'none'\r\n\t\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t\t\treturn\r\n\t\t\t\t\t\t\t}\r\n\t\t\t\t\t\t\t// 显示加载\r\n\t\t\t\t\t\t\tuni.showLoading({ title: '识别中...' });\r\n\t\t\t\t\t\t\t\t  \r\n\t\t\t\t\t\t\t// TODO: 上传 res.tempFilePath 做识别；现用 mock\r\n\t\t\t\t\t\t\t// this.handleRecognizedText(this.mockSpeechToText(duration), res.tempFilePath, Math.ceil(\r\n\t\t\t\t\t\t\t// \tduration / 100) / 10)\r\n\t\t\t\t\t\t\t// 真实识别\r\n\t\t\t\t\t\t\tconst recognizedText = await this.recognizeAudio(tempFilePath);\r\n\t\t\t\t\t\t\tuni.hideLoading();\t \r\n\t\t\t\t\t\t\t if (recognizedText) {\r\n\t\t\t\t\t\t\t\t\t // 成功：填入输入框\r\n\t\t\t\t\t\t\t\t\t this.inputText = recognizedText;\r\n\t\t\t\t\t\t\t\t\t this.$nextTick(() => {\r\n\t\t\t\t\t\t\t\t\t   // 可选：自动发送\r\n\t\t\t\t\t\t\t\t\t   // this.onSend('voice', tempFilePath, Math.ceil(duration / 100) / 10);\r\n\t\t\t\t\t\t\t\t\t });\r\n\t\t\t\t\t\t\t\t   }\r\n\t\t\t\t\t\t})\r\n\t\t\t\t\t}\r\n\t\t\t\t},\r\n\t\t\t\tonPressMic(e) {\r\n\t\t\t\t\tthis.ensureRecorder()\r\n\t\t\t\t\tthis.isRecording = true\r\n\t\t\t\t\t// this.show = true\r\n\t\t\t\t\tthis.willCancel = false\r\n\t\t\t\t\tthis.recordStartTs = Date.now()\r\n\t\t\t\t\tthis.recordStartY = (e.changedTouches && e.changedTouches[0]) ? e.changedTouches[0].clientY : 0\r\n\t\t\t\t\tif (this.recorder) {\r\n\t\t\t\t\t\ttry {\r\n\t\t\t\t\t\t\tthis.recorder.start({\r\n\t\t\t\t\t\t\t\tformat: 'amr',\r\n\t\t\t\t\t\t\t\tsampleRate: 16000,\r\n\t\t\t\t\t\t\t\tencodeBitRate: 16000, // 编码比特率\r\n\t\t\t\t\t\t\t\tframeSize: 4, // 帧大小\r\n\t\t\t\t\t\t\t\tnumberOfChannels: 1,\r\n\t\t\t\t\t\t\t\tduration: 60000\r\n\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t} catch (err) {}\r\n\t\t\t\t\t} else {\r\n\t\t\t\t\t\tif (this.recordSimTimer) clearTimeout(this.recordSimTimer)\r\n\t\t\t\t\t\tthis.recordSimTimer = setTimeout(() => {}, 60000)\r\n\t\t\t\t\t}\r\n\t\t\t\t},\r\n\t\t\t\tonMoveMic(e) {\r\n\t\t\t\t\tif (!this.isRecording) return;\r\n\t\t\t\t\tconst y = (e.changedTouches && e.changedTouches[0]) ? e.changedTouches[0].clientY : 0;\r\n\t\t\t\t\tthis.willCancel = (this.recordStartY - y) > 60\r\n\t\t\t\t},\r\n\t\t\t\tonReleaseMic() {\r\n\t\t\t\t\tconsole.log('onReleaseMic');\r\n\t\t\t\t\tif (!this.isRecording) return\r\n\t\t\t\t\tconst cancel = this.willCancel\r\n\t\t\t\t\tthis.isRecording = false;\r\n\t\t\t\t\tthis.show = false\r\n\t\t\t\t\tif (this.recorder) {\r\n\t\t\t\t\t\ttry {\r\n\t\t\t\t\t\t\tthis.recorder.stop()\r\n\t\t\t\t\t\t} catch (err) {\r\n\t\t\t\t\t\t\tconsole.log('err', err);\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}\r\n\t\t\t\t},\r\n\t\t\t\thandleRecognizedText(text, tempFilePath, duration) {\r\n\t\t\t\t\tif (!text) return\r\n\t\t\t\t\tthis.inputText = text\r\n\t\t\t\t\tthis.onSend('voice', tempFilePath, duration) // 传 'voice'\r\n\t\t\t\t},\r\n\t\t\t\tmockSpeechToText(ms) {\r\n\t\t\t\t\tconst sec = Math.ceil(ms / 100) / 10\r\n\t\t\t\t\tconst pool = [\r\n\t\t\t\t\t\t`语音输入 ${sec}s，模拟识别：帮我统计今天销售额`,\r\n\t\t\t\t\t\t`语音输入 ${sec}s，模拟识别：查询订单20388993483`,\r\n\t\t\t\t\t\t`语音输入 ${sec}s，模拟识别：生成日报`\r\n\t\t\t\t\t]\r\n\t\t\t\t\treturn pool[Math.floor(Math.random() * pool.length)]\r\n\t\t\t\t},\r\n\t\t\t\tonHistoryItemTap(text) {\r\n\t\t\t\t\tthis.inputText = text\r\n\t\t\t\t\tthis.onSend();\r\n\t\t\t\t\tthis.$refs.popup.close()\r\n\t\t\t\t},\r\n\t\t\t\tasync onSend(inputType = 'text', inputContent = '', duration = undefined) {\r\n\t\t\t\t  const text = (this.inputText || '').trim()\r\n\t\t\t\t  if (!text || this.isLoading) return\r\n\t\t\t\t\r\n\t\t\t\t  const baseId = Date.now()\r\n\t\t\t\t\r\n\t\t\t\t  // 1. 用户消息\r\n\t\t\t\t  this.messages.push({\r\n\t\t\t\t\tid: baseId,\r\n\t\t\t\t\trole: 'user',\r\n\t\t\t\t\ttype: 'text',\r\n\t\t\t\t\tcontent: text,\r\n\t\t\t\t\tinputType,\r\n\t\t\t\t\tinputContent,\r\n\t\t\t\t\tduration\r\n\t\t\t\t  })\r\n\t\t\t\t\r\n\t\t\t\t  // 2. loading 消息\r\n\t\t\t\t  const loadingId = baseId + 0.5\r\n\t\t\t\t  this.messages.push({\r\n\t\t\t\t\tid: loadingId,\r\n\t\t\t\t\trole: 'assistant',\r\n\t\t\t\t\tloading: true\r\n\t\t\t\t  })\r\n\t\t\t\t\r\n\t\t\t\t  this.scrollToBottom()\r\n\t\t\t\t  this.inputText = ''\r\n\t\t\t\t  this.isLoading = true\r\n\t\t\t\t\r\n\t\t\t\t  this.addToHistory(text)\r\n\t\t\t\t\r\n\t\t\t\t  try {\r\n\t\t\t\t\t// 3. 真正等待 AI 回复\r\n\t\t\t\t\tconst reply = await this.getAIResponse(text)\r\n\t\t\t\t\r\n\t\t\t\t\t// 4. 移除 loading\r\n\t\t\t\t\tconst loadingIdx = this.messages.findIndex(m => m.id === loadingId)\r\n\t\t\t\t\tif (loadingIdx > -1) this.messages.splice(loadingIdx, 1)\r\n\t\t\t\t\r\n\t\t\t\t\t// 5. 添加回复 + 打字机\r\n\t\t\t\t\tconst replyId = baseId + 1\r\n\t\t\t\t\tthis.messages.push({\r\n\t\t\t\t\t  id: replyId,\r\n\t\t\t\t\t  role: 'assistant',\r\n\t\t\t\t\t  type: 'text',\r\n\t\t\t\t\t  content: reply,\r\n\t\t\t\t\t  displayText: ''\r\n\t\t\t\t\t})\r\n\t\t\t\t\r\n\t\t\t\t\tthis.typewriter(replyId, reply)\r\n\t\t\t\t  } catch (e) {\r\n\t\t\t\t\t// 出错时也展示\r\n\t\t\t\t\tconst loadingIdx = this.messages.findIndex(m => m.id === loadingId)\r\n\t\t\t\t\tif (loadingIdx > -1) this.messages.splice(loadingIdx, 1)\r\n\t\t\t\t\tthis.messages.push({\r\n\t\t\t\t\t  id: baseId + 1,\r\n\t\t\t\t\t  role: 'assistant',\r\n\t\t\t\t\t  content: `请求出错：${e.message || e}`\r\n\t\t\t\t\t})\r\n\t\t\t\t  } finally {\r\n\t\t\t\t\tthis.isLoading = false\r\n\t\t\t\t\tthis.$nextTick(() => this.scrollToBottom())\r\n\t\t\t\t  }\r\n\t\t\t\t},\r\n\t\t\t\ttypewriter(messageId, fullText) {\r\n\t\t\t\t\tconst msg = this.messages.find(m => m.id === messageId)\r\n\t\t\t\t\tif (!msg) return\r\n\t\t\t\t\t// 清理之前的定时器（如果存在）\r\n\t\t\t\t\tif (this.typewriterTimers[messageId]) {\r\n\t\t\t\t\t\tclearInterval(this.typewriterTimers[messageId])\r\n\t\t\t\t\t}\r\n\t\t\t\t\tlet index = 0\r\n\t\t\t\t\tmsg.displayText = fullText.substring(0, index + 1);\r\n\t\t\t\t\tindex += 1;\r\n\t\t\t\t\tconst speed = 50 // 每个字符间隔50ms\r\n\t\t\t\t\tconst timer = setInterval(() => {\r\n\t\t\t\t\t\tif (index < fullText.length) {\r\n\t\t\t\t\t\t\tmsg.displayText = fullText.substring(0, index + 1)\r\n\t\t\t\t\t\t\tindex++\r\n\t\t\t\t\t\t\t// 打字过程中自动滚动到底部\r\n\t\t\t\t\t\t\tthis.$nextTick(() => {\r\n\t\t\t\t\t\t\t\tthis.scrollToBottom()\r\n\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t} else {\r\n\t\t\t\t\t\t\tclearInterval(timer)\r\n\t\t\t\t\t\t\tdelete this.typewriterTimers[messageId]\r\n\t\t\t\t\t\t\t// 完成后使用完整文本\r\n\t\t\t\t\t\t\tmsg.displayText = fullText\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t}, speed)\r\n\t\t\t\t\tthis.typewriterTimers[messageId] = timer\r\n\t\t\t\t},\r\n\t\t\t\tscrollToBottom() {\r\n\t\t\t\t\t\t\t\tlet self = this;\r\n\t\t\t\t\t\t\t\tthis.$nextTick(() => {\r\n\t\t\t\t\t\t\t\t\tuni.createSelectorQuery().select('.content').boundingClientRect((rect) => {\r\n\t\t\t\t\t\t\t\t\t\tif(self.height !== rect.height){\r\n\t\t\t\t\t\t\t\t\t\t\tself.height = rect.height;\r\n\t\t\t\t\t\t\t\t\t\t\tuni.pageScrollTo({\r\n\t\t\t\t\t\t\t\t\t\t\t\tscrollTop: rect.height,\r\n\t\t\t\t\t\t\t\t\t\t\t\tduration: 300,\r\n\t\t\t\t\t\t\t\t\t\t\t\tclass: '.content'\r\n\t\t\t\t\t\t\t\t\t\t\t});\r\n\t\t\t\t\t\t\t\t\t\t}\t\t\t\t\r\n\t\t\t\t\t\t\t\t\t}).exec();\r\n\t\t\t\t\t\t\t\t})\r\n\t\t\t\t\t\t\t},\r\n\t\t\t\tmockReply(text) {\r\n\t\t\t\t\tconst candidates = [\r\n\t\t\t\t\t\t'好的，我已经为您处理。',\r\n\t\t\t\t\t\t'收到请求，以下是结果的概览。',\r\n\t\t\t\t\t\t'我理解了，这是一个示例回复。',\r\n\t\t\t\t\t\t'已记录，稍后将完善报表。'\r\n\t\t\t\t\t]\r\n\t\t\t\t\tconst pick = candidates[Math.floor(Math.random() * candidates.length)]\r\n\t\t\t\t\treturn pick + '（已收到：“' + text + '”）'\r\n\t\t\t\t},\r\n\t\t\t\tonListen(text) {\r\n\t\t\t\t\ttry {\r\n\t\t\t\t\t\t// H5: Web Speech API\r\n\t\t\t\t\t\tif (typeof window !== 'undefined' && window.speechSynthesis) {\r\n\t\t\t\t\t\t\tconst u = new SpeechSynthesisUtterance(String(text))\r\n\t\t\t\t\t\t\tu.lang = 'zh-CN'\r\n\t\t\t\t\t\t\tu.rate = 1\r\n\t\t\t\t\t\t\tu.pitch = 1\r\n\t\t\t\t\t\t\twindow.speechSynthesis.cancel()\r\n\t\t\t\t\t\t\twindow.speechSynthesis.speak(u)\r\n\t\t\t\t\t\t\treturn\r\n\t\t\t\t\t\t}\r\n\t\t\t\t\t} catch (e) {}\r\n\t\t\t\t\tuni.showToast({\r\n\t\t\t\t\t\ttitle: '当前端不支持语音播放',\r\n\t\t\t\t\t\ticon: 'none'\r\n\t\t\t\t\t})\r\n\t\t\t\t}\r\n\t\t\t}\r\n\t\t}\r\n\t</script>\r\n\r\n\t<style scoped>\r\n\t\t::v-deep .uni-nav-bar-text{\r\n\t\t\tfont-size: 18px !important;\r\n\t\t}\r\n\t\t\r\n\t\t::v-deeo .uni-navbar--border{\r\n\t\t\t/* border-bottom: 0px !important; */\r\n\t\t\tborder-bottom: 1px solid #fff !important;\r\n\t\t}\r\n\t\t\r\n\t\t.ai-page {\r\n\t\t\theight: 100vh;\r\n\t\t\tdisplay: flex;\r\n\t\t\tflex-direction: column;\r\n\t\t\tbackground: #f7f8fc;\r\n\t\t}\r\n\r\n\t\t.nav {\r\n\t\t\theight: 44px;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: space-between;\r\n\t\t\tpadding: 0 12px;\r\n\t\t\tbackground: #ffffff;\r\n\t\t\tbox-shadow: 0 2px 8px rgba(0, 0, 0, 0.04);\r\n\t\t\tposition: fixed;\r\n\t\t\tleft: 0;\r\n\t\t\tright: 0;\r\n\t\t\ttop: 0;\r\n\t\t\tz-index: 9;\r\n\t\t}\r\n\r\n\t\t.nav-title {\r\n\t\t\tfont-size: 16px;\r\n\t\t\tfont-weight: 600;\r\n\t\t}\r\n\r\n\t\t.hamburger {\r\n\t\t\twidth: 18px;\r\n\t\t}\r\n\r\n\t\t.hamburger .line {\r\n\t\t\theight: 2px;\r\n\t\t\tbackground: #333;\r\n\t\t\tmargin: 3px 0;\r\n\t\t\tborder-radius: 2px;\r\n\t\t}\r\n\r\n\t\t.gear {\r\n\t\t\twidth: 18px;\r\n\t\t\theight: 18px;\r\n\t\t\tposition: relative;\r\n\t\t\tcolor: #000;\r\n\t\t}\r\n\r\n\t\t.content {\r\n\t\t\tflex: 1;\r\n\t\t\tpadding: 16px 12px 68px 12px;\r\n\t\t\tbackground-color: #f7f8fc;\r\n\t\t\twidth: 100%;\r\n\t\t\tbox-sizing: border-box;\r\n\t\t}\r\n\r\n\t\t.greet-card {\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 14px;\r\n\t\t\tpadding: 12px;\r\n\t\t\tmargin-bottom: 10px;\r\n\t\t}\r\n\r\n\t\t.avatar-inner {\r\n\t\t\tfont-size: 26px;\r\n\t\t}\r\n\r\n\t\t.greet-text .hi {\r\n\t\t\tfont-size: 16px;\r\n\t\t\tfont-weight: 700;\r\n\t\t\tcolor: #0b56ff;\r\n\t\t}\r\n\r\n\t\t.greet-text .sub {\r\n\t\t\tfont-size: 12px;\r\n\t\t\tcolor: #4a76b1;\r\n\t\t\tmargin-top: 4px;\r\n\t\t}\r\n\r\n\t\t.welcome {\r\n\t\t\tfont-size: 13px;\r\n\t\t\tcolor: #333;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 12px;\r\n\t\t\tpadding: 10px 12px;\r\n\t\t\tmargin: 12px 0;\r\n\t\t}\r\n\r\n\t\t.guess-panel {\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 14px;\r\n\t\t\tpadding: 10px;\r\n\t\t\tmargin-bottom: 16px;\r\n\t\t}\r\n\r\n\t\t.guess-title {\r\n\t\t\tcolor: #5f6fff;\r\n\t\t\tfont-size: 14px;\r\n\t\t\tmargin-bottom: 8px;\r\n\t\t}\r\n\r\n\t\t.guess-list {\r\n\t\t\tdisplay: flex;\r\n\t\t\tflex-direction: column;\r\n\t\t}\r\n\r\n\t\t.guess-item {\r\n\t\t\tbackground: #f7f8fc;\r\n\t\t\tborder-radius: 10px;\r\n\t\t\tpadding: 12px;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: space-between;\r\n\t\t\tmargin-bottom: 10px;\r\n\t\t\tbox-sizing: border-box;\r\n\t\t}\r\n\r\n\t\t.guess-item:last-child {\r\n\t\t\tmargin-bottom: 0;\r\n\t\t}\r\n\r\n\t\t.guess-item .arrow {\r\n\t\t\tcolor: #9aa3b2;\r\n\t\t\tfont-size: 18px;\r\n\t\t}\r\n\r\n\t\t.chat {\r\n\t\t\tmargin: 6px 0 12px;\r\n\t\t}\r\n\r\n\t\t.msg {\r\n\t\t\tmargin: 10px 0;\r\n\t\t\tdisplay: flex;\r\n\t\t}\r\n\r\n\t\t.msg.user {\r\n\t\t\tjustify-content: flex-end;\r\n\t\t}\r\n\r\n\t\t.bubble {\r\n\t\t\tmax-width: 80%;\r\n\t\t\tpadding: 10px 12px;\r\n\t\t\tborder-radius: 14px;\r\n\t\t\tfont-size: 14px;\r\n\t\t\tline-height: 1.5;\r\n\t\t}\r\n\r\n\t\t.user-bubble {\r\n\t\t\tbackground: #4e7bff;\r\n\t\t\tcolor: #fff;\r\n\t\t\tborder-bottom-right-radius: 4px;\r\n\t\t\tmargin-right: 6px;\r\n\t\t}\r\n\r\n\t\t.ai-bubble {\r\n\t\t\tbackground: #fff;\r\n\t\t\tcolor: #333;\r\n\t\t\tborder-bottom-left-radius: 4px;\r\n\t\t\tbox-shadow: 0 2px 8px rgba(0, 0, 0, 0.06);\r\n\t\t}\r\n\r\n\t\t.listen-btn {\r\n\t\t\tmargin-left: 8px;\r\n\t\t\tcolor: #6b7280;\r\n\t\t\tfont-size: 14px;\r\n\t\t}\r\n\r\n\t\t.ai-card-title {\r\n\t\t\tcolor: #5f6fff;\r\n\t\t\tfont-weight: 600;\r\n\t\t\tmargin-bottom: 6px;\r\n\t\t}\r\n\r\n\t\t.ai-card-body {\r\n\t\t\tcolor: #666;\r\n\t\t}\r\n\r\n\t\t/* loading animation */\r\n\t\t.ai-loading {\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tgap: 6px;\r\n\t\t\tpadding: 4px 0;\r\n\t\t}\r\n\r\n\t\t.loading-dot {\r\n\t\t\twidth: 8px;\r\n\t\t\theight: 8px;\r\n\t\t\tborder-radius: 50%;\r\n\t\t\tbackground: #9ca3af;\r\n\t\t\tanimation: loading-bounce 1.4s ease-in-out infinite both;\r\n\t\t}\r\n\r\n\t\t.loading-dot:nth-child(1) {\r\n\t\t\tanimation-delay: -0.32s;\r\n\t\t}\r\n\r\n\t\t.loading-dot:nth-child(2) {\r\n\t\t\tanimation-delay: -0.16s;\r\n\t\t}\r\n\r\n\t\t@keyframes loading-bounce {\r\n\t\t\t0%, 80%, 100% {\r\n\t\t\t\ttransform: scale(0.8);\r\n\t\t\t\topacity: 0.5;\r\n\t\t\t}\r\n\t\t\t40% {\r\n\t\t\t\ttransform: scale(1.2);\r\n\t\t\t\topacity: 1;\r\n\t\t\t}\r\n\t\t}\r\n\r\n\t\t/* bottom dock */\r\n\t\t.dock {\r\n\t\t\tposition: fixed;\r\n\t\t\tleft: 0;\r\n\t\t\tright: 0;\r\n\t\t\tbottom: 0;\r\n\t\t\tbackground: #f7f8fc;\r\n\t\t\tbox-shadow: 0 -4px 12px rgba(0, 0, 0, 0.06);\r\n\t\t\tpadding-bottom: constant(safe-area-inset-bottom);\r\n\t\t\tpadding-bottom: env(safe-area-inset-bottom);\r\n\t\t}\r\n\r\n\t\t.quick-actions {\r\n\t\t\tpadding: 6px 10px 4px;\r\n\t\t}\r\n\r\n\t\t.quick-actions.horizontal {\r\n\t\t\twhite-space: nowrap;\r\n\t\t\twidth: 95%;\r\n\t\t}\r\n\r\n\t\t.qa-btn {\r\n\t\t\tdisplay: inline-flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: center;\r\n\t\t\tmin-width: 96px;\r\n\t\t\ttext-align: center;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 10px;\r\n\t\t\tpadding: 8px 10px;\r\n\t\t\tfont-size: 12px;\r\n\t\t\tcolor: #3b3f45;\r\n\t\t\tbox-shadow: 0 2px 8px rgba(0, 0, 0, 0.04);\r\n\t\t\tmargin-right: 10px;\r\n\t\t}\r\n\r\n\t\t.qa-btn.minor {\r\n\t\t\tbackground: #eff1ff;\r\n\t\t\tcolor: #4e7bff;\r\n\t\t}\r\n\r\n\t\t.qa-btn:last-child {\r\n\t\t\tmargin-right: 0;\r\n\t\t}\r\n\r\n\t\t.input-bar {\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tpadding: 8px 10px 12px;\r\n\t\t\tgap: 8px;\r\n\t\t\tbackground: #f7f8fc;\r\n\t\t}\r\n\r\n\t\t.input {\r\n\t\t\tflex: 1;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-radius: 24px;\r\n\t\t\tpadding: 10px 14px;\r\n\t\t\tfont-size: 14px;\r\n\t\t}\r\n\r\n\t\t.ph {\r\n\t\t\tcolor: #9aa3b2;\r\n\t\t}\r\n\r\n\t\t.mic {\r\n\t\t\twidth: 36px;\r\n\t\t\theight: 36px;\r\n\t\t\tborder-radius: 18px;\r\n\t\t\tbackground: #fff;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: center;\r\n\t\t\tbox-shadow: 0 2px 8px rgba(0, 0, 0, 0.06);\r\n\t\t}\r\n\r\n\t\t.mic.recording {\r\n\t\t\tbackground: #fffbf0;\r\n\t\t\tbox-shadow: 0 0 0 2px rgba(255, 193, 7, .25) inset;\r\n\t\t}\r\n\r\n\t\t.send {\r\n\t\t\theight: 36px;\r\n\t\t\tline-height: 36px;\r\n\t\t\tpadding: 0 14px;\r\n\t\t\tborder-radius: 18px;\r\n\t\t\tbackground: #4e7bff;\r\n\t\t\tcolor: #fff;\r\n\t\t\tfont-size: 14px;\r\n\t\t}\r\n\r\n\t\t/* drawer */\r\n\t\t.drawer-mask {\r\n\t\t\twidth: 70vw;\r\n\t\t\theight: 100vh;\r\n\t\t}\r\n\r\n\t\t.drawer {\r\n\t\t\twidth: 100%;\r\n\t\t\theight: 100vh;\r\n\t\t\tbackground: #fff;\r\n\t\t\tborder-top-right-radius: 8px;\r\n\t\t\tborder-bottom-right-radius: 8px;\r\n\t\t\tdisplay: flex;\r\n\t\t\tflex-direction: column;\r\n\t\t}\r\n\r\n\t\t.drawer.show {\r\n\t\t\ttransform: translateX(0);\r\n\t\t}\r\n\r\n\t\t.drawer-scroll {\r\n\t\t\theight: calc(100vh - 64px);\r\n\t\t\tpadding: 12px;\r\n\t\t\tbox-sizing: border-box;\r\n\t\t}\r\n\r\n\t\t.drawer-group {\r\n\t\t\tpadding: 10px 8px 0;\r\n\t\t}\r\n\r\n\t\t.drawer-date {\r\n\t\t\tcolor: #9aa3b2;\r\n\t\t\tfont-size: 12px;\r\n\t\t\tmargin-bottom: 8px;\r\n\t\t}\r\n\r\n\t\t.drawer-item {\r\n\t\t\tcolor: #333;\r\n\t\t\tfont-size: 13px;\r\n\t\t\tline-height: 20px;\r\n\t\t\tmargin: 6px 0;\r\n\t\t}\r\n\r\n\t\t.drawer-divider {\r\n\t\t\theight: 1px;\r\n\t\t\tbackground: #eeeeee;\r\n\t\t\tmargin: 12px 0;\r\n\t\t}\r\n\r\n\t\t.drawer-footer {\r\n\t\t\tpadding: 12px;\r\n\t\t\tborder-top: 1px solid #eeeeee;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t}\r\n\r\n\t\t.drawer-footer {\r\n\t\t\t/* fixed height for calc above */\r\n\t\t\theight: 64px;\r\n\t\t}\r\n\r\n\t\t.user-icon {\r\n\t\t\twidth: 24px;\r\n\t\t\ttext-align: center;\r\n\t\t}\r\n\r\n\t\t.user-name {\r\n\t\t\tflex: 1;\r\n\t\t\tfont-size: 14px;\r\n\t\t\tcolor: #333;\r\n\t\t}\r\n\r\n\t\t.footer-gear {\r\n\t\t\twidth: 24px;\r\n\t\t\ttext-align: center;\r\n\t\t}\r\n\r\n\t\t/* voice overlay */\r\n\t\t.record-mask {\r\n\t\t\tposition: fixed;\r\n\t\t\tleft: 0;\r\n\t\t\tright: 0;\r\n\t\t\ttop: 0;\r\n\t\t\tbottom: 0;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tjustify-content: center;\r\n\t\t\tz-index: 9999;\r\n\t\t}\r\n\r\n\t\t.record-box {\r\n\t\t\tbackground: rgba(0, 0, 0, .75);\r\n\t\t\tcolor: #fff;\r\n\t\t\tpadding: 16px 18px;\r\n\t\t\tborder-radius: 12px;\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t\tgap: 10px;\r\n\t\t\tmin-width: 220rpx;\r\n\t\t}\r\n\r\n\t\t.record-box.cancel {\r\n\t\t\tbackground: rgba(221, 44, 0, .85);\r\n\t\t}\r\n\r\n\t\t.record-icon {\r\n\t\t\tfont-size: 20px;\r\n\t\t}\r\n\r\n\t\t.record-text {\r\n\t\t\tfont-size: 14px;\r\n\t\t}\r\n\r\n\t\t.text-voice {\r\n\t\t\tdisplay: flex;\r\n\t\t\talign-items: center;\r\n\t\t}\r\n\r\n\t\t.voice-play {\r\n\t\t\twidth: 20px;\r\n\t\t\tmargin-left: 5px;\r\n\t\t}\r\n\t\t\r\n\t\t.mask-layer{\r\n\t\t\tposition: fixed;\r\n\t\t\tleft: 0;\r\n\t\t\tright: 0;\r\n\t\t\ttop: 0;\r\n\t\t\tbottom: 0;\r\n\t\t\tbackground-color: rgba(0, 0, 0, .1);\r\n\t\t}\r\n\t\t\r\n\t\t\r\n\t</style>"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///51\n"); /***/ }), /* 52 */