index.vue 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800
  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">
  10. 思考中<el-icon><ArrowDown /></el-icon>
  11. </div>
  12. <div class="header" v-html="msg.text.header" :class="{ 'main-text': !msg.text.content }"></div>
  13. <div class="divider" v-if="msg.text.content"></div>
  14. <div class="content" v-html="msg.text.content"></div>
  15. <div
  16. class="table-wrap"
  17. v-if="msg.text.name === '2024年广东省各市的稻谷播种面积?' && msg.loadEnd"
  18. >
  19. <el-table :data="tableData" border style="width: 100%">
  20. <el-table-column prop="city" label="市" width="100" />
  21. <el-table-column prop="area" label="2024年稻谷播种面积">
  22. <template #default="scope"> {{ scope.row.area }}亩 </template>
  23. </el-table-column>
  24. </el-table>
  25. </div>
  26. <div
  27. class="table-wrap"
  28. v-if="msg.text.name === '花期统防统治报表,有没有与农资对接' && msg.loadEnd"
  29. >
  30. <el-table :data="tableData2" border style="width: 100%">
  31. <el-table-column prop="name" label="企业名称" />
  32. <el-table-column prop="area" label="覆盖范围" show-overflow-tooltip />
  33. <el-table-column prop="step" label="完成进度" show-overflow-tooltip />
  34. <el-table-column prop="rate" label="系统农事成效评分" show-overflow-tooltip />
  35. <el-table-column prop="farmRate" label="农户评价" show-overflow-tooltip />
  36. </el-table>
  37. </div>
  38. <div class="table-wrap" v-if="msg.text.name === '当前高州的作物分布和物候期?' && msg.loadEnd">
  39. <el-table :data="tableData3" border style="width: 100%">
  40. <el-table-column prop="name" label="农作物类型" />
  41. <el-table-column prop="area" label="播种面积(亩)">
  42. <template #default="scope"> {{ scope.row.area }}亩 </template>
  43. </el-table-column>
  44. </el-table>
  45. </div>
  46. </div>
  47. <div v-if="msg.type === 'real'" class="bubble answer">
  48. <div class="think" v-if="!msg.text.name">
  49. 思考中<el-icon><ArrowDown /></el-icon>
  50. </div>
  51. <div class="header" v-html="deepSeekAsk.markdownToHtml(msg.text.header)"></div>
  52. <div class="divider"></div>
  53. <div class="content" v-html="deepSeekAsk.markdownToHtml(msg.text.content)"></div>
  54. </div>
  55. <div v-if="msg.type === 'auto'" class="system auto">
  56. <div class="bubble">
  57. {{ msg.text.header }}
  58. <div>
  59. <div class="ask-title">你可以试着问我</div>
  60. <div class="ask-list">
  61. <li
  62. class="ask-item cursor-pointer"
  63. @click="toMapLayer(askQues.text.askContent[0], askQues.text)"
  64. >
  65. 当前区域八角有什么生长风险?
  66. </li>
  67. </div>
  68. </div>
  69. </div>
  70. </div>
  71. <!-- 猜你想问 -->
  72. <div v-if="msg.type === 'ask'" class="system ask">
  73. <div class="bubble">
  74. <div class="ask-title">你可以试着问我</div>
  75. <div class="ask-list">
  76. <div
  77. class="to-map"
  78. v-for="(ask, askI) in msg.text.content"
  79. :key="askI"
  80. @click="toMapLayer(ask, msg.text)"
  81. >
  82. <li>
  83. {{ ask }}
  84. </li>
  85. <div class="go-icon">
  86. <el-icon><ArrowRight /></el-icon>
  87. </div>
  88. </div>
  89. </div>
  90. </div>
  91. </div>
  92. </div>
  93. </div>
  94. <div class="bottom-send">
  95. <div class="input-box">
  96. <el-input
  97. v-model="userInput"
  98. :autosize="{ minRows: 2, maxRows: 8 }"
  99. type="textarea"
  100. placeholder="给 飞鸟种植大脑 提问吧~"
  101. @keyup.enter="sendMessage"
  102. />
  103. <div class="bottom-group">
  104. <div class="btn-l">
  105. <div class="l-item">
  106. <img src="@/assets/images/warningHome/chat/think.png" />
  107. 深度思考(R1)
  108. </div>
  109. <div class="l-item">
  110. <img src="@/assets/images/warningHome/chat/net.png" />
  111. 联网搜索
  112. </div>
  113. </div>
  114. <div class="btn-r">
  115. <img class="file-icon" src="@/assets/images/warningHome/chat/file.png" />
  116. <img
  117. class="send-icon"
  118. @click="sendMessage"
  119. src="@/assets/images/warningHome/chat/send.png"
  120. alt="send"
  121. />
  122. </div>
  123. </div>
  124. </div>
  125. </div>
  126. </div>
  127. </div>
  128. </template>
  129. <script setup>
  130. import { nextTick, onMounted, ref } from "vue";
  131. import { useRoute, useRouter } from "vue-router";
  132. import chat from "./chat.json";
  133. import DeepSeekAsk from "./deepSeekAsk";
  134. import { useStore } from "vuex";
  135. import eventBus from "@/api/eventBus";
  136. const store = useStore();
  137. let userId = 12321;
  138. let deepSeekAsk = new DeepSeekAsk(userId);
  139. const router = useRouter();
  140. const userInput = ref("");
  141. const messages = ref([]);
  142. const isProcessing = ref(false); // 控制是否在处理消息
  143. const tableData = ref([
  144. { city: "广州", area: 360091 },
  145. { city: "深圳", area: 14726 },
  146. { city: "珠海", area: 66670 },
  147. { city: "汕头", area: 690567 },
  148. { city: "佛山", area: 104095 },
  149. { city: "韶关", area: 1534904 },
  150. { city: "河源", area: 1836469 },
  151. { city: "梅州", area: 2428360 },
  152. { city: "惠州", area: 1275493 },
  153. { city: "汕尾", area: 1033786 },
  154. { city: "东莞", area: 24106 },
  155. { city: "中山", area: 37709 },
  156. { city: "江门", area: 2523802 },
  157. { city: "阳江", area: 1604316 },
  158. { city: "湛江", area: 3357751 },
  159. { city: "茂名", area: 3151002 },
  160. { city: "肇庆", area: 2521029 },
  161. { city: "清远", area: 1832838 },
  162. { city: "潮州", area: 481617 },
  163. { city: "揭阳", area: 1217258 },
  164. { city: "云浮", area: 1317616 },
  165. ]);
  166. const tableData2 = ref([
  167. {
  168. name: "天*股份",
  169. area: "广东省及周边省份,100家配送中心",
  170. step: "此次花期统防统治接收任务28个,完成进度64%。",
  171. rate: "90分",
  172. farmRate: "89分",
  173. },
  174. {
  175. name: "大*农业",
  176. area: "广东省及周边省份",
  177. step: "此次花期统防统治接收任务17个,完成进度80%。",
  178. rate: "95分",
  179. farmRate: "92分",
  180. },
  181. {
  182. name: "美*公司",
  183. area: "广东省及周边省份",
  184. step: "此次花期统防统治接收任务13个,完成进度59%。",
  185. rate: "89分",
  186. farmRate: "90分",
  187. },
  188. {
  189. name: "盛*农业",
  190. area: "广东省、广西省,50家配送中心",
  191. step: "此次花期统防统治接收任务25个,完成进度72%",
  192. rate: "91分",
  193. farmRate: "88分",
  194. },
  195. {
  196. name: "宝*公司",
  197. area: "广东省及周边省份",
  198. step: "此次花期统防统治接收任务20个,完成进度67%",
  199. rate: "90分",
  200. farmRate: "92分",
  201. },
  202. {
  203. name: "优*农业",
  204. area: "广东省、福建省,30家配送中心",
  205. step: "此次花期统防统治接收任务15个,完成进度85%",
  206. rate: "94分",
  207. farmRate: "93分",
  208. },
  209. {
  210. name: "新*科技",
  211. area: "广东省及周边省份",
  212. step: "此次花期统防统治接收任务18个,完成进度70%",
  213. rate: "87分",
  214. farmRate: "86分",
  215. },
  216. {
  217. name: "海*农资",
  218. area: "广东省及周边省份",
  219. step: "此次花期统防统治接收任务21个,完成进度91%",
  220. rate: "96分",
  221. farmRate: "94分",
  222. },
  223. {
  224. name: "丰*公司",
  225. area: "广东省、海南省",
  226. step: "此次花期统防统治接收任务20个,完成进度68%",
  227. rate: "88分",
  228. farmRate: "85分",
  229. },
  230. {
  231. name: "绿*农业",
  232. area: "广东省、广西省,70家配送中心",
  233. step: "此次花期统防统治接收任务30个,完成进度75%",
  234. rate: "91分",
  235. farmRate: "93分",
  236. },
  237. {
  238. name: "亮*集团",
  239. area: "广东省、云南省 ",
  240. step: "此次花期统防统治接收任务16个,完成进度70%",
  241. rate: "91分",
  242. farmRate: "89分",
  243. },
  244. ]);
  245. const tableData3 = ref([
  246. { name: "水稻", area: "839884" },
  247. { name: "玉米", area: "16079" },
  248. { name: "番薯", area: "39467" },
  249. { name: "花生", area: "121093" },
  250. { name: "中草药材", area: "72357" },
  251. { name: "蔬菜", area: "407695" },
  252. { name: "荔枝", area: "590322" },
  253. { name: "龙眼", area: "324390" },
  254. { name: "香蕉", area: "258688" },
  255. { name: "柚子", area: "14527" },
  256. { name: "黄皮", area: "48020" },
  257. ]);
  258. const askQues = {
  259. type: "ask",
  260. text: {
  261. mapName: "分散种植",
  262. askHeader: "当前区域八角有什么生长风险?",
  263. askContent: ["当前区域八角有什么生长风险?"],
  264. },
  265. };
  266. const steps = [
  267. { type: "auto", text: { header: "您好,飞鸟智慧种植大脑是您的私人管家" } },
  268. // {
  269. // type: "ask",
  270. // text: {
  271. // mapName: "分散种植",
  272. // askHeader: "当前区域八角有什么生长风险?",
  273. // askContent: ["当前区域八角有什么生长风险?",],
  274. // },
  275. // },
  276. {
  277. type: "ask",
  278. text: {
  279. mapName: "病虫态势",
  280. askHeader: "当前八角区域有哪些地块病虫害风险较高",
  281. askContent: ["当前八角区域有哪些地块病虫害风险较高"],
  282. },
  283. },
  284. {
  285. type: "ask",
  286. text: {
  287. mapName: "病虫态势",
  288. askHeader: "当前区域哪些有荔枝地块病虫害风险较高",
  289. askContent: ["当前区域哪些有荔枝地块病虫害风险较高"],
  290. },
  291. },
  292. {
  293. type: "ask",
  294. text: {
  295. mapName: "农情需求",
  296. askHeader: "现在哪些地方有农情需求",
  297. askContent: ["现在哪些地方有农情需求"],
  298. },
  299. },
  300. {
  301. type: "ask",
  302. text: {
  303. mapName: "植保机",
  304. askHeader: "哪些地方有植保机",
  305. askContent: ["哪些地方有植保机"],
  306. },
  307. },
  308. {
  309. type: "ask",
  310. text: {
  311. mapName: "",
  312. askHeader: "花期统防统治报表,有没有与农资对接",
  313. askContent: ["花期统防统治报表,有没有与农资对接"],
  314. },
  315. },
  316. ];
  317. const stepIndex = ref(0);
  318. const triggerNextStep = () => {
  319. if (stepIndex.value < steps.length) {
  320. addSystemReply(steps[stepIndex.value].type, steps[stepIndex.value].text, () => {
  321. isProcessing.value = false;
  322. // steps[stepIndex.value].loadEnd = true
  323. });
  324. stepIndex.value++;
  325. saveState();
  326. scrollToBottom();
  327. }
  328. };
  329. const loadState = () => {
  330. const storedMessages = localStorage.getItem(STORAGE_KEY);
  331. const storedDate = localStorage.getItem(DATE_KEY);
  332. const storedStep = localStorage.getItem(STEP_KEY);
  333. const today = new Date().toISOString().split("T")[0];
  334. if (storedDate === today && storedMessages) {
  335. messages.value = JSON.parse(storedMessages);
  336. stepIndex.value = storedStep ? parseInt(storedStep, 10) : 0;
  337. nextTick(() => scrollToBottom());
  338. } else {
  339. localStorage.removeItem(STORAGE_KEY);
  340. localStorage.removeItem(DATE_KEY);
  341. localStorage.removeItem(STEP_KEY);
  342. stepIndex.value = 0;
  343. }
  344. };
  345. const toMapLayer = (name, text) => {
  346. askText(name);
  347. eventBus.emit("chat:showMapLayer", { name, mapName: text.mapName });
  348. // addSystemReply('system', {header: name, content: '', name}, () => {
  349. // steps[stepIndex.value].loadEnd = true
  350. // });
  351. };
  352. const askText = (val) => {
  353. userInput.value = val;
  354. sendMessage();
  355. };
  356. const sendMessage = () => {
  357. if (!userInput.value.trim()) return;
  358. // 先保存用户输入的内容
  359. const userText = userInput.value;
  360. // 添加用户消息
  361. messages.value.push({ text: userInput.value, type: "user" });
  362. saveMessages();
  363. scrollToBottom();
  364. // 模拟系统回复
  365. // setTimeout(() => {
  366. // messages.value.push({ text: "系统回复: " + userText, type: "system" });
  367. // }, 500);
  368. // 模拟 AI 逐字回复
  369. let isSearch = true;
  370. console.log("userText", userText);
  371. chat.map((item) => {
  372. if (userText.indexOf(item.name) !== -1) {
  373. addSystemReply("system", { header: item.header, content: item.content, name: item.name }, () => {
  374. // steps[stepIndex.value].loadEnd = true
  375. messages.value[messages.value.length - 1].loadEnd = true;
  376. setTimeout(triggerNextStep, 2000);
  377. });
  378. isSearch = false;
  379. }
  380. });
  381. if (isSearch) {
  382. messages.value.push({ type: "real", text: { header: "", content: "" } });
  383. deepSeekAsk.ask(userText, function (code) {
  384. if (code == 0) {
  385. let intervalId = setInterval(() => {
  386. if (deepSeekAsk.end) {
  387. clearInterval(intervalId);
  388. }
  389. for (let content of deepSeekAsk.contents) {
  390. console.log(content.content);
  391. messages.value[messages.value.length - 1].text.content = content.content;
  392. messages.value[messages.value.length - 1].text.header = content.header;
  393. scrollToBottom();
  394. saveMessages();
  395. }
  396. }, 1000);
  397. }
  398. });
  399. }
  400. // 清空输入框
  401. userInput.value = "";
  402. };
  403. // **逐字显示系统回复(header 和 content)**
  404. const addSystemReply = (type = "system", textObject, callback) => {
  405. isProcessing.value = true;
  406. let currentHeader = "";
  407. let currentContent = "";
  408. if (type === "ask") {
  409. messages.value.push({
  410. text: { header: textObject.askHeader, content: textObject.askContent, mapName: textObject.mapName },
  411. type,
  412. });
  413. } else {
  414. messages.value.push({ text: { header: currentHeader, content: currentContent, name: textObject.name }, type });
  415. }
  416. saveMessages();
  417. scrollToBottom();
  418. const content = textObject.content;
  419. const header = textObject.header;
  420. // **逐字显示 content**
  421. const showContent = () => {
  422. let contentIndex = 0;
  423. if (content && content.length > 0) {
  424. const contentInterval = setInterval(() => {
  425. if (contentIndex < content.length) {
  426. currentContent += content[contentIndex];
  427. if (type !== "ask") {
  428. messages.value[messages.value.length - 1].text.content = currentContent;
  429. }
  430. saveMessages();
  431. scrollToBottom();
  432. contentIndex++;
  433. } else {
  434. clearInterval(contentInterval);
  435. messages.value[messages.value.length - 1].loadEnd = true;
  436. scrollToBottom();
  437. callback && callback(); // 回复完成后解锁输入
  438. }
  439. }, 100);
  440. } else {
  441. callback && callback(); // 如果 content 为空,直接解锁输入
  442. }
  443. };
  444. let headerIndex = 0;
  445. if (header && header.length > 0) {
  446. const headerInterval = setInterval(() => {
  447. if (headerIndex < header.length) {
  448. currentHeader += header[headerIndex];
  449. messages.value[messages.value.length - 1].text.header = currentHeader;
  450. saveMessages();
  451. scrollToBottom();
  452. headerIndex++;
  453. } else {
  454. clearInterval(headerInterval);
  455. showContent();
  456. }
  457. }, 100); // 50ms 逐字显示 header
  458. } else {
  459. showContent();
  460. }
  461. };
  462. const saveState = () => {
  463. const today = new Date().toISOString().split("T")[0];
  464. localStorage.setItem(STORAGE_KEY, JSON.stringify(messages.value));
  465. localStorage.setItem(DATE_KEY, today);
  466. localStorage.setItem(STEP_KEY, stepIndex.value);
  467. };
  468. // **存入缓存**
  469. const STORAGE_KEY = "chatMessages";
  470. const DATE_KEY = "chatDate";
  471. const STEP_KEY = "chatStep";
  472. const saveMessages = () => {
  473. const today = new Date().toISOString().split("T")[0]; // 仅保存 "YYYY-MM-DD"
  474. localStorage.setItem(STORAGE_KEY, JSON.stringify(messages.value));
  475. localStorage.setItem(DATE_KEY, today);
  476. };
  477. // **恢复缓存**
  478. const loadMessages = () => {
  479. const storedMessages = localStorage.getItem(STORAGE_KEY);
  480. const storedDate = localStorage.getItem(DATE_KEY);
  481. const today = new Date().toISOString().split("T")[0]; // 获取今天的日期
  482. if (storedDate === today && storedMessages) {
  483. messages.value = JSON.parse(storedMessages);
  484. } else {
  485. // 清除过期数据
  486. localStorage.removeItem(STORAGE_KEY);
  487. localStorage.removeItem(DATE_KEY);
  488. }
  489. };
  490. // 路由跳转
  491. const toOtherPage = (val) => {
  492. router.push(val);
  493. };
  494. const chatBox = ref();
  495. onMounted(() => {
  496. // loadMessages()
  497. // scrollToBottom()
  498. loadState();
  499. nextTick(() => {
  500. setTimeout(triggerNextStep, 1000);
  501. });
  502. // setTimeout(() => {
  503. // chatBox.value.scrollTo({top: chatBox.value.scrollHeight, behavior: 'smooth' }); // 滚动到页面顶部
  504. // }, 200)
  505. });
  506. eventBus.on("chat:hideMapLayer", () => {
  507. // setTimeout(triggerNextStep, 2000);
  508. });
  509. // **滚动到底部**
  510. const scrollToBottom = () => {
  511. nextTick(() => {
  512. if (chatBox.value) {
  513. chatBox.value.scrollTo({ top: chatBox.value.scrollHeight, behavior: "smooth" }); // 滚动到页面底部
  514. }
  515. });
  516. };
  517. </script>
  518. <style lang="scss" scoped>
  519. .chat-page {
  520. height: 100%;
  521. padding-top: 34px;
  522. box-sizing: border-box;
  523. .chat-wrap {
  524. height: 100%;
  525. background: #232323;
  526. border-radius: 8px;
  527. border: 1px solid #777777;
  528. display: flex;
  529. flex-direction: column;
  530. justify-content: space-between;
  531. box-sizing: border-box;
  532. .chat-title {
  533. border-radius: 8px 8px 0 0;
  534. background: #2f2f2f;
  535. text-align: center;
  536. padding: 10px 0;
  537. }
  538. }
  539. }
  540. .chat-container {
  541. display: flex;
  542. flex-direction: column;
  543. height: 100vh;
  544. justify-content: space-between;
  545. padding: 10px;
  546. box-sizing: border-box;
  547. background: #fff;
  548. }
  549. .chat-box {
  550. flex: 1;
  551. overflow-y: auto;
  552. padding: 16px;
  553. }
  554. .message {
  555. display: flex;
  556. margin: 10px 0;
  557. .avatar {
  558. width: 28px;
  559. height: 28px;
  560. margin-right: 8px;
  561. }
  562. }
  563. .table-wrap {
  564. padding-top: 12px;
  565. width: 100%;
  566. overflow: auto;
  567. ::v-deep {
  568. .el-table .el-table__header th.el-table__cell {
  569. background: #3b3b3b !important;
  570. border-bottom-color: #555555;
  571. border-right-color: #555555;
  572. }
  573. .el-table .el-table__body tr:hover > td {
  574. background-color: transparent !important;
  575. }
  576. .el-table {
  577. color: #fff;
  578. }
  579. .el-table tr {
  580. background: #2f2f2f;
  581. // pointer-events: none;
  582. }
  583. .el-table thead {
  584. color: #999999;
  585. }
  586. .el-table td.el-table__cell {
  587. border-color: #555555;
  588. }
  589. .el-table--border .el-table__inner-wrapper:after,
  590. .el-table--border:after,
  591. .el-table--border:before,
  592. .el-table__inner-wrapper:before,
  593. .el-table__border-bottom-patch,
  594. .el-table__border-left-patch {
  595. background-color: #555555;
  596. }
  597. }
  598. }
  599. .ask {
  600. width: 100%;
  601. display: flex;
  602. .bubble {
  603. width: 100%;
  604. }
  605. }
  606. .ask-title {
  607. color: #999999;
  608. padding-bottom: 10px;
  609. }
  610. .ask-list {
  611. color: #55e5c6;
  612. }
  613. .to-map {
  614. display: flex;
  615. justify-content: space-between;
  616. background: #3c3c3c;
  617. padding: 6px 8px;
  618. border-radius: 6px;
  619. cursor: pointer;
  620. }
  621. .to-map + .to-map {
  622. margin-top: 8px;
  623. }
  624. .ask-item + .ask-item {
  625. padding-top: 2px;
  626. }
  627. .route-wrap {
  628. display: flex;
  629. .route-img {
  630. width: 100%;
  631. padding-top: 8px;
  632. }
  633. }
  634. .user {
  635. justify-content: flex-end;
  636. }
  637. .system {
  638. justify-content: flex-start;
  639. }
  640. .link {
  641. display: flex;
  642. .header-link {
  643. color: #55e5c6;
  644. display: flex;
  645. align-items: baseline;
  646. text-decoration: underline;
  647. .icon {
  648. margin-right: 5px;
  649. }
  650. }
  651. }
  652. .bubble {
  653. padding: 16px 12px;
  654. box-sizing: border-box;
  655. max-width: 100%;
  656. border-radius: 8px;
  657. // max-width: 60%;
  658. background: rgba(255, 212, 137, 0.1);
  659. color: #55e5c6;
  660. border-radius: 16px 2px 16px 16px;
  661. line-height: 24px;
  662. font-size: 14px;
  663. .header {
  664. color: #999999;
  665. // padding-bottom: 8px;
  666. }
  667. .main-text {
  668. // color: #ffffff;
  669. }
  670. .content {
  671. color: #ffffff;
  672. }
  673. .divider {
  674. width: 100%;
  675. height: 1px;
  676. background: rgba(153, 153, 153, 0.2);
  677. margin: 12px 0;
  678. }
  679. .think {
  680. color: #999999;
  681. }
  682. .expert-img {
  683. margin-top: 12px;
  684. width: 100%;
  685. }
  686. }
  687. .system .bubble,
  688. .real .bubble {
  689. background: #2f2f2f;
  690. color: #fff;
  691. border-radius: 2px 16px 16px 16px;
  692. }
  693. .uploader {
  694. width: 100%;
  695. display: flex;
  696. justify-content: center;
  697. img {
  698. width: 174px;
  699. height: 55px;
  700. }
  701. }
  702. .bottom-send {
  703. padding: 6px 16px 16px;
  704. }
  705. .input-box {
  706. display: flex;
  707. flex-direction: column;
  708. padding: 10px;
  709. background: #2f2f2f;
  710. border-radius: 16px;
  711. /* border-top: 1px solid #ddd; */
  712. ::v-deep {
  713. .el-textarea__inner {
  714. background: transparent;
  715. box-shadow: none;
  716. color: #fff;
  717. }
  718. }
  719. .bottom-group {
  720. display: flex;
  721. justify-content: space-between;
  722. align-items: center;
  723. padding-top: 6px;
  724. .btn-l {
  725. display: flex;
  726. .l-item {
  727. display: flex;
  728. align-items: center;
  729. border-radius: 20px;
  730. border: 1px solid #555555;
  731. padding: 6px 8px;
  732. }
  733. .l-item + .l-item {
  734. margin-left: 12px;
  735. }
  736. img {
  737. width: 16px;
  738. padding-right: 2px;
  739. }
  740. }
  741. .file-icon {
  742. // width: 16px;
  743. height: 16px;
  744. }
  745. .send-icon {
  746. margin-left: 8px;
  747. width: 28px;
  748. }
  749. }
  750. }
  751. input {
  752. flex: 1;
  753. padding: 10px;
  754. border: 1px solid #ddd;
  755. border-radius: 4px;
  756. }
  757. button {
  758. margin-left: 10px;
  759. padding: 10px 15px;
  760. background: #007bff;
  761. color: white;
  762. border: none;
  763. cursor: pointer;
  764. }
  765. </style>