index.vue 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616
  1. <template>
  2. <div class="chat-page">
  3. <div class="chat-wrap">
  4. <div class="chat-title">飞鸟种植大脑</div>
  5. <div class="chat-box" ref="chatBox">
  6. <div v-for="(msg, index) in messages" :key="index" class="message" :class="msg.type">
  7. <div v-if="msg.type === 'user'" class="bubble">{{ msg.text }}</div>
  8. <div v-if="msg.type === 'system'" class="bubble answer">
  9. <div class="think" v-if="!msg.text.name">
  10. 思考中<el-icon><ArrowDown /></el-icon>
  11. </div>
  12. <div class="header" v-html="msg.text.header"></div>
  13. <div class="divider"></div>
  14. <div class="content" v-html="msg.text.content"></div>
  15. <div class="table-wrap" v-if="msg.text.name === '种植面积'">
  16. <el-table :data="tableData" border style="width: 100%">
  17. <el-table-column prop="city" label="市" width="100" />
  18. <el-table-column prop="area" label="2024年水稻种植面积" width="320">
  19. <template #default="scope">
  20. {{ scope.row.area }}亩
  21. </template>
  22. </el-table-column>
  23. </el-table>
  24. </div>
  25. </div>
  26. <div v-if="msg.type === 'real'" class="bubble answer">
  27. <div class="think" v-if="!msg.text.name">
  28. 思考中<el-icon><ArrowDown /></el-icon>
  29. </div>
  30. <div class="header" v-html="deepSeekAsk.markdownToHtml(msg.text.header)"></div>
  31. <div class="divider"></div>
  32. <div class="content" v-html="deepSeekAsk.markdownToHtml(msg.text.content)"></div>
  33. <div class="table-wrap" v-if="msg.text.name === '种植面积'">
  34. <el-table :data="tableData" border style="width: 100%">
  35. <el-table-column prop="city" label="市" width="100" />
  36. <el-table-column prop="area" label="2024年水稻种植面积" width="320">
  37. <template #default="scope">
  38. {{ scope.row.area }}亩
  39. </template>
  40. </el-table-column>
  41. </el-table>
  42. </div>
  43. </div>
  44. <div v-if="msg.type === 'auto'" class="system auto">
  45. <div class="bubble">
  46. {{ msg.text.header }}
  47. <div>
  48. <div class="ask-title">你可以试着问我</div>
  49. <div class="ask-list">
  50. <li
  51. class="ask-item cursor-pointer"
  52. @click="askText('2024年广东省各市的水稻种植面积')"
  53. >
  54. 2024年广东省各市的水稻种植面积
  55. </li>
  56. <li
  57. class="ask-item cursor-pointer"
  58. @click="askText('2024年广东省各市的水稻种植面积排行')"
  59. >
  60. 2024年广东省各市的水稻种植面积排行
  61. </li>
  62. </div>
  63. </div>
  64. </div>
  65. </div>
  66. <!-- 猜你想问 -->
  67. <div v-if="msg.type === 'ask'" class="system ask">
  68. <div class="bubble">
  69. <div class="ask-title">你可以试着问我</div>
  70. <div class="ask-list">
  71. <div class="to-map" v-for="(ask, askI) in msg.text.content" :key="askI" @click="toMapLayer(ask)">
  72. <li>
  73. {{ ask }}
  74. </li>
  75. <div class="go-icon"><el-icon><ArrowRight /></el-icon></div>
  76. </div>
  77. </div>
  78. </div>
  79. </div>
  80. </div>
  81. </div>
  82. <div class="bottom-send">
  83. <div class="input-box">
  84. <el-input
  85. v-model="userInput"
  86. :autosize="{ minRows: 2, maxRows: 8 }"
  87. type="textarea"
  88. placeholder="给 飞鸟种植大脑 提问吧~"
  89. @keyup.enter="sendMessage"
  90. />
  91. <div class="bottom-group">
  92. <div class="btn-l">
  93. <div class="l-item">
  94. <img src="@/assets/images/warningHome/chat/think.png" />
  95. 深度思考(R1)
  96. </div>
  97. <div class="l-item">
  98. <img src="@/assets/images/warningHome/chat/net.png" />
  99. 联网搜索
  100. </div>
  101. </div>
  102. <div class="btn-r">
  103. <img class="file-icon" src="@/assets/images/warningHome/chat/file.png" />
  104. <img class="send-icon" @click="sendMessage" src="@/assets/images/warningHome/chat/send.png" alt="send">
  105. </div>
  106. </div>
  107. </div>
  108. </div>
  109. </div>
  110. </div>
  111. </template>
  112. <script setup>
  113. import { nextTick, onMounted, ref } from "vue";
  114. import { useRoute, useRouter } from "vue-router";
  115. import chat from "./chat.json";
  116. import DeepSeekAsk from "./deepSeekAsk";
  117. import { useStore } from "vuex";
  118. import eventBus from "@/api/eventBus";
  119. const store = useStore();
  120. let userId = 12321;
  121. let deepSeekAsk = new DeepSeekAsk(userId);
  122. const router = useRouter();
  123. const userInput = ref("");
  124. const messages = ref([]);
  125. const isProcessing = ref(false); // 控制是否在处理消息
  126. const tableData = ref([
  127. { city: '广州市', area: 10000 },
  128. { city: '佛山市', area: 900 },
  129. { city: '惠州市', area: 856 },
  130. { city: '茂名市', area: 789 },
  131. ])
  132. const steps = [
  133. { type: "auto", text: { header: "您好,飞鸟智慧种植大脑是您的私人管家" } },
  134. {
  135. type: "ask",
  136. text: {
  137. askHeader: "猜您想问",
  138. askContent: ["2024年广东省各市的水稻种植面积", "2024年广东省各市的水稻种植面积变化"],
  139. },
  140. },
  141. {
  142. type: "ask",
  143. text: {
  144. askHeader: "猜您想问",
  145. askContent: ["荔枝出现花带叶怎么办?", "荔枝抽梢具体做什么农事呢?"],
  146. },
  147. },
  148. ];
  149. const stepIndex = ref(0);
  150. const triggerNextStep = () => {
  151. if (stepIndex.value < steps.length) {
  152. addSystemReply(
  153. steps[stepIndex.value].type,
  154. steps[stepIndex.value].text,
  155. () => {
  156. console.log("stepIndex.value === 76574", stepIndex.value);
  157. isProcessing.value = false;
  158. }
  159. );
  160. stepIndex.value++;
  161. saveState();
  162. scrollToBottom();
  163. }
  164. };
  165. const loadState = () => {
  166. const storedMessages = localStorage.getItem(STORAGE_KEY);
  167. const storedDate = localStorage.getItem(DATE_KEY);
  168. const storedStep = localStorage.getItem(STEP_KEY);
  169. const today = new Date().toISOString().split("T")[0];
  170. if (storedDate === today && storedMessages) {
  171. messages.value = JSON.parse(storedMessages);
  172. stepIndex.value = storedStep ? parseInt(storedStep, 10) : 0;
  173. nextTick(() => scrollToBottom());
  174. } else {
  175. localStorage.removeItem(STORAGE_KEY);
  176. localStorage.removeItem(DATE_KEY);
  177. localStorage.removeItem(STEP_KEY);
  178. stepIndex.value = 0;
  179. }
  180. };
  181. const toMapLayer = (name) => {
  182. eventBus.emit("chat:showMapLayer", name)
  183. }
  184. const askText = (val) => {
  185. userInput.value = val;
  186. sendMessage();
  187. };
  188. const sendMessage = () => {
  189. if (!userInput.value.trim()) return;
  190. // 先保存用户输入的内容
  191. const userText = userInput.value;
  192. // 添加用户消息
  193. messages.value.push({ text: userInput.value, type: "user" });
  194. saveMessages();
  195. scrollToBottom();
  196. // 模拟系统回复
  197. // setTimeout(() => {
  198. // console.log("userInput", userText);
  199. // messages.value.push({ text: "系统回复: " + userText, type: "system" });
  200. // }, 500);
  201. // 模拟 AI 逐字回复
  202. let isSearch = true;
  203. console.log("userText", userText);
  204. chat.map((item) => {
  205. if (userText.indexOf(item.name) !== -1) {
  206. addSystemReply('system', {header: item.header, content: item.content, name: item.name}, () => {
  207. console.log("sendMessage eeeeee",);
  208. setTimeout(triggerNextStep, 2000);
  209. });
  210. isSearch = false;
  211. }
  212. });
  213. if (isSearch) {
  214. messages.value.push({ type: "real", text: { header: "", content: "" } });
  215. deepSeekAsk.ask(userText, function (code) {
  216. if (code == 0) {
  217. let intervalId = setInterval(() => {
  218. if (deepSeekAsk.end) {
  219. clearInterval(intervalId);
  220. }
  221. for (let content of deepSeekAsk.contents) {
  222. console.log(content.content);
  223. messages.value[messages.value.length - 1].text.content = content.content;
  224. messages.value[messages.value.length - 1].text.header = content.header;
  225. scrollToBottom();
  226. saveMessages();
  227. }
  228. }, 1000);
  229. }
  230. });
  231. }
  232. // 清空输入框
  233. userInput.value = "";
  234. };
  235. const loadEnd = ref(false);
  236. // **逐字显示系统回复(header 和 content)**
  237. const addSystemReply = (type = "system", textObject, callback) => {
  238. isProcessing.value = true
  239. let currentHeader = "";
  240. let currentContent = "";
  241. console.log("stepIndex.value", type);
  242. if (type === "ask") {
  243. messages.value.push({ text: { header: textObject.askHeader, content: textObject.askContent }, type });
  244. } else {
  245. messages.value.push({ text: { header: currentHeader, content: currentContent, name: textObject.name }, type });
  246. }
  247. saveMessages();
  248. scrollToBottom();
  249. const content = textObject.content
  250. const header = textObject.header
  251. // **逐字显示 content**
  252. const showContent = () => {
  253. let contentIndex = 0;
  254. if (content && content.length > 0) {
  255. const contentInterval = setInterval(() => {
  256. if (contentIndex < content.length) {
  257. currentContent += content[contentIndex];
  258. if (type !== "ask") {
  259. messages.value[messages.value.length - 1].text.content = currentContent;
  260. }
  261. saveMessages();
  262. scrollToBottom();
  263. contentIndex++;
  264. } else {
  265. clearInterval(contentInterval);
  266. console.log("ddddddddone", stepIndex.value);
  267. // if (stepIndex.value === 2 || stepIndex.value === 6) {
  268. // nextTick(() => {
  269. // setTimeout(triggerNextStep, 1000);
  270. // });
  271. // }
  272. callback && callback(); // 回复完成后解锁输入
  273. }
  274. }, 50);
  275. } else {
  276. callback && callback(); // 如果 content 为空,直接解锁输入
  277. }
  278. };
  279. let headerIndex = 0;
  280. if (header && header.length > 0) {
  281. const headerInterval = setInterval(() => {
  282. if (headerIndex < header.length) {
  283. currentHeader += header[headerIndex];
  284. messages.value[messages.value.length - 1].text.header = currentHeader;
  285. saveMessages();
  286. scrollToBottom();
  287. headerIndex++;
  288. } else {
  289. clearInterval(headerInterval);
  290. showContent();
  291. }
  292. }, 5); // 50ms 逐字显示 header
  293. } else {
  294. showContent();
  295. }
  296. };
  297. const saveState = () => {
  298. const today = new Date().toISOString().split("T")[0];
  299. localStorage.setItem(STORAGE_KEY, JSON.stringify(messages.value));
  300. localStorage.setItem(DATE_KEY, today);
  301. localStorage.setItem(STEP_KEY, stepIndex.value);
  302. };
  303. // **存入缓存**
  304. const STORAGE_KEY = "chatMessages";
  305. const DATE_KEY = "chatDate";
  306. const STEP_KEY = "chatStep";
  307. const saveMessages = () => {
  308. const today = new Date().toISOString().split("T")[0]; // 仅保存 "YYYY-MM-DD"
  309. localStorage.setItem(STORAGE_KEY, JSON.stringify(messages.value));
  310. localStorage.setItem(DATE_KEY, today);
  311. };
  312. // **恢复缓存**
  313. const loadMessages = () => {
  314. const storedMessages = localStorage.getItem(STORAGE_KEY);
  315. const storedDate = localStorage.getItem(DATE_KEY);
  316. const today = new Date().toISOString().split("T")[0]; // 获取今天的日期
  317. if (storedDate === today && storedMessages) {
  318. messages.value = JSON.parse(storedMessages);
  319. } else {
  320. // 清除过期数据
  321. localStorage.removeItem(STORAGE_KEY);
  322. localStorage.removeItem(DATE_KEY);
  323. }
  324. };
  325. // 路由跳转
  326. const toOtherPage = (val) => {
  327. router.push(val);
  328. };
  329. const chatBox = ref();
  330. onMounted(() => {
  331. // loadMessages()
  332. // scrollToBottom()
  333. loadState();
  334. nextTick(() => {
  335. setTimeout(triggerNextStep, 1000);
  336. });
  337. // setTimeout(() => {
  338. // chatBox.value.scrollTo({top: chatBox.value.scrollHeight, behavior: 'smooth' }); // 滚动到页面顶部
  339. // }, 200)
  340. });
  341. // **滚动到底部**
  342. const scrollToBottom = () => {
  343. nextTick(() => {
  344. if (chatBox.value) {
  345. chatBox.value.scrollTo({ top: chatBox.value.scrollHeight, behavior: "smooth" }); // 滚动到页面底部
  346. }
  347. });
  348. };
  349. </script>
  350. <style lang="scss" scoped>
  351. .chat-page {
  352. height: 100%;
  353. padding-top: 34px;
  354. box-sizing: border-box;
  355. .chat-wrap {
  356. height: 100%;
  357. background: #232323;
  358. border-radius: 8px;
  359. border: 1px solid #777777;
  360. display: flex;
  361. flex-direction: column;
  362. justify-content: space-between;
  363. box-sizing: border-box;
  364. .chat-title {
  365. border-radius: 8px 8px 0 0;
  366. background: #2F2F2F;
  367. text-align: center;
  368. padding: 10px 0;
  369. }
  370. }
  371. }
  372. .chat-container {
  373. display: flex;
  374. flex-direction: column;
  375. height: 100vh;
  376. justify-content: space-between;
  377. padding: 10px;
  378. box-sizing: border-box;
  379. background: #fff;
  380. }
  381. .chat-box {
  382. flex: 1;
  383. overflow-y: auto;
  384. padding: 16px;
  385. }
  386. .message {
  387. display: flex;
  388. margin: 10px 0;
  389. .avatar {
  390. width: 28px;
  391. height: 28px;
  392. margin-right: 8px;
  393. }
  394. }
  395. .table-wrap {
  396. ::v-deep {
  397. .el-table .el-table__header th.el-table__cell {
  398. background: #3B3B3B !important;
  399. border-bottom-color: #555555;
  400. border-right-color: #555555;
  401. }
  402. .el-table {
  403. color: #fff;
  404. }
  405. .el-table tr {
  406. background: #2F2F2F;
  407. pointer-events: none;
  408. }
  409. .el-table thead {
  410. color: #999999;
  411. }
  412. .el-table td.el-table__cell {
  413. border-color: #555555;
  414. }
  415. .el-table--border .el-table__inner-wrapper:after, .el-table--border:after, .el-table--border:before, .el-table__inner-wrapper:before,
  416. .el-table__border-bottom-patch, .el-table__border-left-patch{
  417. background-color: #555555;
  418. }
  419. }
  420. }
  421. .ask {
  422. width: 100%;
  423. display: flex;
  424. .bubble {
  425. width: 100%;
  426. }
  427. }
  428. .ask-title {
  429. color: #999999;
  430. padding-bottom: 10px;
  431. }
  432. .ask-list {
  433. color: #FFD489;
  434. }
  435. .to-map {
  436. display: flex;
  437. justify-content: space-between;
  438. background: #3C3C3C;
  439. padding: 6px 8px;
  440. border-radius: 6px;
  441. cursor: pointer;
  442. }
  443. .to-map + .to-map {
  444. margin-top: 8px;
  445. }
  446. .ask-item + .ask-item {
  447. padding-top: 2px;
  448. }
  449. .route-wrap {
  450. display: flex;
  451. .route-img {
  452. width: 100%;
  453. padding-top: 8px;
  454. }
  455. }
  456. .user {
  457. justify-content: flex-end;
  458. }
  459. .system {
  460. justify-content: flex-start;
  461. }
  462. .link {
  463. display: flex;
  464. .header-link {
  465. color: #FFD489;
  466. display: flex;
  467. align-items: baseline;
  468. text-decoration: underline;
  469. .icon {
  470. margin-right: 5px;
  471. }
  472. }
  473. }
  474. .bubble {
  475. padding: 16px 12px;
  476. border-radius: 8px;
  477. // max-width: 60%;
  478. background: rgba(255, 212, 137, 0.1);
  479. color: #FFD489;
  480. border-radius: 16px 2px 16px 16px;
  481. line-height: 24px;
  482. font-size: 14px;
  483. .header {
  484. color: #999999;
  485. // padding-bottom: 8px;
  486. }
  487. .content {
  488. color: #ffffff;
  489. }
  490. .divider {
  491. width: 100%;
  492. height: 1px;
  493. background: rgba(153, 153, 153, 0.2);
  494. margin: 12px 0;
  495. }
  496. .think {
  497. color: #999999;
  498. }
  499. .expert-img {
  500. margin-top: 12px;
  501. width: 100%;
  502. }
  503. }
  504. .system .bubble {
  505. background: #2F2F2F;
  506. color: #fff;
  507. border-radius: 2px 16px 16px 16px;
  508. }
  509. .uploader {
  510. width: 100%;
  511. display: flex;
  512. justify-content: center;
  513. img {
  514. width: 174px;
  515. height: 55px;
  516. }
  517. }
  518. .bottom-send {
  519. padding: 6px 16px 16px;
  520. }
  521. .input-box {
  522. display: flex;
  523. flex-direction: column;
  524. padding: 10px;
  525. background: #2F2F2F;
  526. border-radius: 16px;
  527. /* border-top: 1px solid #ddd; */
  528. ::v-deep {
  529. .el-textarea__inner {
  530. background: transparent;
  531. box-shadow: none;
  532. color: #fff;
  533. }
  534. }
  535. .bottom-group {
  536. display: flex;
  537. justify-content: space-between;
  538. align-items: center;
  539. padding-top: 6px;
  540. .btn-l {
  541. display: flex;
  542. .l-item {
  543. display: flex;
  544. align-items: center;
  545. border-radius: 20px;
  546. border: 1px solid #555555;
  547. padding: 6px 8px;
  548. }
  549. .l-item + .l-item {
  550. margin-left: 12px;
  551. }
  552. img {
  553. width: 16px;
  554. padding-right: 2px;
  555. }
  556. }
  557. .file-icon {
  558. width: 16px;
  559. }
  560. .send-icon {
  561. margin-left: 8px;
  562. width: 28px;
  563. }
  564. }
  565. }
  566. input {
  567. flex: 1;
  568. padding: 10px;
  569. border: 1px solid #ddd;
  570. border-radius: 4px;
  571. }
  572. button {
  573. margin-left: 10px;
  574. padding: 10px 15px;
  575. background: #007bff;
  576. color: white;
  577. border: none;
  578. cursor: pointer;
  579. }
  580. </style>