bottomSider2.vue 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. <template>
  2. <div>
  3. <div class="trigger-button">
  4. <div class="icon" @click="openMenu"></div>
  5. </div>
  6. <transition name="fade">
  7. <div v-if="dialogVisible" class="dialog-overlay">
  8. <!-- 左侧折叠区域 -->
  9. <div class="left-side" :class="{ collapsed: isFold }" id="leftSide">
  10. <div
  11. class="addBtn"
  12. :style="{
  13. backgroundImage: `url(${isFold ? '/src/assets/images/vent/home/add.svg' : ''})`,
  14. backgroundColor: isFold ? '' : '#2cb6ff',
  15. width: isFold ? '20px' : 'auto',
  16. }"
  17. >
  18. <span
  19. class="btn-text-bg"
  20. :style="{
  21. backgroundImage: `url(${!isFold ? '/src/assets/images/vent/home/addB.svg' : ''})`,
  22. }"
  23. ></span>
  24. <span v-if="!isFold" class="btn-text" @click="addNew">添加新对话</span>
  25. </div>
  26. <div class="divider0"></div>
  27. <div
  28. v-if="isFold"
  29. class="historyBtn"
  30. :style="{ backgroundImage: `url(${isFold ? '/src/assets/images/vent/home/history.svg' : '/src/assets/images/vent/home/history.svg'})` }"
  31. @click="addNew"
  32. ></div>
  33. <div v-else class="historyBtn1">
  34. <span
  35. class="btn-text-bg"
  36. :style="{
  37. backgroundImage: `url(${!isFold ? '/src/assets/images/vent/home/history.svg' : ''})`,
  38. }"
  39. ></span>
  40. <span v-if="!isFold" class="btn-text">历史对话</span>
  41. <a-list style="width: 100px" :split="false">
  42. <a-list-item style="padding: 5px 0 0 30px; color: #5e7081; font-size: 12px; white-space: nowrap; text-overflow: ellipsis"
  43. >历史数据1</a-list-item
  44. >
  45. <a-list-item style="padding: 5px 0 0 30px; color: #5e7081; font-size: 12px; white-space: nowrap; text-overflow: ellipsis"
  46. >历史数据12</a-list-item
  47. >
  48. <a-list-item style="padding: 5px 0 0 30px; color: #5e7081; font-size: 12px; white-space: nowrap; text-overflow: ellipsis"
  49. >历史数据123</a-list-item
  50. >
  51. </a-list>
  52. </div>
  53. <div
  54. class="foldBtn"
  55. :style="{ backgroundImage: `url(${isFold ? '/src/assets/images/vent/home/Fold.svg' : '/src/assets/images/vent/home/unfold.svg'})` }"
  56. @click="fold"
  57. ></div>
  58. </div>
  59. <!-- 右侧对话框 -->
  60. <div class="right-side">
  61. <div class="input-content">
  62. <!-- 对话区域 -->
  63. <div class="dialog-area">
  64. <div v-for="message in sortedMessages" :key="message.id" :class="['message-item', message.type]">
  65. <!-- 用户提问样式 -->
  66. <div v-if="message.type === 'user'" class="ask-message">
  67. <span>{{ message.content }}</span>
  68. </div>
  69. <!-- 系统回答样式 -->
  70. <div v-else class="system-message">
  71. <div class="answerIcon"></div>
  72. <div class="answer-message">
  73. <div>
  74. <span>{{ message.content }}</span>
  75. </div>
  76. </div>
  77. </div>
  78. </div>
  79. </div>
  80. <!-- 文本输入区域 -->
  81. <div v-if="spinning" class="thinking-area">
  82. <span style="color: #fff">思考中···</span>
  83. <a-spin :spinning="spinning"></a-spin>
  84. </div>
  85. <div class="input-area">
  86. <textarea v-model="inputText" placeholder="请输入你的问题"> </textarea>
  87. <!-- 底部操作栏 -->
  88. <div class="action-bar">
  89. <!-- 文件上传按钮 -->
  90. <label class="upload-btn">
  91. <div class="send-file" />
  92. <span class="divider"> | </span>
  93. <div class="send-img" />
  94. </label>
  95. <!-- 发送按钮 -->
  96. <div class="send-btn" @click="handleSend"></div>
  97. </div>
  98. </div>
  99. </div>
  100. </div>
  101. </div>
  102. </transition>
  103. </div>
  104. </template>
  105. <script lang="ts" setup>
  106. import { ref, onMounted, nextTick, computed } from 'vue';
  107. // 响应式变量声明
  108. const dialogVisible = ref(false);
  109. const isFold = ref(false); // 是否折叠
  110. const inputText = ref(''); // 输入框内容
  111. // const messages = ref([]); // 消息列表
  112. const spinning = ref(false); // 加载状态
  113. const systemMessage = ref(''); // 系统返回信息
  114. const session_id = ref(''); // 会话id
  115. const hasCreated = ref(false); // 标志位,防止重复调用create接口
  116. type MessageItem = {
  117. id: string; // 唯一标识(可用时间戳生成)
  118. type: 'user' | 'system';
  119. content: string;
  120. timestamp: number; // 排序依据
  121. };
  122. const messageList = ref<MessageItem[]>([]);
  123. const sortedMessages = computed(() => {
  124. return messageList.value.sort((a, b) => a.timestamp - b.timestamp);
  125. });
  126. // const scrollToBottom = () => {
  127. // nextTick(() => {
  128. // const container = document.querySelector('.dialog-area');
  129. // container.scrollTop = container.scrollHeight;
  130. // });
  131. // };
  132. const openMenu = () => {
  133. dialogVisible.value = !dialogVisible.value;
  134. if (dialogVisible.value) {
  135. hasCreated.value = true;
  136. addNew();
  137. }
  138. };
  139. const fold = () => {
  140. isFold.value = !isFold.value;
  141. };
  142. async function addNew() {
  143. let response = await fetch('http://182.92.126.35:6005/sessions/create', {
  144. method: 'post',
  145. headers: {
  146. 'Content-Type': 'application/json',
  147. },
  148. });
  149. const data = await response.json();
  150. session_id.value = data.session_id;
  151. }
  152. //获取消息列表
  153. async function handleSend() {
  154. spinning.value = true;
  155. // 添加用户消息
  156. messageList.value.push({
  157. id: `user_${Date.now()}`,
  158. type: 'user',
  159. content: inputText.value,
  160. timestamp: Date.now(),
  161. });
  162. // 调用接口获取答案
  163. // const answer = await fetchAnswerFromAPI(question);
  164. const params = {
  165. chat_session_id: session_id.value,
  166. prompt: inputText.value,
  167. ref_file_ids: [],
  168. thinking_enabled: false,
  169. };
  170. inputText.value = ''; // 清空输入框
  171. //将用户输入的内容发送到后端
  172. let response = await fetch('http://182.92.126.35:6005/chat', {
  173. method: 'POST',
  174. headers: {
  175. 'Content-Type': 'application/json',
  176. },
  177. body: JSON.stringify(params),
  178. });
  179. const data = await response.json();
  180. const assistantReply = data.reply; // 获取助手回复
  181. systemMessage.value = assistantReply;
  182. // 添加系统回答
  183. messageList.value.push({
  184. id: `system_${Date.now()}`,
  185. type: 'system',
  186. content: systemMessage.value,
  187. timestamp: Date.now(),
  188. });
  189. spinning.value = false;
  190. }
  191. // async function handleSend() {
  192. // spinning.value = true;
  193. // userMessage.value.push({
  194. // msg: inputText.value, // 消息内容
  195. // });
  196. // inputText.value = ''; // 清空输入框
  197. // const params = {
  198. // messages: [{ role: 'user', content: inputText.value }],
  199. // };
  200. // //将用户输入的内容发送到后端
  201. // let response = await fetch('http://182.92.126.35:6005/chat', {
  202. // method: 'POST',
  203. // headers: {
  204. // 'Content-Type': 'application/json',
  205. // },
  206. // body: JSON.stringify(params),
  207. // });
  208. // const data = await response.json();
  209. // spinning.value = false;
  210. // const assistantReply = data.reply; // 获取助手回复
  211. // systemMessage.value = assistantReply;
  212. // }
  213. // 初始化按钮定位
  214. onMounted(() => {});
  215. </script>
  216. <style lang="less" scoped>
  217. @keyframes menuShow {
  218. 0% {
  219. width: 0;
  220. height: 0;
  221. }
  222. 100% {
  223. width: 480px;
  224. height: 100vh;
  225. }
  226. }
  227. .trigger-button {
  228. position: fixed;
  229. bottom: 10px;
  230. right: 10px;
  231. z-index: 1000000;
  232. .icon {
  233. width: 60px;
  234. height: 60px;
  235. position: relative;
  236. background-image: url('/@/assets/images/vent/home/wakeBtn.png');
  237. background-position: center;
  238. background-size: 100% 100%;
  239. }
  240. }
  241. .dialog-overlay {
  242. width: 32%;
  243. height: 55%;
  244. z-index: 999;
  245. display: flex;
  246. position: fixed;
  247. right: 90px;
  248. bottom: 20px;
  249. box-shadow: 0 0 3px 3px #1074c1;
  250. background-color: #09172c;
  251. }
  252. /* 遮罩层淡入淡出 */
  253. .fade-enter-active,
  254. .fade-leave-active {
  255. transition: opacity 0.3s;
  256. }
  257. .fade-enter-from,
  258. .fade-leave-to {
  259. opacity: 0;
  260. }
  261. /* 弹窗缩放动画 */
  262. .scale-enter-active,
  263. .scale-leave-active {
  264. transition: all 0.3s ease;
  265. }
  266. .scale-enter-from {
  267. transform: scale(0.5) translate(-50%, -50%);
  268. opacity: 0;
  269. }
  270. .scale-leave-to {
  271. transform: scale(1.2) translate(-50%, -50%);
  272. opacity: 0;
  273. }
  274. .left-side {
  275. background: #0c2842;
  276. transition: width 0.5s ease; /* 平滑过渡动画 */
  277. width: 120px; /* 展开时宽度 */
  278. position: relative; /* 用于按钮定位 */
  279. }
  280. .left-side.collapsed {
  281. width: 40px; /* 折叠时宽度 */
  282. }
  283. .addBtn {
  284. height: 30px;
  285. position: absolute;
  286. background-size: 100% 100%;
  287. background-position: center;
  288. padding: 2px;
  289. right: 10px;
  290. top: 10px;
  291. left: 10px;
  292. align-items: center;
  293. border-radius: 3px;
  294. cursor: pointer;
  295. }
  296. .btn-text-bg {
  297. width: 14px;
  298. height: 14px;
  299. position: absolute;
  300. background-size: 100% 100%;
  301. right: 10px;
  302. top: 9px;
  303. left: 10px;
  304. bottom: 10px;
  305. }
  306. .btn-text {
  307. margin-left: 3px;
  308. font-size: 12px;
  309. color: #fff;
  310. white-space: nowrap;
  311. margin-left: 30px;
  312. line-height: 26px;
  313. }
  314. .historyBtn {
  315. width: 20px;
  316. height: 20px;
  317. position: absolute;
  318. background-size: 100% 100%;
  319. background-position: center;
  320. padding: 2px;
  321. right: 10px;
  322. top: 100px;
  323. }
  324. .historyBtn1 {
  325. width: 20px;
  326. height: 20px;
  327. position: absolute;
  328. background-size: 100% 100%;
  329. background-position: center;
  330. left: 3px;
  331. top: 80px;
  332. }
  333. .divider0 {
  334. border-bottom: 1px solid #1074c1;
  335. width: auto;
  336. margin: 0 10px;
  337. height: 13%;
  338. display: block;
  339. background: transparent;
  340. }
  341. .foldBtn {
  342. width: 20px;
  343. height: 20px;
  344. position: absolute;
  345. background-size: 100% 100%;
  346. background-position: center;
  347. padding: 2px;
  348. right: 10px;
  349. bottom: 10px;
  350. cursor: pointer;
  351. }
  352. .right-side {
  353. flex: 1; /* 占据剩余空间 */
  354. background: #09172c;
  355. }
  356. .input-content {
  357. display: flex;
  358. flex-direction: column;
  359. justify-content: flex-end; /* 内容底部对齐 */
  360. height: 100%;
  361. padding: 20px; /* 统一内边距 */
  362. }
  363. .ask-message {
  364. align-self: flex-end;
  365. float: right;
  366. max-width: 70%;
  367. padding: 10px;
  368. margin: 10px;
  369. border-radius: 5px;
  370. color: #fff;
  371. background: #0c2842;
  372. align-self: flex-end; /* 右侧对齐‌:ml-citation{ref="2" data="citationList"} */
  373. }
  374. .answer {
  375. display: flex;
  376. flex-direction: row;
  377. }
  378. .answerIcon {
  379. margin-top: 10px;
  380. width: 35px;
  381. height: 35px;
  382. background-image: url('/@/assets/images/vent/home/answerIcon.svg');
  383. background-size: 100% 100%;
  384. }
  385. .answer-message {
  386. float: left;
  387. max-width: 80%;
  388. padding: 10px;
  389. margin: 10px;
  390. border-radius: 5px;
  391. color: #fff;
  392. background: #0c2842;
  393. }
  394. /** 系统返回信息**/
  395. .system-message {
  396. display: flex;
  397. flex-direction: row;
  398. align-self: flex-start;
  399. max-width: 90%;
  400. padding: 12px;
  401. display: flex;
  402. }
  403. .answerIcon {
  404. margin-top: 10px;
  405. width: 35px;
  406. height: 35px;
  407. background-image: url('/@/assets/images/vent/home/answerIcon.svg');
  408. background-size: 100% 100%;
  409. }
  410. .answer-message {
  411. float: left;
  412. max-width: 80%;
  413. padding: 10px;
  414. margin: 10px;
  415. border-radius: 5px;
  416. color: #fff;
  417. background: #0c2842;
  418. }
  419. .dialog-area {
  420. flex: 1; /* 占据剩余空间 */
  421. gap: 50px; /* 消息块间隔统一控制 */
  422. overflow-y: auto; /* 垂直滚动条 */
  423. margin-bottom: 10px;
  424. }
  425. .loading-wrapper,
  426. .content-wrapper {
  427. min-height: 40px; /* 避免高度塌陷 */
  428. }
  429. .message-item.user {
  430. margin-bottom: 30px;
  431. }
  432. .input-area {
  433. background-color: #043256 !important;
  434. display: flex;
  435. flex-direction: column;
  436. gap: 10px;
  437. }
  438. textarea {
  439. background-color: #043256 !important;
  440. width: 100%;
  441. height: 40px;
  442. border: none;
  443. resize: none;
  444. outline: none;
  445. overflow: hidden;
  446. padding: 10px; /* 统一内边距 */
  447. color: #fff;
  448. }
  449. .action-bar {
  450. display: flex;
  451. align-items: center;
  452. gap: 12px;
  453. }
  454. .upload-btn {
  455. display: flex;
  456. cursor: pointer;
  457. padding: 6px 12px;
  458. }
  459. .divider {
  460. color: #ccc;
  461. font-weight: 300;
  462. margin: 0 10px;
  463. }
  464. .send-file {
  465. width: 20px;
  466. height: 20px;
  467. background-image: url('/@/assets/images/vent/home/sendFile.svg');
  468. background-size: 100% 100%;
  469. border-radius: 4px;
  470. cursor: pointer;
  471. }
  472. .send-img {
  473. width: 20px;
  474. height: 20px;
  475. background-image: url('/@/assets/images/vent/home/sendImg.svg');
  476. background-size: 100% 100%;
  477. border-radius: 4px;
  478. cursor: pointer;
  479. }
  480. .send-btn {
  481. width: 20px;
  482. height: 20px;
  483. margin-left: auto;
  484. margin-right: 10px;
  485. background-color: #1074c1;
  486. background-image: url('/@/assets/images/vent/home/send.svg');
  487. background-position: center;
  488. background-size: 100% 100%;
  489. border-radius: 2px;
  490. cursor: pointer;
  491. }
  492. </style>