【新增】前端代码第一次提交
This commit is contained in:
		| @@ -0,0 +1,472 @@ | ||||
| <!--  AI 对话  --> | ||||
| <template> | ||||
|   <el-aside width="260px" class="conversation-container h-100%"> | ||||
|     <!-- 左顶部:对话 --> | ||||
|     <div class="h-100%"> | ||||
|       <el-button class="w-1/1 btn-new-conversation" type="primary" @click="createConversation"> | ||||
|         <Icon icon="ep:plus" class="mr-5px" /> | ||||
|         新建对话 | ||||
|       </el-button> | ||||
|  | ||||
|       <!-- 左顶部:搜索对话 --> | ||||
|       <el-input | ||||
|         v-model="searchName" | ||||
|         size="large" | ||||
|         class="mt-10px search-input" | ||||
|         placeholder="搜索历史记录" | ||||
|         @keyup="searchConversation" | ||||
|       > | ||||
|         <template #prefix> | ||||
|           <Icon icon="ep:search" /> | ||||
|         </template> | ||||
|       </el-input> | ||||
|  | ||||
|       <!-- 左中间:对话列表 --> | ||||
|       <div class="conversation-list"> | ||||
|         <!-- 情况一:加载中 --> | ||||
|         <el-empty v-if="loading" description="." :v-loading="loading" /> | ||||
|         <!-- 情况二:按照 group 分组,展示聊天会话 list 列表 --> | ||||
|         <div v-for="conversationKey in Object.keys(conversationMap)" :key="conversationKey"> | ||||
|           <div | ||||
|             class="conversation-item classify-title" | ||||
|             v-if="conversationMap[conversationKey].length" | ||||
|           > | ||||
|             <el-text class="mx-1" size="small" tag="b">{{ conversationKey }}</el-text> | ||||
|           </div> | ||||
|           <div | ||||
|             class="conversation-item" | ||||
|             v-for="conversation in conversationMap[conversationKey]" | ||||
|             :key="conversation.id" | ||||
|             @click="handleConversationClick(conversation.id)" | ||||
|             @mouseover="hoverConversationId = conversation.id" | ||||
|             @mouseout="hoverConversationId = ''" | ||||
|           > | ||||
|             <div | ||||
|               :class=" | ||||
|                 conversation.id === activeConversationId ? 'conversation active' : 'conversation' | ||||
|               " | ||||
|             > | ||||
|               <div class="title-wrapper"> | ||||
|                 <img class="avatar" :src="conversation.roleAvatar || roleAvatarDefaultImg" /> | ||||
|                 <span class="title">{{ conversation.title }}</span> | ||||
|               </div> | ||||
|               <div class="button-wrapper" v-show="hoverConversationId === conversation.id"> | ||||
|                 <el-button class="btn" link @click.stop="handleTop(conversation)"> | ||||
|                   <el-icon title="置顶" v-if="!conversation.pinned"><Top /></el-icon> | ||||
|                   <el-icon title="置顶" v-if="conversation.pinned"><Bottom /></el-icon> | ||||
|                 </el-button> | ||||
|                 <el-button class="btn" link @click.stop="updateConversationTitle(conversation)"> | ||||
|                   <el-icon title="编辑"> | ||||
|                     <Icon icon="ep:edit" /> | ||||
|                   </el-icon> | ||||
|                 </el-button> | ||||
|                 <el-button class="btn" link @click.stop="deleteChatConversation(conversation)"> | ||||
|                   <el-icon title="删除对话"> | ||||
|                     <Icon icon="ep:delete" /> | ||||
|                   </el-icon> | ||||
|                 </el-button> | ||||
|               </div> | ||||
|             </div> | ||||
|           </div> | ||||
|         </div> | ||||
|         <!-- 底部占位  --> | ||||
|         <div class="h-160px w-100%"></div> | ||||
|       </div> | ||||
|     </div> | ||||
|  | ||||
|     <!-- 左底部:工具栏 --> | ||||
|     <div class="tool-box"> | ||||
|       <div @click="handleRoleRepository"> | ||||
|         <Icon icon="ep:user" /> | ||||
|         <el-text size="small">角色仓库</el-text> | ||||
|       </div> | ||||
|       <div @click="handleClearConversation"> | ||||
|         <Icon icon="ep:delete" /> | ||||
|         <el-text size="small">清空未置顶对话</el-text> | ||||
|       </div> | ||||
|     </div> | ||||
|  | ||||
|     <!-- 角色仓库抽屉 --> | ||||
|     <el-drawer v-model="roleRepositoryOpen" title="角色仓库" size="754px"> | ||||
|       <RoleRepository /> | ||||
|     </el-drawer> | ||||
|   </el-aside> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { ChatConversationApi, ChatConversationVO } from '@/api/ai/chat/conversation' | ||||
| import RoleRepository from '../role/RoleRepository.vue' | ||||
| import { Bottom, Top } from '@element-plus/icons-vue' | ||||
| import roleAvatarDefaultImg from '@/assets/ai/gpt.svg' | ||||
|  | ||||
| const message = useMessage() // 消息弹窗 | ||||
|  | ||||
| // 定义属性 | ||||
| const searchName = ref<string>('') // 对话搜索 | ||||
| const activeConversationId = ref<number | null>(null) // 选中的对话,默认为 null | ||||
| const hoverConversationId = ref<number | null>(null) // 悬浮上去的对话 | ||||
| const conversationList = ref([] as ChatConversationVO[]) // 对话列表 | ||||
| const conversationMap = ref<any>({}) // 对话分组 (置顶、今天、三天前、一星期前、一个月前) | ||||
| const loading = ref<boolean>(false) // 加载中 | ||||
| const loadingTime = ref<any>() // 加载中定时器 | ||||
|  | ||||
| // 定义组件 props | ||||
| const props = defineProps({ | ||||
|   activeId: { | ||||
|     type: String || null, | ||||
|     required: true | ||||
|   } | ||||
| }) | ||||
|  | ||||
| // 定义钩子 | ||||
| const emits = defineEmits([ | ||||
|   'onConversationCreate', | ||||
|   'onConversationClick', | ||||
|   'onConversationClear', | ||||
|   'onConversationDelete' | ||||
| ]) | ||||
|  | ||||
| /** 搜索对话 */ | ||||
| const searchConversation = async (e) => { | ||||
|   // 恢复数据 | ||||
|   if (!searchName.value.trim().length) { | ||||
|     conversationMap.value = await getConversationGroupByCreateTime(conversationList.value) | ||||
|   } else { | ||||
|     // 过滤 | ||||
|     const filterValues = conversationList.value.filter((item) => { | ||||
|       return item.title.includes(searchName.value.trim()) | ||||
|     }) | ||||
|     conversationMap.value = await getConversationGroupByCreateTime(filterValues) | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 点击对话 */ | ||||
| const handleConversationClick = async (id: number) => { | ||||
|   // 过滤出选中的对话 | ||||
|   const filterConversation = conversationList.value.filter((item) => { | ||||
|     return item.id === id | ||||
|   }) | ||||
|   // 回调 onConversationClick | ||||
|   // noinspection JSVoidFunctionReturnValueUsed | ||||
|   const success = emits('onConversationClick', filterConversation[0]) | ||||
|   // 切换对话 | ||||
|   if (success) { | ||||
|     activeConversationId.value = id | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 获取对话列表 */ | ||||
| const getChatConversationList = async () => { | ||||
|   try { | ||||
|     // 加载中 | ||||
|     loadingTime.value = setTimeout(() => { | ||||
|       loading.value = true | ||||
|     }, 50) | ||||
|  | ||||
|     // 1.1 获取 对话数据 | ||||
|     conversationList.value = await ChatConversationApi.getChatConversationMyList() | ||||
|     // 1.2 排序 | ||||
|     conversationList.value.sort((a, b) => { | ||||
|       return b.createTime - a.createTime | ||||
|     }) | ||||
|     // 1.3 没有任何对话情况 | ||||
|     if (conversationList.value.length === 0) { | ||||
|       activeConversationId.value = null | ||||
|       conversationMap.value = {} | ||||
|       return | ||||
|     } | ||||
|  | ||||
|     // 2. 对话根据时间分组(置顶、今天、一天前、三天前、七天前、30 天前) | ||||
|     conversationMap.value = await getConversationGroupByCreateTime(conversationList.value) | ||||
|   } finally { | ||||
|     // 清理定时器 | ||||
|     if (loadingTime.value) { | ||||
|       clearTimeout(loadingTime.value) | ||||
|     } | ||||
|     // 加载完成 | ||||
|     loading.value = false | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 按照 creteTime 创建时间,进行分组 */ | ||||
| const getConversationGroupByCreateTime = async (list: ChatConversationVO[]) => { | ||||
|   // 排序、指定、时间分组(今天、一天前、三天前、七天前、30天前) | ||||
|   // noinspection NonAsciiCharacters | ||||
|   const groupMap = { | ||||
|     置顶: [], | ||||
|     今天: [], | ||||
|     一天前: [], | ||||
|     三天前: [], | ||||
|     七天前: [], | ||||
|     三十天前: [] | ||||
|   } | ||||
|   // 当前时间的时间戳 | ||||
|   const now = Date.now() | ||||
|   // 定义时间间隔常量(单位:毫秒) | ||||
|   const oneDay = 24 * 60 * 60 * 1000 | ||||
|   const threeDays = 3 * oneDay | ||||
|   const sevenDays = 7 * oneDay | ||||
|   const thirtyDays = 30 * oneDay | ||||
|   for (const conversation of list) { | ||||
|     // 置顶 | ||||
|     if (conversation.pinned) { | ||||
|       groupMap['置顶'].push(conversation) | ||||
|       continue | ||||
|     } | ||||
|     // 计算时间差(单位:毫秒) | ||||
|     const diff = now - conversation.createTime | ||||
|     // 根据时间间隔判断 | ||||
|     if (diff < oneDay) { | ||||
|       groupMap['今天'].push(conversation) | ||||
|     } else if (diff < threeDays) { | ||||
|       groupMap['一天前'].push(conversation) | ||||
|     } else if (diff < sevenDays) { | ||||
|       groupMap['三天前'].push(conversation) | ||||
|     } else if (diff < thirtyDays) { | ||||
|       groupMap['七天前'].push(conversation) | ||||
|     } else { | ||||
|       groupMap['三十天前'].push(conversation) | ||||
|     } | ||||
|   } | ||||
|   return groupMap | ||||
| } | ||||
|  | ||||
| /** 新建对话 */ | ||||
| const createConversation = async () => { | ||||
|   // 1. 新建对话 | ||||
|   const conversationId = await ChatConversationApi.createChatConversationMy( | ||||
|     {} as unknown as ChatConversationVO | ||||
|   ) | ||||
|   // 2. 获取对话内容 | ||||
|   await getChatConversationList() | ||||
|   // 3. 选中对话 | ||||
|   await handleConversationClick(conversationId) | ||||
|   // 4. 回调 | ||||
|   emits('onConversationCreate') | ||||
| } | ||||
|  | ||||
| /** 修改对话的标题 */ | ||||
| const updateConversationTitle = async (conversation: ChatConversationVO) => { | ||||
|   // 1. 二次确认 | ||||
|   const { value } = await ElMessageBox.prompt('修改标题', { | ||||
|     inputPattern: /^[\s\S]*.*\S[\s\S]*$/, // 判断非空,且非空格 | ||||
|     inputErrorMessage: '标题不能为空', | ||||
|     inputValue: conversation.title | ||||
|   }) | ||||
|   // 2. 发起修改 | ||||
|   await ChatConversationApi.updateChatConversationMy({ | ||||
|     id: conversation.id, | ||||
|     title: value | ||||
|   } as ChatConversationVO) | ||||
|   message.success('重命名成功') | ||||
|   // 3. 刷新列表 | ||||
|   await getChatConversationList() | ||||
|   // 4. 过滤当前切换的 | ||||
|   const filterConversationList = conversationList.value.filter((item) => { | ||||
|     return item.id === conversation.id | ||||
|   }) | ||||
|   if (filterConversationList.length > 0) { | ||||
|     // tip:避免切换对话 | ||||
|     if (activeConversationId.value === filterConversationList[0].id) { | ||||
|       emits('onConversationClick', filterConversationList[0]) | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 删除聊天对话 */ | ||||
| const deleteChatConversation = async (conversation: ChatConversationVO) => { | ||||
|   try { | ||||
|     // 删除的二次确认 | ||||
|     await message.delConfirm(`是否确认删除对话 - ${conversation.title}?`) | ||||
|     // 发起删除 | ||||
|     await ChatConversationApi.deleteChatConversationMy(conversation.id) | ||||
|     message.success('对话已删除') | ||||
|     // 刷新列表 | ||||
|     await getChatConversationList() | ||||
|     // 回调 | ||||
|     emits('onConversationDelete', conversation) | ||||
|   } catch {} | ||||
| } | ||||
|  | ||||
| /** 清空对话 */ | ||||
| const handleClearConversation = async () => { | ||||
|   try { | ||||
|     await message.confirm('确认后对话会全部清空,置顶的对话除外。') | ||||
|     await ChatConversationApi.deleteChatConversationMyByUnpinned() | ||||
|     ElMessage({ | ||||
|       message: '操作成功!', | ||||
|       type: 'success' | ||||
|     }) | ||||
|     // 清空 对话 和 对话内容 | ||||
|     activeConversationId.value = null | ||||
|     // 获取 对话列表 | ||||
|     await getChatConversationList() | ||||
|     // 回调 方法 | ||||
|     emits('onConversationClear') | ||||
|   } catch {} | ||||
| } | ||||
|  | ||||
| /** 对话置顶 */ | ||||
| const handleTop = async (conversation: ChatConversationVO) => { | ||||
|   // 更新对话置顶 | ||||
|   conversation.pinned = !conversation.pinned | ||||
|   await ChatConversationApi.updateChatConversationMy(conversation) | ||||
|   // 刷新对话 | ||||
|   await getChatConversationList() | ||||
| } | ||||
|  | ||||
| // ============ 角色仓库 ============ | ||||
|  | ||||
| /** 角色仓库抽屉 */ | ||||
| const roleRepositoryOpen = ref<boolean>(false) // 角色仓库是否打开 | ||||
| const handleRoleRepository = async () => { | ||||
|   roleRepositoryOpen.value = !roleRepositoryOpen.value | ||||
| } | ||||
|  | ||||
| /** 监听选中的对话 */ | ||||
| const { activeId } = toRefs(props) | ||||
| watch(activeId, async (newValue, oldValue) => { | ||||
|   activeConversationId.value = newValue as string | ||||
| }) | ||||
|  | ||||
| // 定义 public 方法 | ||||
| defineExpose({ createConversation }) | ||||
|  | ||||
| /** 初始化 */ | ||||
| onMounted(async () => { | ||||
|   // 获取 对话列表 | ||||
|   await getChatConversationList() | ||||
|   // 默认选中 | ||||
|   if (props.activeId) { | ||||
|     activeConversationId.value = props.activeId | ||||
|   } else { | ||||
|     // 首次默认选中第一个 | ||||
|     if (conversationList.value.length) { | ||||
|       activeConversationId.value = conversationList.value[0].id | ||||
|       // 回调 onConversationClick | ||||
|       await emits('onConversationClick', conversationList.value[0]) | ||||
|     } | ||||
|   } | ||||
| }) | ||||
| </script> | ||||
|  | ||||
| <style scoped lang="scss"> | ||||
| .conversation-container { | ||||
|   position: relative; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   justify-content: space-between; | ||||
|   padding: 10px 10px 0; | ||||
|   overflow: hidden; | ||||
|  | ||||
|   .btn-new-conversation { | ||||
|     padding: 18px 0; | ||||
|   } | ||||
|  | ||||
|   .search-input { | ||||
|     margin-top: 20px; | ||||
|   } | ||||
|  | ||||
|   .conversation-list { | ||||
|     overflow: auto; | ||||
|     height: 100%; | ||||
|  | ||||
|     .classify-title { | ||||
|       padding-top: 10px; | ||||
|     } | ||||
|  | ||||
|     .conversation-item { | ||||
|       margin-top: 5px; | ||||
|     } | ||||
|  | ||||
|     .conversation { | ||||
|       display: flex; | ||||
|       flex-direction: row; | ||||
|       justify-content: space-between; | ||||
|       flex: 1; | ||||
|       padding: 0 5px; | ||||
|       cursor: pointer; | ||||
|       border-radius: 5px; | ||||
|       align-items: center; | ||||
|       line-height: 30px; | ||||
|  | ||||
|       &.active { | ||||
|         background-color: #e6e6e6; | ||||
|  | ||||
|         .button { | ||||
|           display: inline-block; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .title-wrapper { | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         align-items: center; | ||||
|       } | ||||
|  | ||||
|       .title { | ||||
|         padding: 2px 10px; | ||||
|         max-width: 220px; | ||||
|         font-size: 14px; | ||||
|         font-weight: 400; | ||||
|         color: rgba(0, 0, 0, 0.77); | ||||
|         overflow: hidden; | ||||
|         white-space: nowrap; | ||||
|         text-overflow: ellipsis; | ||||
|       } | ||||
|  | ||||
|       .avatar { | ||||
|         width: 25px; | ||||
|         height: 25px; | ||||
|         border-radius: 5px; | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         justify-items: center; | ||||
|       } | ||||
|  | ||||
|       // 对话编辑、删除 | ||||
|       .button-wrapper { | ||||
|         right: 2px; | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         justify-items: center; | ||||
|         color: #606266; | ||||
|  | ||||
|         .btn { | ||||
|           margin: 0; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // 角色仓库、清空未设置对话 | ||||
|   .tool-box { | ||||
|     position: absolute; | ||||
|     bottom: 0; | ||||
|     left: 0; | ||||
|     right: 0; | ||||
|     //width: 100%; | ||||
|     padding: 0 20px; | ||||
|     background-color: #f4f4f4; | ||||
|     box-shadow: 0 0 1px 1px rgba(228, 228, 228, 0.8); | ||||
|     line-height: 35px; | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     align-items: center; | ||||
|     color: var(--el-text-color); | ||||
|  | ||||
|     > div { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       color: #606266; | ||||
|       padding: 0; | ||||
|       margin: 0; | ||||
|       cursor: pointer; | ||||
|  | ||||
|       > span { | ||||
|         margin-left: 5px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @@ -0,0 +1,148 @@ | ||||
| <template> | ||||
|   <Dialog title="设定" v-model="dialogVisible"> | ||||
|     <el-form | ||||
|       ref="formRef" | ||||
|       :model="formData" | ||||
|       :rules="formRules" | ||||
|       label-width="130px" | ||||
|       v-loading="formLoading" | ||||
|     > | ||||
|       <el-form-item label="角色设定" prop="systemMessage"> | ||||
|         <el-input | ||||
|           type="textarea" | ||||
|           v-model="formData.systemMessage" | ||||
|           :rows="4" | ||||
|           placeholder="请输入角色设定" | ||||
|         /> | ||||
|       </el-form-item> | ||||
|       <el-form-item label="模型" prop="modelId"> | ||||
|         <el-select v-model="formData.modelId" placeholder="请选择模型"> | ||||
|           <el-option | ||||
|             v-for="model in models" | ||||
|             :key="model.id" | ||||
|             :label="model.name" | ||||
|             :value="model.id" | ||||
|           /> | ||||
|         </el-select> | ||||
|       </el-form-item> | ||||
|       <el-form-item label="温度参数" prop="temperature"> | ||||
|         <el-input-number | ||||
|           v-model="formData.temperature" | ||||
|           placeholder="请输入温度参数" | ||||
|           :min="0" | ||||
|           :max="2" | ||||
|           :precision="2" | ||||
|           class="!w-1/1" | ||||
|         /> | ||||
|       </el-form-item> | ||||
|       <el-form-item label="回复数 Token 数" prop="maxTokens"> | ||||
|         <el-input-number | ||||
|           v-model="formData.maxTokens" | ||||
|           placeholder="请输入回复数 Token 数" | ||||
|           :min="0" | ||||
|           :max="8192" | ||||
|           class="!w-1/1" | ||||
|         /> | ||||
|       </el-form-item> | ||||
|       <el-form-item label="上下文数量" prop="maxContexts"> | ||||
|         <el-input-number | ||||
|           v-model="formData.maxContexts" | ||||
|           placeholder="请输入上下文数量" | ||||
|           :min="0" | ||||
|           :max="20" | ||||
|           class="!w-1/1" | ||||
|         /> | ||||
|       </el-form-item> | ||||
|     </el-form> | ||||
|     <template #footer> | ||||
|       <el-button @click="submitForm" type="primary" :disabled="formLoading">确 定</el-button> | ||||
|       <el-button @click="dialogVisible = false">取 消</el-button> | ||||
|     </template> | ||||
|   </Dialog> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import { ModelApi, ModelVO } from '@/api/ai/model/model' | ||||
| import { ChatConversationApi, ChatConversationVO } from '@/api/ai/chat/conversation' | ||||
| import { AiModelTypeEnum } from '@/views/ai/utils/constants' | ||||
|  | ||||
| /** AI 聊天对话的更新表单 */ | ||||
| defineOptions({ name: 'ChatConversationUpdateForm' }) | ||||
|  | ||||
| const message = useMessage() // 消息弹窗 | ||||
|  | ||||
| const dialogVisible = ref(false) // 弹窗的是否展示 | ||||
| const formLoading = ref(false) // 表单的加载中:1)修改时的数据加载;2)提交的按钮禁用 | ||||
| const formData = ref({ | ||||
|   id: undefined, | ||||
|   systemMessage: undefined, | ||||
|   modelId: undefined, | ||||
|   temperature: undefined, | ||||
|   maxTokens: undefined, | ||||
|   maxContexts: undefined | ||||
| }) | ||||
| const formRules = reactive({ | ||||
|   modelId: [{ required: true, message: '模型不能为空', trigger: 'blur' }], | ||||
|   status: [{ required: true, message: '状态不能为空', trigger: 'blur' }], | ||||
|   temperature: [{ required: true, message: '温度参数不能为空', trigger: 'blur' }], | ||||
|   maxTokens: [{ required: true, message: '回复数 Token 数不能为空', trigger: 'blur' }], | ||||
|   maxContexts: [{ required: true, message: '上下文数量不能为空', trigger: 'blur' }] | ||||
| }) | ||||
| const formRef = ref() // 表单 Ref | ||||
| const models = ref([] as ModelVO[]) // 聊天模型列表 | ||||
|  | ||||
| /** 打开弹窗 */ | ||||
| const open = async (id: number) => { | ||||
|   dialogVisible.value = true | ||||
|   resetForm() | ||||
|   // 修改时,设置数据 | ||||
|   if (id) { | ||||
|     formLoading.value = true | ||||
|     try { | ||||
|       const data = await ChatConversationApi.getChatConversationMy(id) | ||||
|       formData.value = Object.keys(formData.value).reduce((obj, key) => { | ||||
|         if (data.hasOwnProperty(key)) { | ||||
|           obj[key] = data[key] | ||||
|         } | ||||
|         return obj | ||||
|       }, {}) | ||||
|     } finally { | ||||
|       formLoading.value = false | ||||
|     } | ||||
|   } | ||||
|   // 获得下拉数据 | ||||
|   models.value = await ModelApi.getModelSimpleList(AiModelTypeEnum.CHAT) | ||||
| } | ||||
| defineExpose({ open }) // 提供 open 方法,用于打开弹窗 | ||||
|  | ||||
| /** 提交表单 */ | ||||
| const emit = defineEmits(['success']) // 定义 success 事件,用于操作成功后的回调 | ||||
| const submitForm = async () => { | ||||
|   // 校验表单 | ||||
|   await formRef.value.validate() | ||||
|   // 提交请求 | ||||
|   formLoading.value = true | ||||
|   try { | ||||
|     const data = formData.value as unknown as ChatConversationVO | ||||
|     await ChatConversationApi.updateChatConversationMy(data) | ||||
|     message.success('对话配置已更新') | ||||
|     dialogVisible.value = false | ||||
|     // 发送操作成功的事件 | ||||
|     emit('success') | ||||
|   } finally { | ||||
|     formLoading.value = false | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 重置表单 */ | ||||
| const resetForm = () => { | ||||
|   formData.value = { | ||||
|     id: undefined, | ||||
|     systemMessage: undefined, | ||||
|     modelId: undefined, | ||||
|     temperature: undefined, | ||||
|     maxTokens: undefined, | ||||
|     maxContexts: undefined | ||||
|   } | ||||
|   formRef.value?.resetFields() | ||||
| } | ||||
| </script> | ||||
							
								
								
									
										104
									
								
								src/views/ai/chat/index/components/message/MessageKnowledge.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										104
									
								
								src/views/ai/chat/index/components/message/MessageKnowledge.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,104 @@ | ||||
| <!-- 知识引用组件 --> | ||||
| <template> | ||||
|   <!-- 知识引用列表 --> | ||||
|   <div v-if="segments && segments.length > 0" class="mt-10px p-10px rounded-8px bg-[#f5f5f5]"> | ||||
|     <div class="text-14px text-[#666] mb-8px flex items-center"> | ||||
|       <Icon icon="ep:document" class="mr-5px" /> 知识引用 | ||||
|     </div> | ||||
|     <div class="flex flex-wrap gap-8px"> | ||||
|       <div | ||||
|         v-for="(doc, index) in documentList" | ||||
|         :key="index" | ||||
|         class="p-8px px-12px bg-white rounded-6px cursor-pointer transition-all hover:bg-[#e6f4ff]" | ||||
|         @click="handleClick(doc)" | ||||
|       > | ||||
|         <div class="text-14px text-[#333] mb-4px"> | ||||
|           {{ doc.title }} | ||||
|           <span class="text-12px text-[#999] ml-4px">({{ doc.segments.length }} 条)</span> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|  | ||||
|   <!-- 知识引用详情弹窗 --> | ||||
|   <el-popover | ||||
|     v-model:visible="dialogVisible" | ||||
|     :width="600" | ||||
|     trigger="click" | ||||
|     placement="top-start" | ||||
|     :offset="55" | ||||
|     popper-class="knowledge-popover" | ||||
|   > | ||||
|     <template #reference> | ||||
|       <div ref="documentRef"></div> | ||||
|     </template> | ||||
|     <template #default> | ||||
|       <div class="text-16px font-bold mb-12px">{{ document?.title }}</div> | ||||
|       <div class="max-h-[60vh] overflow-y-auto"> | ||||
|         <div | ||||
|           v-for="(segment, index) in document?.segments" | ||||
|           :key="index" | ||||
|           class="p-12px border-b-solid border-b-[#eee] last:border-b-0" | ||||
|         > | ||||
|           <div | ||||
|             class="block mb-8px px-8px py-2px bg-[#f5f5f5] rounded-4px text-12px text-[#666] w-fit" | ||||
|           > | ||||
|             分段 {{ segment.id }} | ||||
|           </div> | ||||
|           <div class="text-14px leading-[1.6] text-[#333] mt-[10px]"> | ||||
|             {{ segment.content }} | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </template> | ||||
|   </el-popover> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| const props = defineProps<{ | ||||
|   segments: { | ||||
|     id: number | ||||
|     documentId: number | ||||
|     documentName: string | ||||
|     content: string | ||||
|   }[] | ||||
| }>() | ||||
|  | ||||
| const document = ref<{ | ||||
|   id: number | ||||
|   title: string | ||||
|   segments: { | ||||
|     id: number | ||||
|     content: string | ||||
|   }[] | ||||
| } | null>(null) // 知识库文档列表 | ||||
| const dialogVisible = ref(false) // 知识引用详情弹窗 | ||||
| const documentRef = ref<HTMLElement>() // 知识引用详情弹窗 Ref | ||||
|  | ||||
| /** 按照 document 聚合 segments */ | ||||
| const documentList = computed(() => { | ||||
|   if (!props.segments) return [] | ||||
|  | ||||
|   const docMap = new Map() | ||||
|   props.segments.forEach((segment) => { | ||||
|     if (!docMap.has(segment.documentId)) { | ||||
|       docMap.set(segment.documentId, { | ||||
|         id: segment.documentId, | ||||
|         title: segment.documentName, | ||||
|         segments: [] | ||||
|       }) | ||||
|     } | ||||
|     docMap.get(segment.documentId).segments.push({ | ||||
|       id: segment.id, | ||||
|       content: segment.content | ||||
|     }) | ||||
|   }) | ||||
|   return Array.from(docMap.values()) | ||||
| }) | ||||
|  | ||||
| /** 点击 document 处理 */ | ||||
| const handleClick = (doc: any) => { | ||||
|   document.value = doc | ||||
|   dialogVisible.value = true | ||||
| } | ||||
| </script> | ||||
							
								
								
									
										284
									
								
								src/views/ai/chat/index/components/message/MessageList.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										284
									
								
								src/views/ai/chat/index/components/message/MessageList.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,284 @@ | ||||
| <template> | ||||
|   <div ref="messageContainer" class="h-100% overflow-y-auto relative"> | ||||
|     <div class="chat-list" v-for="(item, index) in list" :key="index"> | ||||
|       <!-- 靠左 message:system、assistant 类型 --> | ||||
|       <div class="left-message message-item" v-if="item.type !== 'user'"> | ||||
|         <div class="avatar"> | ||||
|           <el-avatar :src="roleAvatar" /> | ||||
|         </div> | ||||
|         <div class="message"> | ||||
|           <div> | ||||
|             <el-text class="time">{{ formatDate(item.createTime) }}</el-text> | ||||
|           </div> | ||||
|           <div class="left-text-container" ref="markdownViewRef"> | ||||
|             <MarkdownView class="left-text" :content="item.content" /> | ||||
|             <MessageKnowledge v-if="item.segments" :segments="item.segments" /> | ||||
|           </div> | ||||
|           <div class="left-btns"> | ||||
|             <el-button class="btn-cus" link @click="copyContent(item.content)"> | ||||
|               <img class="btn-image" src="@/assets/ai/copy.svg" /> | ||||
|             </el-button> | ||||
|             <el-button v-if="item.id > 0" class="btn-cus" link @click="onDelete(item.id)"> | ||||
|               <img class="btn-image h-17px" src="@/assets/ai/delete.svg" /> | ||||
|             </el-button> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|       <!-- 靠右 message:user 类型 --> | ||||
|       <div class="right-message message-item" v-if="item.type === 'user'"> | ||||
|         <div class="avatar"> | ||||
|           <el-avatar :src="userAvatar" /> | ||||
|         </div> | ||||
|         <div class="message"> | ||||
|           <div> | ||||
|             <el-text class="time">{{ formatDate(item.createTime) }}</el-text> | ||||
|           </div> | ||||
|           <div class="right-text-container"> | ||||
|             <div class="right-text">{{ item.content }}</div> | ||||
|           </div> | ||||
|           <div class="right-btns"> | ||||
|             <el-button class="btn-cus" link @click="copyContent(item.content)"> | ||||
|               <img class="btn-image" src="@/assets/ai/copy.svg" /> | ||||
|             </el-button> | ||||
|             <el-button class="btn-cus" link @click="onDelete(item.id)"> | ||||
|               <img class="btn-image h-17px mr-12px" src="@/assets/ai/delete.svg" /> | ||||
|             </el-button> | ||||
|             <el-button class="btn-cus" link @click="onRefresh(item)"> | ||||
|               <el-icon size="17"><RefreshRight /></el-icon> | ||||
|             </el-button> | ||||
|             <el-button class="btn-cus" link @click="onEdit(item)"> | ||||
|               <el-icon size="17"><Edit /></el-icon> | ||||
|             </el-button> | ||||
|           </div> | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
|   <!-- 回到底部 --> | ||||
|   <div v-if="isScrolling" class="to-bottom" @click="handleGoBottom"> | ||||
|     <el-button :icon="ArrowDownBold" circle /> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import { PropType } from 'vue' | ||||
| import { formatDate } from '@/utils/formatTime' | ||||
| import MarkdownView from '@/components/MarkdownView/index.vue' | ||||
| import MessageKnowledge from './MessageKnowledge.vue' | ||||
| import { useClipboard } from '@vueuse/core' | ||||
| import { ArrowDownBold, Edit, RefreshRight } from '@element-plus/icons-vue' | ||||
| import { ChatMessageApi, ChatMessageVO } from '@/api/ai/chat/message' | ||||
| import { ChatConversationVO } from '@/api/ai/chat/conversation' | ||||
| import { useUserStore } from '@/store/modules/user' | ||||
| import userAvatarDefaultImg from '@/assets/imgs/avatar.gif' | ||||
| import roleAvatarDefaultImg from '@/assets/ai/gpt.svg' | ||||
|  | ||||
| const message = useMessage() // 消息弹窗 | ||||
| const { copy } = useClipboard() // 初始化 copy 到粘贴板 | ||||
| const userStore = useUserStore() | ||||
|  | ||||
| // 判断“消息列表”滚动的位置(用于判断是否需要滚动到消息最下方) | ||||
| const messageContainer: any = ref(null) | ||||
| const isScrolling = ref(false) //用于判断用户是否在滚动 | ||||
|  | ||||
| const userAvatar = computed(() => userStore.user.avatar || userAvatarDefaultImg) | ||||
| const roleAvatar = computed(() => props.conversation.roleAvatar ?? roleAvatarDefaultImg) | ||||
|  | ||||
| // 定义 props | ||||
| const props = defineProps({ | ||||
|   conversation: { | ||||
|     type: Object as PropType<ChatConversationVO>, | ||||
|     required: true | ||||
|   }, | ||||
|   list: { | ||||
|     type: Array as PropType<ChatMessageVO[]>, | ||||
|     required: true | ||||
|   } | ||||
| }) | ||||
|  | ||||
| const { list } = toRefs(props) // 消息列表 | ||||
|  | ||||
| const emits = defineEmits(['onDeleteSuccess', 'onRefresh', 'onEdit']) // 定义 emits | ||||
|  | ||||
| // ============ 处理对话滚动 ============== | ||||
|  | ||||
| /** 滚动到底部 */ | ||||
| const scrollToBottom = async (isIgnore?: boolean) => { | ||||
|   // 注意要使用 nextTick 以免获取不到 dom | ||||
|   await nextTick() | ||||
|   if (isIgnore || !isScrolling.value) { | ||||
|     messageContainer.value.scrollTop = | ||||
|       messageContainer.value.scrollHeight - messageContainer.value.offsetHeight | ||||
|   } | ||||
| } | ||||
|  | ||||
| function handleScroll() { | ||||
|   const scrollContainer = messageContainer.value | ||||
|   const scrollTop = scrollContainer.scrollTop | ||||
|   const scrollHeight = scrollContainer.scrollHeight | ||||
|   const offsetHeight = scrollContainer.offsetHeight | ||||
|   if (scrollTop + offsetHeight < scrollHeight - 100) { | ||||
|     // 用户开始滚动并在最底部之上,取消保持在最底部的效果 | ||||
|     isScrolling.value = true | ||||
|   } else { | ||||
|     // 用户停止滚动并滚动到最底部,开启保持到最底部的效果 | ||||
|     isScrolling.value = false | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 回到底部 */ | ||||
| const handleGoBottom = async () => { | ||||
|   const scrollContainer = messageContainer.value | ||||
|   scrollContainer.scrollTop = scrollContainer.scrollHeight | ||||
| } | ||||
|  | ||||
| /** 回到顶部 */ | ||||
| const handlerGoTop = async () => { | ||||
|   const scrollContainer = messageContainer.value | ||||
|   scrollContainer.scrollTop = 0 | ||||
| } | ||||
|  | ||||
| defineExpose({ scrollToBottom, handlerGoTop }) // 提供方法给 parent 调用 | ||||
|  | ||||
| // ============ 处理消息操作 ============== | ||||
|  | ||||
| /** 复制 */ | ||||
| const copyContent = async (content) => { | ||||
|   await copy(content) | ||||
|   message.success('复制成功!') | ||||
| } | ||||
|  | ||||
| /** 删除 */ | ||||
| const onDelete = async (id) => { | ||||
|   // 删除 message | ||||
|   await ChatMessageApi.deleteChatMessage(id) | ||||
|   message.success('删除成功!') | ||||
|   // 回调 | ||||
|   emits('onDeleteSuccess') | ||||
| } | ||||
|  | ||||
| /** 刷新 */ | ||||
| const onRefresh = async (message: ChatMessageVO) => { | ||||
|   emits('onRefresh', message) | ||||
| } | ||||
|  | ||||
| /** 编辑 */ | ||||
| const onEdit = async (message: ChatMessageVO) => { | ||||
|   emits('onEdit', message) | ||||
| } | ||||
|  | ||||
| /** 初始化 */ | ||||
| onMounted(async () => { | ||||
|   messageContainer.value.addEventListener('scroll', handleScroll) | ||||
| }) | ||||
| </script> | ||||
|  | ||||
| <style scoped lang="scss"> | ||||
| .message-container { | ||||
|   position: relative; | ||||
|   overflow-y: scroll; | ||||
| } | ||||
|  | ||||
| // 中间 | ||||
| .chat-list { | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   overflow-y: hidden; | ||||
|   padding: 0 20px; | ||||
|   .message-item { | ||||
|     margin-top: 50px; | ||||
|   } | ||||
|  | ||||
|   .left-message { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|   } | ||||
|  | ||||
|   .right-message { | ||||
|     display: flex; | ||||
|     flex-direction: row-reverse; | ||||
|     justify-content: flex-start; | ||||
|   } | ||||
|  | ||||
|   .message { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     text-align: left; | ||||
|     margin: 0 15px; | ||||
|  | ||||
|     .time { | ||||
|       text-align: left; | ||||
|       line-height: 30px; | ||||
|     } | ||||
|  | ||||
|     .left-text-container { | ||||
|       position: relative; | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       overflow-wrap: break-word; | ||||
|       background-color: rgba(228, 228, 228, 0.8); | ||||
|       box-shadow: 0 0 0 1px rgba(228, 228, 228, 0.8); | ||||
|       border-radius: 10px; | ||||
|       padding: 10px 10px 5px 10px; | ||||
|  | ||||
|       .left-text { | ||||
|         color: #393939; | ||||
|         font-size: 0.95rem; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     .right-text-container { | ||||
|       display: flex; | ||||
|       flex-direction: row-reverse; | ||||
|  | ||||
|       .right-text { | ||||
|         font-size: 0.95rem; | ||||
|         color: #fff; | ||||
|         display: inline; | ||||
|         background-color: #267fff; | ||||
|         box-shadow: 0 0 0 1px #267fff; | ||||
|         border-radius: 10px; | ||||
|         padding: 10px; | ||||
|         width: auto; | ||||
|         overflow-wrap: break-word; | ||||
|         white-space: pre-wrap; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     .left-btns { | ||||
|       display: flex; | ||||
|       flex-direction: row; | ||||
|       margin-top: 8px; | ||||
|     } | ||||
|  | ||||
|     .right-btns { | ||||
|       display: flex; | ||||
|       flex-direction: row-reverse; | ||||
|       margin-top: 8px; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // 复制、删除按钮 | ||||
|   .btn-cus { | ||||
|     display: flex; | ||||
|     background-color: transparent; | ||||
|     align-items: center; | ||||
|  | ||||
|     .btn-image { | ||||
|       height: 20px; | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   .btn-cus:hover { | ||||
|     cursor: pointer; | ||||
|     background-color: #f6f6f6; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // 回到底部 | ||||
| .to-bottom { | ||||
|   position: absolute; | ||||
|   z-index: 1000; | ||||
|   bottom: 0; | ||||
|   right: 50%; | ||||
| } | ||||
| </style> | ||||
| @@ -0,0 +1,83 @@ | ||||
| <!-- 消息列表为空时,展示 prompt 列表 --> | ||||
| <template> | ||||
|   <div class="chat-empty"> | ||||
|     <!-- title --> | ||||
|     <div class="center-container"> | ||||
|       <div class="title">鹏辰 AI</div> | ||||
|       <div class="role-list"> | ||||
|         <div | ||||
|           class="role-item" | ||||
|           v-for="prompt in promptList" | ||||
|           :key="prompt.prompt" | ||||
|           @click="handlerPromptClick(prompt)" | ||||
|         > | ||||
|           {{ prompt.prompt }} | ||||
|         </div> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| const promptList = [ | ||||
|   { | ||||
|     prompt: '今天气怎么样?' | ||||
|   }, | ||||
|   { | ||||
|     prompt: '写一首好听的诗歌?' | ||||
|   } | ||||
| ] // prompt 列表 | ||||
|  | ||||
| const emits = defineEmits(['onPrompt']) | ||||
|  | ||||
| /** 选中 prompt 点击 */ | ||||
| const handlerPromptClick = async ({ prompt }) => { | ||||
|   emits('onPrompt', prompt) | ||||
| } | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| .chat-empty { | ||||
|   position: relative; | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   justify-content: center; | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|  | ||||
|   .center-container { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     justify-content: center; | ||||
|  | ||||
|     .title { | ||||
|       font-size: 28px; | ||||
|       font-weight: bold; | ||||
|       text-align: center; | ||||
|     } | ||||
|  | ||||
|     .role-list { | ||||
|       display: flex; | ||||
|       flex-direction: row; | ||||
|       flex-wrap: wrap; | ||||
|       align-items: center; | ||||
|       justify-content: center; | ||||
|       width: 460px; | ||||
|       margin-top: 20px; | ||||
|  | ||||
|       .role-item { | ||||
|         display: flex; | ||||
|         justify-content: center; | ||||
|         width: 180px; | ||||
|         line-height: 50px; | ||||
|         border: 1px solid #e4e4e4; | ||||
|         border-radius: 10px; | ||||
|         margin: 10px; | ||||
|         cursor: pointer; | ||||
|       } | ||||
|  | ||||
|       .role-item:hover { | ||||
|         background-color: rgba(243, 243, 243, 0.73); | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
| @@ -0,0 +1,15 @@ | ||||
| <!-- message 加载页面 --> | ||||
| <template> | ||||
|   <div class="message-loading" > | ||||
|     <el-skeleton animated /> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
|  | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| .message-loading { | ||||
|   padding: 30px 30px; | ||||
| } | ||||
| </style> | ||||
| @@ -0,0 +1,46 @@ | ||||
| <!-- 无聊天对话时,在 message 区域,可以新增对话 --> | ||||
| <template> | ||||
|   <div class="new-chat"> | ||||
|     <div class="box-center"> | ||||
|       <div class="tip">点击下方按钮,开始你的对话吧</div> | ||||
|       <div class="btns"> | ||||
|         <el-button type="primary" round @click="handlerNewChat">新建对话</el-button> | ||||
|       </div> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| const emits = defineEmits(['onNewConversation']) | ||||
|  | ||||
| /** 新建 conversation 聊天对话 */ | ||||
| const handlerNewChat = () => { | ||||
|   emits('onNewConversation') | ||||
| } | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| .new-chat { | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   justify-content: center; | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|  | ||||
|   .box-center { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     justify-content: center; | ||||
|  | ||||
|     .tip { | ||||
|       font-size: 14px; | ||||
|       color: #858585; | ||||
|     } | ||||
|  | ||||
|     .btns { | ||||
|       display: flex; | ||||
|       flex-direction: row; | ||||
|       justify-content: center; | ||||
|       margin-top: 20px; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										53
									
								
								src/views/ai/chat/index/components/role/RoleCategoryList.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										53
									
								
								src/views/ai/chat/index/components/role/RoleCategoryList.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,53 @@ | ||||
| <template> | ||||
|   <div class="category-list"> | ||||
|     <div class="category" v-for="category in categoryList" :key="category"> | ||||
|       <el-button | ||||
|         plain | ||||
|         round | ||||
|         size="small" | ||||
|         :type="category === active ? 'primary' : ''" | ||||
|         @click="handleCategoryClick(category)" | ||||
|       > | ||||
|         {{ category }} | ||||
|       </el-button> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
| <script setup lang="ts"> | ||||
| import { PropType } from 'vue' | ||||
|  | ||||
| // 定义属性 | ||||
| defineProps({ | ||||
|   categoryList: { | ||||
|     type: Array as PropType<string[]>, | ||||
|     required: true | ||||
|   }, | ||||
|   active: { | ||||
|     type: String, | ||||
|     required: false, | ||||
|     default: '全部' | ||||
|   } | ||||
| }) | ||||
|  | ||||
| // 定义回调 | ||||
| const emits = defineEmits(['onCategoryClick']) | ||||
|  | ||||
| /** 处理分类点击事件 */ | ||||
| const handleCategoryClick = async (category: string) => { | ||||
|   emits('onCategoryClick', category) | ||||
| } | ||||
| </script> | ||||
| <style scoped lang="scss"> | ||||
| .category-list { | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   flex-wrap: wrap; | ||||
|   align-items: center; | ||||
|  | ||||
|   .category { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     margin-right: 10px; | ||||
|   } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										48
									
								
								src/views/ai/chat/index/components/role/RoleHeader.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										48
									
								
								src/views/ai/chat/index/components/role/RoleHeader.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,48 @@ | ||||
| <!-- header --> | ||||
| <template> | ||||
|   <el-header class="chat-header"> | ||||
|     <div class="title"> | ||||
|       {{ title }} | ||||
|     </div> | ||||
|     <div class="title-right"> | ||||
|       <slot></slot> | ||||
|     </div> | ||||
|   </el-header> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| // 设置组件属性 | ||||
| defineProps({ | ||||
|   title: { | ||||
|     type: String, | ||||
|     required: true | ||||
|   } | ||||
| }) | ||||
| </script> | ||||
|  | ||||
| <style scoped lang="scss"> | ||||
| .chat-header { | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   justify-content: space-between; | ||||
|   align-items: center; | ||||
|   padding: 0 10px; | ||||
|   white-space: nowrap; | ||||
|   text-overflow: ellipsis; | ||||
|   background-color: #ececec; | ||||
|   width: 100%; | ||||
|  | ||||
|   .title { | ||||
|     font-size: 20px; | ||||
|     font-weight: bold; | ||||
|     overflow: hidden; | ||||
|     color: #3e3e3e; | ||||
|     max-width: 220px; | ||||
|   } | ||||
|  | ||||
|   .title-right { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|   } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										174
									
								
								src/views/ai/chat/index/components/role/RoleList.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										174
									
								
								src/views/ai/chat/index/components/role/RoleList.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,174 @@ | ||||
| <template> | ||||
|   <div class="card-list" ref="tabsRef" @scroll="handleTabsScroll"> | ||||
|     <div class="card-item" v-for="role in roleList" :key="role.id"> | ||||
|       <el-card class="card" body-class="card-body"> | ||||
|         <!-- 更多操作 --> | ||||
|         <div class="more-container" v-if="showMore"> | ||||
|           <el-dropdown @command="handleMoreClick"> | ||||
|             <span class="el-dropdown-link"> | ||||
|               <el-button type="text"> | ||||
|                 <el-icon><More /></el-icon> | ||||
|               </el-button> | ||||
|             </span> | ||||
|             <template #dropdown> | ||||
|               <el-dropdown-menu> | ||||
|                 <el-dropdown-item :command="['edit', role]"> | ||||
|                   <Icon icon="ep:edit" color="#787878" />编辑 | ||||
|                 </el-dropdown-item> | ||||
|                 <el-dropdown-item :command="['delete', role]" style="color: red"> | ||||
|                   <Icon icon="ep:delete" color="red" />删除 | ||||
|                 </el-dropdown-item> | ||||
|               </el-dropdown-menu> | ||||
|             </template> | ||||
|           </el-dropdown> | ||||
|         </div> | ||||
|         <!-- 角色信息 --> | ||||
|         <div> | ||||
|           <img class="avatar" :src="role.avatar" /> | ||||
|         </div> | ||||
|         <div class="right-container"> | ||||
|           <div class="content-container"> | ||||
|             <div class="title">{{ role.name }}</div> | ||||
|             <div class="description">{{ role.description }}</div> | ||||
|           </div> | ||||
|           <div class="btn-container"> | ||||
|             <el-button type="primary" size="small" @click="handleUseClick(role)">使用</el-button> | ||||
|           </div> | ||||
|         </div> | ||||
|       </el-card> | ||||
|     </div> | ||||
|   </div> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { ChatRoleVO } from '@/api/ai/model/chatRole' | ||||
| import { PropType, ref } from 'vue' | ||||
| import { More } from '@element-plus/icons-vue' | ||||
|  | ||||
| const tabsRef = ref<any>() // tabs ref | ||||
|  | ||||
| // 定义属性 | ||||
| const props = defineProps({ | ||||
|   loading: { | ||||
|     type: Boolean, | ||||
|     required: true | ||||
|   }, | ||||
|   roleList: { | ||||
|     type: Array as PropType<ChatRoleVO[]>, | ||||
|     required: true | ||||
|   }, | ||||
|   showMore: { | ||||
|     type: Boolean, | ||||
|     required: false, | ||||
|     default: false | ||||
|   } | ||||
| }) | ||||
|  | ||||
| // 定义钩子 | ||||
| const emits = defineEmits(['onDelete', 'onEdit', 'onUse', 'onPage']) | ||||
|  | ||||
| /** 操作:编辑、删除 */ | ||||
| const handleMoreClick = async (data) => { | ||||
|   const type = data[0] | ||||
|   const role = data[1] | ||||
|   if (type === 'delete') { | ||||
|     emits('onDelete', role) | ||||
|   } else { | ||||
|     emits('onEdit', role) | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 选中 */ | ||||
| const handleUseClick = (role) => { | ||||
|   emits('onUse', role) | ||||
| } | ||||
|  | ||||
| /** 滚动 */ | ||||
| const handleTabsScroll = async () => { | ||||
|   if (tabsRef.value) { | ||||
|     const { scrollTop, scrollHeight, clientHeight } = tabsRef.value | ||||
|     if (scrollTop + clientHeight >= scrollHeight - 20 && !props.loading) { | ||||
|       await emits('onPage') | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </script> | ||||
|  | ||||
| <style lang="scss"> | ||||
| // 重写 card 组件 body 样式 | ||||
| .card-body { | ||||
|   max-width: 240px; | ||||
|   width: 240px; | ||||
|   padding: 15px 15px 10px 15px; | ||||
|  | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   justify-content: flex-start; | ||||
|   position: relative; | ||||
| } | ||||
| </style> | ||||
| <style scoped lang="scss"> | ||||
| // 卡片列表 | ||||
| .card-list { | ||||
|   display: flex; | ||||
|   flex-direction: row; | ||||
|   flex-wrap: wrap; | ||||
|   position: relative; | ||||
|   height: 100%; | ||||
|   overflow: auto; | ||||
|   padding: 0px 25px; | ||||
|   padding-bottom: 140px; | ||||
|   align-items: start; | ||||
|   align-content: flex-start; | ||||
|   justify-content: start; | ||||
|  | ||||
|   .card { | ||||
|     display: inline-block; | ||||
|     margin-right: 20px; | ||||
|     border-radius: 10px; | ||||
|     margin-bottom: 20px; | ||||
|     position: relative; | ||||
|  | ||||
|     .more-container { | ||||
|       position: absolute; | ||||
|       top: 0; | ||||
|       right: 12px; | ||||
|     } | ||||
|  | ||||
|     .avatar { | ||||
|       width: 40px; | ||||
|       height: 40px; | ||||
|       border-radius: 10px; | ||||
|       overflow: hidden; | ||||
|     } | ||||
|  | ||||
|     .right-container { | ||||
|       margin-left: 10px; | ||||
|       width: 100%; | ||||
|       //height: 100px; | ||||
|  | ||||
|       .content-container { | ||||
|         height: 85px; | ||||
|  | ||||
|         .title { | ||||
|           font-size: 18px; | ||||
|           font-weight: bold; | ||||
|           color: #3e3e3e; | ||||
|         } | ||||
|  | ||||
|         .description { | ||||
|           margin-top: 10px; | ||||
|           font-size: 14px; | ||||
|           color: #6a6a6a; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .btn-container { | ||||
|         display: flex; | ||||
|         flex-direction: row-reverse; | ||||
|         margin-top: 2px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										289
									
								
								src/views/ai/chat/index/components/role/RoleRepository.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										289
									
								
								src/views/ai/chat/index/components/role/RoleRepository.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,289 @@ | ||||
| <!-- chat 角色仓库 --> | ||||
| <template> | ||||
|   <el-container class="role-container"> | ||||
|     <ChatRoleForm ref="formRef" @success="handlerAddRoleSuccess" /> | ||||
|     <!-- header  --> | ||||
|     <RoleHeader title="角色仓库" class="relative" /> | ||||
|     <!--  main  --> | ||||
|     <el-main class="role-main"> | ||||
|       <div class="search-container"> | ||||
|         <!-- 搜索按钮 --> | ||||
|         <el-input | ||||
|           :loading="loading" | ||||
|           v-model="search" | ||||
|           class="search-input" | ||||
|           size="default" | ||||
|           placeholder="请输入搜索的内容" | ||||
|           :suffix-icon="Search" | ||||
|           @change="getActiveTabsRole" | ||||
|         /> | ||||
|         <el-button | ||||
|           v-if="activeTab == 'my-role'" | ||||
|           type="primary" | ||||
|           @click="handlerAddRole" | ||||
|           class="ml-20px" | ||||
|         > | ||||
|           <Icon icon="ep:user" style="margin-right: 5px;" /> | ||||
|           添加角色 | ||||
|         </el-button> | ||||
|       </div> | ||||
|       <!-- tabs --> | ||||
|       <el-tabs v-model="activeTab" class="tabs" @tab-click="handleTabsClick"> | ||||
|         <el-tab-pane class="role-pane" label="我的角色" name="my-role"> | ||||
|           <RoleList | ||||
|             :loading="loading" | ||||
|             :role-list="myRoleList" | ||||
|             :show-more="true" | ||||
|             @on-delete="handlerCardDelete" | ||||
|             @on-edit="handlerCardEdit" | ||||
|             @on-use="handlerCardUse" | ||||
|             @on-page="handlerCardPage('my')" | ||||
|             class="mt-20px" | ||||
|           /> | ||||
|         </el-tab-pane> | ||||
|         <el-tab-pane label="公共角色" name="public-role"> | ||||
|           <RoleCategoryList | ||||
|             class="role-category-list" | ||||
|             :category-list="categoryList" | ||||
|             :active="activeCategory" | ||||
|             @on-category-click="handlerCategoryClick" | ||||
|           /> | ||||
|           <RoleList | ||||
|             :role-list="publicRoleList" | ||||
|             @on-delete="handlerCardDelete" | ||||
|             @on-edit="handlerCardEdit" | ||||
|             @on-use="handlerCardUse" | ||||
|             @on-page="handlerCardPage('public')" | ||||
|             class="mt-20px" | ||||
|             loading | ||||
|           /> | ||||
|         </el-tab-pane> | ||||
|       </el-tabs> | ||||
|     </el-main> | ||||
|   </el-container> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import {ref} from 'vue' | ||||
| import RoleHeader from './RoleHeader.vue' | ||||
| import RoleList from './RoleList.vue' | ||||
| import ChatRoleForm from '@/views/ai/model/chatRole/ChatRoleForm.vue' | ||||
| import RoleCategoryList from './RoleCategoryList.vue' | ||||
| import {ChatRoleApi, ChatRolePageReqVO, ChatRoleVO} from '@/api/ai/model/chatRole' | ||||
| import {ChatConversationApi, ChatConversationVO} from '@/api/ai/chat/conversation' | ||||
| import {Search} from '@element-plus/icons-vue' | ||||
| import {TabsPaneContext} from 'element-plus' | ||||
|  | ||||
| const router = useRouter() // 路由对象 | ||||
|  | ||||
| // 属性定义 | ||||
| const loading = ref<boolean>(false) // 加载中 | ||||
| const activeTab = ref<string>('my-role') // 选中的角色 Tab | ||||
| const search = ref<string>('') // 加载中 | ||||
| const myRoleParams = reactive({ | ||||
|   pageNo: 1, | ||||
|   pageSize: 50 | ||||
| }) | ||||
| const myRoleList = ref<ChatRoleVO[]>([]) // my 分页大小 | ||||
| const publicRoleParams = reactive({ | ||||
|   pageNo: 1, | ||||
|   pageSize: 50 | ||||
| }) | ||||
| const publicRoleList = ref<ChatRoleVO[]>([]) // public 分页大小 | ||||
| const activeCategory = ref<string>('全部') // 选择中的分类 | ||||
| const categoryList = ref<string[]>([]) // 角色分类类别 | ||||
|  | ||||
| /** tabs 点击 */ | ||||
| const handleTabsClick = async (tab: TabsPaneContext) => { | ||||
|   // 设置切换状态 | ||||
|   activeTab.value = tab.paneName + '' | ||||
|   // 切换的时候重新加载数据 | ||||
|   await getActiveTabsRole() | ||||
| } | ||||
|  | ||||
| /** 获取 my role 我的角色 */ | ||||
| const getMyRole = async (append?: boolean) => { | ||||
|   const params: ChatRolePageReqVO = { | ||||
|     ...myRoleParams, | ||||
|     name: search.value, | ||||
|     publicStatus: false | ||||
|   } | ||||
|   const { list } = await ChatRoleApi.getMyPage(params) | ||||
|   if (append) { | ||||
|     myRoleList.value.push.apply(myRoleList.value, list) | ||||
|   } else { | ||||
|     myRoleList.value = list | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 获取 public role 公共角色 */ | ||||
| const getPublicRole = async (append?: boolean) => { | ||||
|   const params: ChatRolePageReqVO = { | ||||
|     ...publicRoleParams, | ||||
|     category: activeCategory.value === '全部' ? '' : activeCategory.value, | ||||
|     name: search.value, | ||||
|     publicStatus: true | ||||
|   } | ||||
|   const { total, list } = await ChatRoleApi.getMyPage(params) | ||||
|   if (append) { | ||||
|     publicRoleList.value.push.apply(publicRoleList.value, list) | ||||
|   } else { | ||||
|     publicRoleList.value = list | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 获取选中的 tabs 角色 */ | ||||
| const getActiveTabsRole = async () => { | ||||
|   if (activeTab.value === 'my-role') { | ||||
|     myRoleParams.pageNo = 1 | ||||
|     await getMyRole() | ||||
|   } else { | ||||
|     publicRoleParams.pageNo = 1 | ||||
|     await getPublicRole() | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 获取角色分类列表 */ | ||||
| const getRoleCategoryList = async () => { | ||||
|   categoryList.value = ['全部', ...(await ChatRoleApi.getCategoryList())] | ||||
| } | ||||
|  | ||||
| /** 处理分类点击 */ | ||||
| const handlerCategoryClick = async (category: string) => { | ||||
|   // 切换选择的分类 | ||||
|   activeCategory.value = category | ||||
|   // 筛选 | ||||
|   await getActiveTabsRole() | ||||
| } | ||||
|  | ||||
| /** 添加/修改操作 */ | ||||
| const formRef = ref() | ||||
| const handlerAddRole = async () => { | ||||
|   formRef.value.open('my-create', null, '添加角色') | ||||
| } | ||||
| /** 编辑角色 */ | ||||
| const handlerCardEdit = async (role) => { | ||||
|   formRef.value.open('my-update', role.id, '编辑角色') | ||||
| } | ||||
|  | ||||
| /** 添加角色成功 */ | ||||
| const handlerAddRoleSuccess = async (e) => { | ||||
|   // 刷新数据 | ||||
|   await getActiveTabsRole() | ||||
| } | ||||
|  | ||||
| /** 删除角色 */ | ||||
| const handlerCardDelete = async (role) => { | ||||
|   await ChatRoleApi.deleteMy(role.id) | ||||
|   // 刷新数据 | ||||
|   await getActiveTabsRole() | ||||
| } | ||||
|  | ||||
| /** 角色分页:获取下一页 */ | ||||
| const handlerCardPage = async (type) => { | ||||
|   try { | ||||
|     loading.value = true | ||||
|     if (type === 'public') { | ||||
|       publicRoleParams.pageNo++ | ||||
|       await getPublicRole(true) | ||||
|     } else { | ||||
|       myRoleParams.pageNo++ | ||||
|       await getMyRole(true) | ||||
|     } | ||||
|   } finally { | ||||
|     loading.value = false | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 选择 card 角色:新建聊天对话 */ | ||||
| const handlerCardUse = async (role) => { | ||||
|   // 1. 创建对话 | ||||
|   const data: ChatConversationVO = { | ||||
|     roleId: role.id | ||||
|   } as unknown as ChatConversationVO | ||||
|   const conversationId = await ChatConversationApi.createChatConversationMy(data) | ||||
|  | ||||
|   // 2. 跳转页面 | ||||
|   await router.push({ | ||||
|     name: 'AiChat', | ||||
|     query: { | ||||
|       conversationId: conversationId | ||||
|     } | ||||
|   }) | ||||
| } | ||||
|  | ||||
| /** 初始化 **/ | ||||
| onMounted(async () => { | ||||
|   // 获取分类 | ||||
|   await getRoleCategoryList() | ||||
|   // 获取 role 数据 | ||||
|   await getActiveTabsRole() | ||||
| }) | ||||
| </script> | ||||
| <!-- 覆盖 element ui css --> | ||||
| <style lang="scss"> | ||||
| .el-tabs__content { | ||||
|   position: relative; | ||||
|   height: 100%; | ||||
|   overflow: hidden; | ||||
| } | ||||
| .el-tabs__nav-scroll { | ||||
|   margin: 10px 20px; | ||||
| } | ||||
| </style> | ||||
| <!-- 样式 --> | ||||
| <style scoped lang="scss"> | ||||
| // 跟容器 | ||||
| .role-container { | ||||
|   position: absolute; | ||||
|   width: 100%; | ||||
|   height: 100%; | ||||
|   margin: 0; | ||||
|   padding: 0; | ||||
|   left: 0; | ||||
|   right: 0; | ||||
|   top: 0; | ||||
|   bottom: 0; | ||||
|   background-color: #ffffff; | ||||
|   overflow: hidden; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|  | ||||
|   .role-main { | ||||
|     flex: 1; | ||||
|     overflow: hidden; | ||||
|     margin: 0; | ||||
|     padding: 0; | ||||
|     position: relative; | ||||
|  | ||||
|     .search-container { | ||||
|       margin: 20px 20px 0px 20px; | ||||
|       position: absolute; | ||||
|       right: 0; | ||||
|       top: -5px; | ||||
|       z-index: 100; | ||||
|     } | ||||
|  | ||||
|     .search-input { | ||||
|       width: 240px; | ||||
|     } | ||||
|  | ||||
|     .tabs { | ||||
|       position: relative; | ||||
|       height: 100%; | ||||
|  | ||||
|       .role-category-list { | ||||
|         margin: 0 27px; | ||||
|       } | ||||
|     } | ||||
|  | ||||
|     .role-pane { | ||||
|       display: flex; | ||||
|       flex-direction: column; | ||||
|       height: 100%; | ||||
|       overflow-y: auto; | ||||
|       position: relative; | ||||
|     } | ||||
|   } | ||||
| } | ||||
| </style> | ||||
							
								
								
									
										772
									
								
								src/views/ai/chat/index/index.vue
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										772
									
								
								src/views/ai/chat/index/index.vue
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,772 @@ | ||||
| <template> | ||||
|   <el-container class="ai-layout"> | ||||
|     <!-- 左侧:对话列表 --> | ||||
|     <ConversationList | ||||
|       :active-id="activeConversationId" | ||||
|       ref="conversationListRef" | ||||
|       @on-conversation-create="handleConversationCreateSuccess" | ||||
|       @on-conversation-click="handleConversationClick" | ||||
|       @on-conversation-clear="handleConversationClear" | ||||
|       @on-conversation-delete="handlerConversationDelete" | ||||
|     /> | ||||
|     <!-- 右侧:对话详情 --> | ||||
|     <el-container class="detail-container"> | ||||
|       <el-header class="header"> | ||||
|         <div class="title"> | ||||
|           {{ activeConversation?.title ? activeConversation?.title : '对话' }} | ||||
|           <span v-if="activeMessageList.length">({{ activeMessageList.length }})</span> | ||||
|         </div> | ||||
|         <div class="btns" v-if="activeConversation"> | ||||
|           <el-button type="primary" bg plain size="small" @click="openChatConversationUpdateForm"> | ||||
|             <span v-html="activeConversation?.modelName"></span> | ||||
|             <Icon icon="ep:setting" class="ml-10px" /> | ||||
|           </el-button> | ||||
|           <el-button size="small" class="btn" @click="handlerMessageClear"> | ||||
|             <Icon icon="heroicons-outline:archive-box-x-mark" color="#787878" /> | ||||
|           </el-button> | ||||
|           <el-button size="small" class="btn"> | ||||
|             <Icon icon="ep:download" color="#787878" /> | ||||
|           </el-button> | ||||
|           <el-button size="small" class="btn" @click="handleGoTopMessage"> | ||||
|             <Icon icon="ep:top" color="#787878" /> | ||||
|           </el-button> | ||||
|         </div> | ||||
|       </el-header> | ||||
|  | ||||
|       <!-- main:消息列表 --> | ||||
|       <el-main class="main-container"> | ||||
|         <div> | ||||
|           <div class="message-container"> | ||||
|             <!-- 情况一:消息加载中 --> | ||||
|             <MessageLoading v-if="activeMessageListLoading" /> | ||||
|             <!-- 情况二:无聊天对话时 --> | ||||
|             <MessageNewConversation | ||||
|               v-if="!activeConversation" | ||||
|               @on-new-conversation="handleConversationCreate" | ||||
|             /> | ||||
|             <!-- 情况三:消息列表为空 --> | ||||
|             <MessageListEmpty | ||||
|               v-if="!activeMessageListLoading && messageList.length === 0 && activeConversation" | ||||
|               @on-prompt="doSendMessage" | ||||
|             /> | ||||
|             <!-- 情况四:消息列表不为空 --> | ||||
|             <MessageList | ||||
|               v-if="!activeMessageListLoading && messageList.length > 0" | ||||
|               ref="messageRef" | ||||
|               :conversation="activeConversation" | ||||
|               :list="messageList" | ||||
|               @on-delete-success="handleMessageDelete" | ||||
|               @on-edit="handleMessageEdit" | ||||
|               @on-refresh="handleMessageRefresh" | ||||
|             /> | ||||
|           </div> | ||||
|         </div> | ||||
|       </el-main> | ||||
|  | ||||
|       <!-- 底部 --> | ||||
|       <el-footer class="footer-container"> | ||||
|         <form class="prompt-from"> | ||||
|           <textarea | ||||
|             class="prompt-input" | ||||
|             v-model="prompt" | ||||
|             @keydown="handleSendByKeydown" | ||||
|             @input="handlePromptInput" | ||||
|             @compositionstart="onCompositionstart" | ||||
|             @compositionend="onCompositionend" | ||||
|             placeholder="问我任何问题...(Shift+Enter 换行,按下 Enter 发送)" | ||||
|           ></textarea> | ||||
|           <div class="prompt-btns"> | ||||
|             <div> | ||||
|               <el-switch v-model="enableContext" /> | ||||
|               <span class="ml-5px text-14px text-#8f8f8f">上下文</span> | ||||
|             </div> | ||||
|             <el-button | ||||
|               type="primary" | ||||
|               size="default" | ||||
|               @click="handleSendByButton" | ||||
|               :loading="conversationInProgress" | ||||
|               v-if="conversationInProgress == false" | ||||
|             > | ||||
|               {{ conversationInProgress ? '进行中' : '发送' }} | ||||
|             </el-button> | ||||
|             <el-button | ||||
|               type="danger" | ||||
|               size="default" | ||||
|               @click="stopStream()" | ||||
|               v-if="conversationInProgress == true" | ||||
|             > | ||||
|               停止 | ||||
|             </el-button> | ||||
|           </div> | ||||
|         </form> | ||||
|       </el-footer> | ||||
|     </el-container> | ||||
|  | ||||
|     <!-- 更新对话 Form --> | ||||
|     <ConversationUpdateForm | ||||
|       ref="conversationUpdateFormRef" | ||||
|       @success="handleConversationUpdateSuccess" | ||||
|     /> | ||||
|   </el-container> | ||||
| </template> | ||||
|  | ||||
| <script setup lang="ts"> | ||||
| import { ChatMessageApi, ChatMessageVO } from '@/api/ai/chat/message' | ||||
| import { ChatConversationApi, ChatConversationVO } from '@/api/ai/chat/conversation' | ||||
| import ConversationList from './components/conversation/ConversationList.vue' | ||||
| import ConversationUpdateForm from './components/conversation/ConversationUpdateForm.vue' | ||||
| import MessageList from './components/message/MessageList.vue' | ||||
| import MessageListEmpty from './components/message/MessageListEmpty.vue' | ||||
| import MessageLoading from './components/message/MessageLoading.vue' | ||||
| import MessageNewConversation from './components/message/MessageNewConversation.vue' | ||||
|  | ||||
| /** AI 聊天对话 列表 */ | ||||
| defineOptions({ name: 'AiChat' }) | ||||
|  | ||||
| const route = useRoute() // 路由 | ||||
| const message = useMessage() // 消息弹窗 | ||||
|  | ||||
| // 聊天对话 | ||||
| const conversationListRef = ref() | ||||
| const activeConversationId = ref<number | null>(null) // 选中的对话编号 | ||||
| const activeConversation = ref<ChatConversationVO | null>(null) // 选中的 Conversation | ||||
| const conversationInProgress = ref(false) // 对话是否正在进行中。目前只有【发送】消息时,会更新为 true,避免切换对话、删除对话等操作 | ||||
|  | ||||
| // 消息列表 | ||||
| const messageRef = ref() | ||||
| const activeMessageList = ref<ChatMessageVO[]>([]) // 选中对话的消息列表 | ||||
| const activeMessageListLoading = ref<boolean>(false) // activeMessageList 是否正在加载中 | ||||
| const activeMessageListLoadingTimer = ref<any>() // activeMessageListLoading Timer 定时器。如果加载速度很快,就不进入加载中 | ||||
| // 消息滚动 | ||||
| const textSpeed = ref<number>(50) // Typing speed in milliseconds | ||||
| const textRoleRunning = ref<boolean>(false) // Typing speed in milliseconds | ||||
|  | ||||
| // 发送消息输入框 | ||||
| const isComposing = ref(false) // 判断用户是否在输入 | ||||
| const conversationInAbortController = ref<any>() // 对话进行中 abort 控制器(控制 stream 对话) | ||||
| const inputTimeout = ref<any>() // 处理输入中回车的定时器 | ||||
| const prompt = ref<string>() // prompt | ||||
| const enableContext = ref<boolean>(true) // 是否开启上下文 | ||||
| // 接收 Stream 消息 | ||||
| const receiveMessageFullText = ref('') | ||||
| const receiveMessageDisplayedText = ref('') | ||||
|  | ||||
| // =========== 【聊天对话】相关 =========== | ||||
|  | ||||
| /** 获取对话信息 */ | ||||
| const getConversation = async (id: number | null) => { | ||||
|   if (!id) { | ||||
|     return | ||||
|   } | ||||
|   const conversation: ChatConversationVO = await ChatConversationApi.getChatConversationMy(id) | ||||
|   if (!conversation) { | ||||
|     return | ||||
|   } | ||||
|   activeConversation.value = conversation | ||||
|   activeConversationId.value = conversation.id | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * 点击某个对话 | ||||
|  * | ||||
|  * @param conversation 选中的对话 | ||||
|  * @return 是否切换成功 | ||||
|  */ | ||||
| const handleConversationClick = async (conversation: ChatConversationVO) => { | ||||
|   // 对话进行中,不允许切换 | ||||
|   if (conversationInProgress.value) { | ||||
|     message.alert('对话中,不允许切换!') | ||||
|     return false | ||||
|   } | ||||
|  | ||||
|   // 更新选中的对话 id | ||||
|   activeConversationId.value = conversation.id | ||||
|   activeConversation.value = conversation | ||||
|   // 刷新 message 列表 | ||||
|   await getMessageList() | ||||
|   // 滚动底部 | ||||
|   scrollToBottom(true) | ||||
|   // 清空输入框 | ||||
|   prompt.value = '' | ||||
|   return true | ||||
| } | ||||
|  | ||||
| /** 删除某个对话*/ | ||||
| const handlerConversationDelete = async (delConversation: ChatConversationVO) => { | ||||
|   // 删除的对话如果是当前选中的,那么就重置 | ||||
|   if (activeConversationId.value === delConversation.id) { | ||||
|     await handleConversationClear() | ||||
|   } | ||||
| } | ||||
| /** 清空选中的对话 */ | ||||
| const handleConversationClear = async () => { | ||||
|   // 对话进行中,不允许切换 | ||||
|   if (conversationInProgress.value) { | ||||
|     message.alert('对话中,不允许切换!') | ||||
|     return false | ||||
|   } | ||||
|   activeConversationId.value = null | ||||
|   activeConversation.value = null | ||||
|   activeMessageList.value = [] | ||||
| } | ||||
|  | ||||
| /** 修改聊天对话 */ | ||||
| const conversationUpdateFormRef = ref() | ||||
| const openChatConversationUpdateForm = async () => { | ||||
|   conversationUpdateFormRef.value.open(activeConversationId.value) | ||||
| } | ||||
| const handleConversationUpdateSuccess = async () => { | ||||
|   // 对话更新成功,刷新最新信息 | ||||
|   await getConversation(activeConversationId.value) | ||||
| } | ||||
|  | ||||
| /** 处理聊天对话的创建成功 */ | ||||
| const handleConversationCreate = async () => { | ||||
|   // 创建对话 | ||||
|   await conversationListRef.value.createConversation() | ||||
| } | ||||
| /** 处理聊天对话的创建成功 */ | ||||
| const handleConversationCreateSuccess = async () => { | ||||
|   // 创建新的对话,清空输入框 | ||||
|   prompt.value = '' | ||||
| } | ||||
|  | ||||
| // =========== 【消息列表】相关 =========== | ||||
|  | ||||
| /** 获取消息 message 列表 */ | ||||
| const getMessageList = async () => { | ||||
|   try { | ||||
|     if (activeConversationId.value === null) { | ||||
|       return | ||||
|     } | ||||
|     // Timer 定时器,如果加载速度很快,就不进入加载中 | ||||
|     activeMessageListLoadingTimer.value = setTimeout(() => { | ||||
|       activeMessageListLoading.value = true | ||||
|     }, 60) | ||||
|  | ||||
|     // 获取消息列表 | ||||
|     activeMessageList.value = await ChatMessageApi.getChatMessageListByConversationId( | ||||
|       activeConversationId.value | ||||
|     ) | ||||
|  | ||||
|     // 滚动到最下面 | ||||
|     await nextTick() | ||||
|     await scrollToBottom() | ||||
|   } finally { | ||||
|     // time 定时器,如果加载速度很快,就不进入加载中 | ||||
|     if (activeMessageListLoadingTimer.value) { | ||||
|       clearTimeout(activeMessageListLoadingTimer.value) | ||||
|     } | ||||
|     // 加载结束 | ||||
|     activeMessageListLoading.value = false | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** | ||||
|  * 消息列表 | ||||
|  * | ||||
|  * 和 {@link #getMessageList()} 的差异是,把 systemMessage 考虑进去 | ||||
|  */ | ||||
| const messageList = computed(() => { | ||||
|   if (activeMessageList.value.length > 0) { | ||||
|     return activeMessageList.value | ||||
|   } | ||||
|   // 没有消息时,如果有 systemMessage 则展示它 | ||||
|   if (activeConversation.value?.systemMessage) { | ||||
|     return [ | ||||
|       { | ||||
|         id: 0, | ||||
|         type: 'system', | ||||
|         content: activeConversation.value.systemMessage | ||||
|       } | ||||
|     ] | ||||
|   } | ||||
|   return [] | ||||
| }) | ||||
|  | ||||
| /** 处理删除 message 消息 */ | ||||
| const handleMessageDelete = () => { | ||||
|   if (conversationInProgress.value) { | ||||
|     message.alert('回答中,不能删除!') | ||||
|     return | ||||
|   } | ||||
|   // 刷新 message 列表 | ||||
|   getMessageList() | ||||
| } | ||||
|  | ||||
| /** 处理 message 清空 */ | ||||
| const handlerMessageClear = async () => { | ||||
|   if (!activeConversationId.value) { | ||||
|     return | ||||
|   } | ||||
|   try { | ||||
|     // 确认提示 | ||||
|     await message.delConfirm('确认清空对话消息?') | ||||
|     // 清空对话 | ||||
|     await ChatMessageApi.deleteByConversationId(activeConversationId.value) | ||||
|     // 刷新 message 列表 | ||||
|     activeMessageList.value = [] | ||||
|   } catch {} | ||||
| } | ||||
|  | ||||
| /** 回到 message 列表的顶部 */ | ||||
| const handleGoTopMessage = () => { | ||||
|   messageRef.value.handlerGoTop() | ||||
| } | ||||
|  | ||||
| // =========== 【发送消息】相关 =========== | ||||
|  | ||||
| /** 处理来自 keydown 的发送消息 */ | ||||
| const handleSendByKeydown = async (event) => { | ||||
|   // 判断用户是否在输入 | ||||
|   if (isComposing.value) { | ||||
|     return | ||||
|   } | ||||
|   // 进行中不允许发送 | ||||
|   if (conversationInProgress.value) { | ||||
|     return | ||||
|   } | ||||
|   const content = prompt.value?.trim() as string | ||||
|   if (event.key === 'Enter') { | ||||
|     if (event.shiftKey) { | ||||
|       // 插入换行 | ||||
|       prompt.value += '\r\n' | ||||
|       event.preventDefault() // 防止默认的换行行为 | ||||
|     } else { | ||||
|       // 发送消息 | ||||
|       await doSendMessage(content) | ||||
|       event.preventDefault() // 防止默认的提交行为 | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 处理来自【发送】按钮的发送消息 */ | ||||
| const handleSendByButton = () => { | ||||
|   doSendMessage(prompt.value?.trim() as string) | ||||
| } | ||||
|  | ||||
| /** 处理 prompt 输入变化 */ | ||||
| const handlePromptInput = (event) => { | ||||
|   // 非输入法 输入设置为 true | ||||
|   if (!isComposing.value) { | ||||
|     // 回车 event data 是 null | ||||
|     if (event.data == null) { | ||||
|       return | ||||
|     } | ||||
|     isComposing.value = true | ||||
|   } | ||||
|   // 清理定时器 | ||||
|   if (inputTimeout.value) { | ||||
|     clearTimeout(inputTimeout.value) | ||||
|   } | ||||
|   // 重置定时器 | ||||
|   inputTimeout.value = setTimeout(() => { | ||||
|     isComposing.value = false | ||||
|   }, 400) | ||||
| } | ||||
| // TODO @芋艿:是不是可以通过 @keydown.enter、@keydown.shift.enter 来实现,回车发送、shift+回车换行;主要看看,是不是可以简化 isComposing 相关的逻辑 | ||||
| const onCompositionstart = () => { | ||||
|   isComposing.value = true | ||||
| } | ||||
| const onCompositionend = () => { | ||||
|   // console.log('输入结束...') | ||||
|   setTimeout(() => { | ||||
|     isComposing.value = false | ||||
|   }, 200) | ||||
| } | ||||
|  | ||||
| /** 真正执行【发送】消息操作 */ | ||||
| const doSendMessage = async (content: string) => { | ||||
|   // 校验 | ||||
|   if (content.length < 1) { | ||||
|     message.error('发送失败,原因:内容为空!') | ||||
|     return | ||||
|   } | ||||
|   if (activeConversationId.value == null) { | ||||
|     message.error('还没创建对话,不能发送!') | ||||
|     return | ||||
|   } | ||||
|   // 清空输入框 | ||||
|   prompt.value = '' | ||||
|   // 执行发送 | ||||
|   await doSendMessageStream({ | ||||
|     conversationId: activeConversationId.value, | ||||
|     content: content | ||||
|   } as ChatMessageVO) | ||||
| } | ||||
|  | ||||
| /** 真正执行【发送】消息操作 */ | ||||
| const doSendMessageStream = async (userMessage: ChatMessageVO) => { | ||||
|   // 创建 AbortController 实例,以便中止请求 | ||||
|   conversationInAbortController.value = new AbortController() | ||||
|   // 标记对话进行中 | ||||
|   conversationInProgress.value = true | ||||
|   // 设置为空 | ||||
|   receiveMessageFullText.value = '' | ||||
|  | ||||
|   try { | ||||
|     // 1.1 先添加两个假数据,等 stream 返回再替换 | ||||
|     activeMessageList.value.push({ | ||||
|       id: -1, | ||||
|       conversationId: activeConversationId.value, | ||||
|       type: 'user', | ||||
|       content: userMessage.content, | ||||
|       createTime: new Date() | ||||
|     } as ChatMessageVO) | ||||
|     activeMessageList.value.push({ | ||||
|       id: -2, | ||||
|       conversationId: activeConversationId.value, | ||||
|       type: 'assistant', | ||||
|       content: '思考中...', | ||||
|       createTime: new Date() | ||||
|     } as ChatMessageVO) | ||||
|     // 1.2 滚动到最下面 | ||||
|     await nextTick() | ||||
|     await scrollToBottom() // 底部 | ||||
|     // 1.3 开始滚动 | ||||
|     textRoll() | ||||
|  | ||||
|     // 2. 发送 event stream | ||||
|     let isFirstChunk = true // 是否是第一个 chunk 消息段 | ||||
|     await ChatMessageApi.sendChatMessageStream( | ||||
|       userMessage.conversationId, | ||||
|       userMessage.content, | ||||
|       conversationInAbortController.value, | ||||
|       enableContext.value, | ||||
|       async (res) => { | ||||
|         const { code, data, msg } = JSON.parse(res.data) | ||||
|         if (code !== 0) { | ||||
|           message.alert(`对话异常! ${msg}`) | ||||
|           return | ||||
|         } | ||||
|  | ||||
|         // 如果内容为空,就不处理。 | ||||
|         if (data.receive.content === '') { | ||||
|           return | ||||
|         } | ||||
|         // 首次返回需要添加一个 message 到页面,后面的都是更新 | ||||
|         if (isFirstChunk) { | ||||
|           isFirstChunk = false | ||||
|           // 弹出两个假数据 | ||||
|           activeMessageList.value.pop() | ||||
|           activeMessageList.value.pop() | ||||
|           // 更新返回的数据 | ||||
|           activeMessageList.value.push(data.send) | ||||
|           activeMessageList.value.push(data.receive) | ||||
|         } | ||||
|         // debugger | ||||
|         receiveMessageFullText.value = receiveMessageFullText.value + data.receive.content | ||||
|         // 滚动到最下面 | ||||
|         await scrollToBottom() | ||||
|       }, | ||||
|       (error) => { | ||||
|         message.alert(`对话异常! ${error}`) | ||||
|         stopStream() | ||||
|       }, | ||||
|       () => { | ||||
|         stopStream() | ||||
|       } | ||||
|     ) | ||||
|   } catch {} | ||||
| } | ||||
|  | ||||
| /** 停止 stream 流式调用 */ | ||||
| const stopStream = async () => { | ||||
|   // tip:如果 stream 进行中的 message,就需要调用 controller 结束 | ||||
|   if (conversationInAbortController.value) { | ||||
|     conversationInAbortController.value.abort() | ||||
|   } | ||||
|   // 设置为 false | ||||
|   conversationInProgress.value = false | ||||
| } | ||||
|  | ||||
| /** 编辑 message:设置为 prompt,可以再次编辑 */ | ||||
| const handleMessageEdit = (message: ChatMessageVO) => { | ||||
|   prompt.value = message.content | ||||
| } | ||||
|  | ||||
| /** 刷新 message:基于指定消息,再次发起对话 */ | ||||
| const handleMessageRefresh = (message: ChatMessageVO) => { | ||||
|   doSendMessage(message.content) | ||||
| } | ||||
|  | ||||
| // ============== 【消息滚动】相关 ============= | ||||
|  | ||||
| /** 滚动到 message 底部 */ | ||||
| const scrollToBottom = async (isIgnore?: boolean) => { | ||||
|   await nextTick() | ||||
|   if (messageRef.value) { | ||||
|     messageRef.value.scrollToBottom(isIgnore) | ||||
|   } | ||||
| } | ||||
|  | ||||
| /** 自提滚动效果 */ | ||||
| const textRoll = async () => { | ||||
|   let index = 0 | ||||
|   try { | ||||
|     // 只能执行一次 | ||||
|     if (textRoleRunning.value) { | ||||
|       return | ||||
|     } | ||||
|     // 设置状态 | ||||
|     textRoleRunning.value = true | ||||
|     receiveMessageDisplayedText.value = '' | ||||
|     const task = async () => { | ||||
|       // 调整速度 | ||||
|       const diff = | ||||
|         (receiveMessageFullText.value.length - receiveMessageDisplayedText.value.length) / 10 | ||||
|       if (diff > 5) { | ||||
|         textSpeed.value = 10 | ||||
|       } else if (diff > 2) { | ||||
|         textSpeed.value = 30 | ||||
|       } else if (diff > 1.5) { | ||||
|         textSpeed.value = 50 | ||||
|       } else { | ||||
|         textSpeed.value = 100 | ||||
|       } | ||||
|       // 对话结束,就按 30 的速度 | ||||
|       if (!conversationInProgress.value) { | ||||
|         textSpeed.value = 10 | ||||
|       } | ||||
|  | ||||
|       if (index < receiveMessageFullText.value.length) { | ||||
|         receiveMessageDisplayedText.value += receiveMessageFullText.value[index] | ||||
|         index++ | ||||
|  | ||||
|         // 更新 message | ||||
|         const lastMessage = activeMessageList.value[activeMessageList.value.length - 1] | ||||
|         lastMessage.content = receiveMessageDisplayedText.value | ||||
|         // 滚动到住下面 | ||||
|         await scrollToBottom() | ||||
|         // 重新设置任务 | ||||
|         timer = setTimeout(task, textSpeed.value) | ||||
|       } else { | ||||
|         // 不是对话中可以结束 | ||||
|         if (!conversationInProgress.value) { | ||||
|           textRoleRunning.value = false | ||||
|           clearTimeout(timer) | ||||
|         } else { | ||||
|           // 重新设置任务 | ||||
|           timer = setTimeout(task, textSpeed.value) | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|     let timer = setTimeout(task, textSpeed.value) | ||||
|   } catch {} | ||||
| } | ||||
|  | ||||
| /** 初始化 **/ | ||||
| onMounted(async () => { | ||||
|   // 如果有 conversationId 参数,则默认选中 | ||||
|   if (route.query.conversationId) { | ||||
|     const id = route.query.conversationId as unknown as number | ||||
|     activeConversationId.value = id | ||||
|     await getConversation(id) | ||||
|   } | ||||
|  | ||||
|   // 获取列表数据 | ||||
|   activeMessageListLoading.value = true | ||||
|   await getMessageList() | ||||
| }) | ||||
| </script> | ||||
|  | ||||
| <style lang="scss" scoped> | ||||
| .ai-layout { | ||||
|   position: absolute; | ||||
|   flex: 1; | ||||
|   top: 0; | ||||
|   left: 0; | ||||
|   height: 100%; | ||||
|   width: 100%; | ||||
| } | ||||
|  | ||||
| .conversation-container { | ||||
|   position: relative; | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   justify-content: space-between; | ||||
|   padding: 10px 10px 0; | ||||
|  | ||||
|   .btn-new-conversation { | ||||
|     padding: 18px 0; | ||||
|   } | ||||
|  | ||||
|   .search-input { | ||||
|     margin-top: 20px; | ||||
|   } | ||||
|  | ||||
|   .conversation-list { | ||||
|     margin-top: 20px; | ||||
|  | ||||
|     .conversation { | ||||
|       display: flex; | ||||
|       flex-direction: row; | ||||
|       justify-content: space-between; | ||||
|       flex: 1; | ||||
|       padding: 0 5px; | ||||
|       margin-top: 10px; | ||||
|       cursor: pointer; | ||||
|       border-radius: 5px; | ||||
|       align-items: center; | ||||
|       line-height: 30px; | ||||
|  | ||||
|       &.active { | ||||
|         background-color: #e6e6e6; | ||||
|  | ||||
|         .button { | ||||
|           display: inline-block; | ||||
|         } | ||||
|       } | ||||
|  | ||||
|       .title-wrapper { | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         align-items: center; | ||||
|       } | ||||
|  | ||||
|       .title { | ||||
|         padding: 5px 10px; | ||||
|         max-width: 220px; | ||||
|         font-size: 14px; | ||||
|         overflow: hidden; | ||||
|         white-space: nowrap; | ||||
|         text-overflow: ellipsis; | ||||
|       } | ||||
|  | ||||
|       .avatar { | ||||
|         width: 28px; | ||||
|         height: 28px; | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         justify-items: center; | ||||
|       } | ||||
|  | ||||
|       // 对话编辑、删除 | ||||
|       .button-wrapper { | ||||
|         right: 2px; | ||||
|         display: flex; | ||||
|         flex-direction: row; | ||||
|         justify-items: center; | ||||
|         color: #606266; | ||||
|  | ||||
|         .el-icon { | ||||
|           margin-right: 5px; | ||||
|         } | ||||
|       } | ||||
|     } | ||||
|   } | ||||
|  | ||||
|   // 角色仓库、清空未设置对话 | ||||
|   .tool-box { | ||||
|     line-height: 35px; | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     align-items: center; | ||||
|     color: var(--el-text-color); | ||||
|  | ||||
|     > div { | ||||
|       display: flex; | ||||
|       align-items: center; | ||||
|       color: #606266; | ||||
|       padding: 0; | ||||
|       margin: 0; | ||||
|       cursor: pointer; | ||||
|  | ||||
|       > span { | ||||
|         margin-left: 5px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| // 头部 | ||||
| .detail-container { | ||||
|   background: #ffffff; | ||||
|  | ||||
|   .header { | ||||
|     display: flex; | ||||
|     flex-direction: row; | ||||
|     align-items: center; | ||||
|     justify-content: space-between; | ||||
|     background: #fbfbfb; | ||||
|     box-shadow: 0 0 0 0 #dcdfe6; | ||||
|  | ||||
|     .title { | ||||
|       font-size: 18px; | ||||
|       font-weight: bold; | ||||
|     } | ||||
|  | ||||
|     .btns { | ||||
|       display: flex; | ||||
|       width: 300px; | ||||
|       flex-direction: row; | ||||
|       justify-content: flex-end; | ||||
|       //justify-content: space-between; | ||||
|  | ||||
|       .btn { | ||||
|         padding: 10px; | ||||
|       } | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| // main 容器 | ||||
| .main-container { | ||||
|   margin: 0; | ||||
|   padding: 0; | ||||
|   position: relative; | ||||
|   height: 100%; | ||||
|   width: 100%; | ||||
|  | ||||
|   .message-container { | ||||
|     position: absolute; | ||||
|     top: 0; | ||||
|     bottom: 0; | ||||
|     left: 0; | ||||
|     right: 0; | ||||
|     overflow-y: hidden; | ||||
|     padding: 0; | ||||
|     margin: 0; | ||||
|   } | ||||
| } | ||||
|  | ||||
| // 底部 | ||||
| .footer-container { | ||||
|   display: flex; | ||||
|   flex-direction: column; | ||||
|   height: auto; | ||||
|   margin: 0; | ||||
|   padding: 0; | ||||
|  | ||||
|   .prompt-from { | ||||
|     display: flex; | ||||
|     flex-direction: column; | ||||
|     height: auto; | ||||
|     border: 1px solid #e3e3e3; | ||||
|     border-radius: 10px; | ||||
|     margin: 10px 20px 20px 20px; | ||||
|     padding: 9px 10px; | ||||
|   } | ||||
|  | ||||
|   .prompt-input { | ||||
|     height: 80px; | ||||
|     //box-shadow: none; | ||||
|     border: none; | ||||
|     box-sizing: border-box; | ||||
|     resize: none; | ||||
|     padding: 0 2px; | ||||
|     overflow: auto; | ||||
|   } | ||||
|  | ||||
|   .prompt-input:focus { | ||||
|     outline: none; | ||||
|   } | ||||
|  | ||||
|   .prompt-btns { | ||||
|     display: flex; | ||||
|     justify-content: space-between; | ||||
|     padding-bottom: 0; | ||||
|     padding-top: 5px; | ||||
|   } | ||||
| } | ||||
| </style> | ||||
		Reference in New Issue
	
	Block a user
	 YOHO\20373
					YOHO\20373