WorkView.vue 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311
  1. <script setup>
  2. import { ref, unref, computed, onMounted, onUnmounted } from 'vue';
  3. import { useMessage } from 'naive-ui';
  4. import { BaseButton, RecodeCardItem, TheSubMenu, TheChatView, ChatWelcome, SvgIcon } from '@/components';
  5. import { ChatAsk, ChatAnswer, ChatInputCopy } from '@/components/Chat';
  6. import { chatApi } from '@/api/chat';
  7. import { helperApi } from '@/api/helper';
  8. import { useInfinite, useScroll, useChat } from '@/composables';
  9. const ANSWER_ID_KEY = '@@id@@';
  10. let controller = new AbortController();
  11. const { recordList, isFetching, onScrolltolower, onReset, } = useInfinite('/front/bigModel/qa/pageList', { module: 2 });
  12. const { scrollRef, scrollToBottom, scrollToBottomIfAtBottom } = useScroll();
  13. const { chatDataSource, addChat, updateChat, clearChat, updateById } = useChat();
  14. const helperList = ref([]);
  15. const message = useMessage();
  16. const switchActive = ref(false);
  17. const isLoading = ref(false);
  18. const inputRef = ref(null);
  19. const recordActive = ref(null);
  20. const activeItem = ref({});
  21. const currenSessionId = ref(null);
  22. const isExistInHistory = computed(() => (recordList.value.findIndex(({ sessionId: sId }) => sId === unref(currenSessionId)) === -1));
  23. // 新建对话
  24. const handleCreateDialog = async () => {
  25. message.destroyAll();
  26. if (unref(isLoading)) {
  27. return message.warning('当前对话生成中');
  28. }
  29. if (!unref(chatDataSource).length) {
  30. return message.info('已切换最新会话');
  31. }
  32. inputRef.value.clearInpVal();
  33. recordActive.value = null;
  34. currenSessionId.value = null;
  35. clearChat();
  36. }
  37. // 查询对话详情
  38. const handleChatDetail = async ({ sessionId }) => {
  39. isLoading.value = false;
  40. recordActive.value = sessionId;
  41. inputRef.value.clearInpVal();
  42. controller.abort();
  43. const { data } = await chatApi.getAnswerHistoryDetail({ sessionId });
  44. chatDataSource.value = data.map(item => ({ ...item, loading: false, }));
  45. currenSessionId.value = sessionId;
  46. scrollToBottom();
  47. }
  48. const onRegenerate = async ({ question, tools }) => {
  49. controller = new AbortController();
  50. const sessionId = unref(currenSessionId);
  51. const params = {
  52. data: {
  53. sessionId,
  54. showVal: question,
  55. question: question,
  56. module: 2,
  57. tools: activeItem.value.tools || tools,
  58. isStrong: Number(unref(switchActive))
  59. },
  60. signal: controller.signal,
  61. onDownloadProgress: ({ event }) => {
  62. const xhr = event.target;
  63. const { responseText } = xhr;
  64. const [answer] = responseText.split(ANSWER_ID_KEY);
  65. updateChat({
  66. sessionId,
  67. question,
  68. answer,
  69. loading: true,
  70. delayLoading: false
  71. })
  72. scrollToBottomIfAtBottom();
  73. }
  74. }
  75. try {
  76. const { data } = await chatApi.getChatStream(params);
  77. const [answer, id] = data.split(ANSWER_ID_KEY);
  78. updateChat({
  79. id,
  80. sessionId,
  81. question,
  82. answer,
  83. loading: false,
  84. delayLoading: false
  85. })
  86. scrollToBottomIfAtBottom();
  87. }
  88. catch (error) {
  89. console.log("取消了请求 - catch", error);
  90. }
  91. finally {
  92. isLoading.value = false;
  93. onReset();
  94. }
  95. }
  96. // 提交问题
  97. const handleSubmit = async ({question, selectedOption}) => {
  98. if (unref(isExistInHistory)) {
  99. const { data: sessionId } = await chatApi.getChatSessionTag();
  100. currenSessionId.value = sessionId;
  101. }
  102. isLoading.value = true;
  103. addChat({
  104. sessionId: unref(currenSessionId),
  105. question,
  106. answer: '',
  107. loading: true,
  108. delayLoading: true
  109. })
  110. scrollToBottom();
  111. setTimeout(() => onRegenerate({ question, tools: selectedOption?.tools || null }), 2 * 1000);
  112. }
  113. // 处理推荐问题
  114. const handleWelcomeRecommend = (item) => {
  115. activeItem.value = item;
  116. inputRef.value.inpVal = item.content;
  117. inputRef.value.handleInpFocus();
  118. }
  119. // 删除历史对话
  120. const handeChatDelete = async (id) => {
  121. await chatApi.deleteHistory(id);
  122. onReset();
  123. clearChat();
  124. message.success('删除成功');
  125. }
  126. // 返回操作
  127. const handleback = async () => {
  128. controller?.abort();
  129. // await chatApi.getStopChatStream(currenSessionId.value);
  130. inputRef.value.clearInpVal();
  131. recordActive.value = null;
  132. currenSessionId.value = null;
  133. clearChat();
  134. }
  135. onMounted(async () => {
  136. const { data } = await helperApi.getHelperList();
  137. helperList.value = data;
  138. })
  139. onUnmounted(() => {
  140. controller.abort();
  141. })
  142. </script>
  143. <template>
  144. <section class="flex items-start h-full">
  145. <TheSubMenu title="历史记录" @scrollToLower="onScrolltolower" :loading="isFetching">
  146. <template #top>
  147. <div class="create-btn px-[11px] pb-[22px]">
  148. <BaseButton @click="handleCreateDialog" icon-name="tool-add-circle">新建指令</BaseButton>
  149. </div>
  150. </template>
  151. <div class="pr-[4px] text-[#5e5e5e]">
  152. <RecodeCardItem v-for="item, index in recordList" :key="item.sessionId + index" :title="item.showVal"
  153. :time="item.createTime" :data-item="item"
  154. :class="{ 'recode-card-item_active': recordActive === item.sessionId }" @on-click="handleChatDetail"
  155. @on-delete="handeChatDelete" />
  156. </div>
  157. </TheSubMenu>
  158. <TheChatView ref="scrollRef" :is-back-btn="!!chatDataSource.length" @on-click-back="handleback">
  159. <div v-show="!chatDataSource.length">
  160. <ChatWelcome title="您好,我是LibraAI智能助手" :sub-title="[
  161. 'LibarAI智能助手模块提供撰写文章、生成报告等服务',
  162. '请替换问题中##的内容'
  163. ]" />
  164. <div class="grid-container">
  165. <div class="grid-content">
  166. <div class="grid-item" v-for="item in helperList" :key="item.id" @click="handleWelcomeRecommend(item)">
  167. <div class="grid-item-icon space-x-[8px]">
  168. <img :src="item.banner" alt="" class="w-[24px]">
  169. <!-- <SvgIcon name="tool-report" size="24"></SvgIcon> -->
  170. <h3 class="grid-item-title">{{ item.title }}</h3>
  171. </div>
  172. <div class="text-[#5E5E5E] mt-[8px] text-justify">
  173. <template v-if="item.content.indexOf('#') !== -1">
  174. <span v-for="word, i in item.content.split('#')" :key="word">
  175. {{ word }}<i v-if="i !== item.content.split('#').length - 1" class="text-[#2454FF]">#</i>
  176. </span>
  177. </template>
  178. <template v-else>
  179. <p>{{ item.content }}</p>
  180. </template>
  181. </div>
  182. </div>
  183. </div>
  184. </div>
  185. </div>
  186. <div class="conversation-item" v-if="chatDataSource.length">
  187. <template v-for="item in chatDataSource" :key="item.id">
  188. <ChatAsk :content="item.question" :sessionId="item.sessionId"></ChatAsk>
  189. <ChatAnswer :id="item.id" :content="item.answer" :loading="item.loading" :delay-loading="item.delayLoading"
  190. :isSatisfied="item.isSatisfied" @on-click-icon="params => updateById(params)"></ChatAnswer>
  191. </template>
  192. </div>
  193. <template #footer>
  194. <ChatInputCopy
  195. :options="helperList"
  196. :active-item="activeItem"
  197. ref="inputRef"
  198. v-model:loading="isLoading"
  199. v-model:switch="switchActive"
  200. @on-click="handleSubmit"
  201. @on-enter="handleSubmit"
  202. ></ChatInputCopy>
  203. </template>
  204. </TheChatView>
  205. </section>
  206. </template>
  207. <style scoped lang="scss">
  208. .grid-container {
  209. position: relative;
  210. padding-bottom: 20px;
  211. margin-top: 36px;
  212. overflow: hidden;
  213. overflow-y: scroll;
  214. &::-webkit-scrollbar {
  215. width: 0px;
  216. }
  217. .grid-content {
  218. column-count: 3;
  219. column-gap: 16px;
  220. -moz-column-count: 3;
  221. -webkit-column-count: 3;
  222. -moz-column-gap: 16px;
  223. -webkit-column-gap: 16px;
  224. .grid-item {
  225. height: auto;
  226. padding: 16px;
  227. margin-bottom: 16px;
  228. border-radius: 10px;
  229. background-color: #fff;
  230. -webkit-column-break-inside: avoid;
  231. break-inside: avoid;
  232. border: 1px solid #fff;
  233. cursor: pointer;
  234. &:hover {
  235. border: 1px solid #2454FF;
  236. box-shadow: 0 2px 4px rgba(0, 0, 0, 0.08);
  237. }
  238. .grid-item-icon {
  239. display: flex;
  240. align-items: center;
  241. justify-content: start;
  242. .grid-item-title {
  243. font-size: 14px;
  244. font-weight: 600;
  245. color: #1A2029;
  246. }
  247. }
  248. }
  249. }
  250. }
  251. </style>@/api/order