WorkView.vue 9.3 KB

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