【增加】AI 对话增加 GROUP 分组

pull/449/head^2
cherishsince 2024-05-16 11:28:31 +08:00
parent ad02983098
commit f21991f4fc
1 changed files with 102 additions and 48 deletions

View File

@ -5,7 +5,7 @@
<div> <div>
<!-- 左顶部新建对话 --> <!-- 左顶部新建对话 -->
<el-button class="w-1/1 btn-new-conversation" type="primary" @click="createConversation"> <el-button class="w-1/1 btn-new-conversation" type="primary" @click="createConversation">
<Icon icon="ep:plus" class="mr-5px" /> <Icon icon="ep:plus" class="mr-5px"/>
新建对话 新建对话
</el-button> </el-button>
<!-- 左顶部搜索对话 --> <!-- 左顶部搜索对话 -->
@ -17,45 +17,50 @@
@keyup="searchConversation" @keyup="searchConversation"
> >
<template #prefix> <template #prefix>
<Icon icon="ep:search" /> <Icon icon="ep:search"/>
</template> </template>
</el-input> </el-input>
<!-- 左中间对话列表 --> <!-- 左中间对话列表 -->
<div class="conversation-list"> <div class="conversation-list">
<!-- TODO @fain置顶聊天记录一星期钱30天前前端对数据重新做一下分组或者后端接口改一下 --> <!-- TODO @fain置顶聊天记录一星期钱30天前前端对数据重新做一下分组或者后端接口改一下 -->
<div> <div v-for="conversationKey in Object.keys(conversationMap)" :key="conversationKey" >
<el-text class="mx-1" size="small" tag="b">置顶</el-text> <div v-if="conversationMap[conversationKey].length">
</div> <el-text class="mx-1" size="small" tag="b">{{conversationKey}}</el-text>
<el-row v-for="conversation in conversationList" :key="conversation.id" @click="handleConversationClick(conversation.id)">
<div
:class="conversation.id === conversationId ? 'conversation active' : 'conversation'"
@click="changeConversation(conversation.id)"
>
<div class="title-wrapper">
<img class="avatar" :src="conversation.roleAvatar" />
<span class="title">{{ conversation.title }}</span>
</div>
<!-- TODO @fan缺一个置顶按钮效果改成 hover 上去展示 -->
<div class="button-wrapper">
<el-icon title="编辑" @click="updateConversationTitle(conversation)">
<Icon icon="ep:edit" />
</el-icon>
<el-icon title="删除会话" @click="deleteChatConversation(conversation)">
<Icon icon="ep:delete" />
</el-icon>
</div>
</div> </div>
</el-row> <el-row
v-for="conversation in conversationMap[conversationKey]"
:key="conversation.id"
@click="handleConversationClick(conversation.id)">
<div
:class="conversation.id === conversationId ? 'conversation active' : 'conversation'"
@click="changeConversation(conversation.id)"
>
<div class="title-wrapper">
<img class="avatar" :src="conversation.roleAvatar"/>
<span class="title">{{ conversation.title }}</span>
</div>
<!-- TODO @fan缺一个置顶按钮效果改成 hover 上去展示 -->
<div class="button-wrapper">
<el-icon title="编辑" @click="updateConversationTitle(conversation)">
<Icon icon="ep:edit"/>
</el-icon>
<el-icon title="删除会话" @click="deleteChatConversation(conversation)">
<Icon icon="ep:delete"/>
</el-icon>
</div>
</div>
</el-row>
</div>
</div> </div>
</div> </div>
<!-- 左底部工具栏 --> <!-- 左底部工具栏 -->
<div class="tool-box"> <div class="tool-box">
<div @click="handleRoleRepository"> <div @click="handleRoleRepository">
<Icon icon="ep:user" /> <Icon icon="ep:user"/>
<el-text size="small">角色仓库</el-text> <el-text size="small">角色仓库</el-text>
</div> </div>
<div> <div>
<Icon icon="ep:delete" /> <Icon icon="ep:delete"/>
<el-text size="small">清空未置顶对话</el-text> <el-text size="small">清空未置顶对话</el-text>
</div> </div>
</div> </div>
@ -71,16 +76,16 @@
<!-- TODO @fan样式改下这里我已经改成点击后弹出了 --> <!-- TODO @fan样式改下这里我已经改成点击后弹出了 -->
<el-button type="primary" @click="openChatConversationUpdateForm"> <el-button type="primary" @click="openChatConversationUpdateForm">
<span v-html="useConversation?.modelName"></span> <span v-html="useConversation?.modelName"></span>
<Icon icon="ep:setting" style="margin-left: 10px" /> <Icon icon="ep:setting" style="margin-left: 10px"/>
</el-button> </el-button>
<el-button> <el-button>
<Icon icon="ep:user" /> <Icon icon="ep:user"/>
</el-button> </el-button>
<el-button> <el-button>
<Icon icon="ep:download" /> <Icon icon="ep:download"/>
</el-button> </el-button>
<el-button> <el-button>
<Icon icon="ep:arrow-up" /> <Icon icon="ep:arrow-up"/>
</el-button> </el-button>
</div> </div>
</el-header> </el-header>
@ -107,11 +112,11 @@
</div> </div>
<div class="left-btns"> <div class="left-btns">
<div class="btn-cus" @click="noCopy(item.content)"> <div class="btn-cus" @click="noCopy(item.content)">
<img class="btn-image" src="../../../assets/ai/copy.svg" /> <img class="btn-image" src="../../../assets/ai/copy.svg"/>
<el-text class="btn-cus-text">复制</el-text> <el-text class="btn-cus-text">复制</el-text>
</div> </div>
<div class="btn-cus" style="margin-left: 20px" @click="onDelete(item.id)"> <div class="btn-cus" style="margin-left: 20px" @click="onDelete(item.id)">
<img class="btn-image" src="@/assets/ai/delete.svg" style="height: 17px" /> <img class="btn-image" src="@/assets/ai/delete.svg" style="height: 17px"/>
<el-text class="btn-cus-text">删除</el-text> <el-text class="btn-cus-text">删除</el-text>
</div> </div>
</div> </div>
@ -133,11 +138,11 @@
</div> </div>
<div class="right-btns"> <div class="right-btns">
<div class="btn-cus" @click="noCopy(item.content)"> <div class="btn-cus" @click="noCopy(item.content)">
<img class="btn-image" src="@/assets/ai/copy.svg" /> <img class="btn-image" src="@/assets/ai/copy.svg"/>
<el-text class="btn-cus-text">复制</el-text> <el-text class="btn-cus-text">复制</el-text>
</div> </div>
<div class="btn-cus" style="margin-left: 20px" @click="onDelete(item.id)"> <div class="btn-cus" style="margin-left: 20px" @click="onDelete(item.id)">
<img class="btn-image" src="@/assets/ai/delete.svg" style="height: 17px" /> <img class="btn-image" src="@/assets/ai/delete.svg" style="height: 17px"/>
<el-text class="btn-cus-text">删除</el-text> <el-text class="btn-cus-text">删除</el-text>
</div> </div>
</div> </div>
@ -145,10 +150,10 @@
</div> </div>
</div> </div>
</div> </div>
<!-- 角色仓库抽屉 --> <!-- 角色仓库抽屉 -->
<el-drawer v-model="drawer" title="角色仓库" size="50%"> <el-drawer v-model="drawer" title="角色仓库" size="50%">
<Role /> <Role/>
</el-drawer> </el-drawer>
</el-main> </el-main>
<el-footer class="footer-container"> <el-footer class="footer-container">
<form @submit.prevent="onSend" class="prompt-from"> <form @submit.prevent="onSend" class="prompt-from">
@ -162,7 +167,7 @@
placeholder="问我任何问题...Shift+Enter 换行,按下 Enter 发送)" placeholder="问我任何问题...Shift+Enter 换行,按下 Enter 发送)"
></textarea> ></textarea>
<div class="prompt-btns"> <div class="prompt-btns">
<el-switch /> <el-switch/>
<el-button <el-button
type="primary" type="primary"
size="default" size="default"
@ -193,24 +198,25 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ChatMessageApi, ChatMessageSendVO, ChatMessageVO } from '@/api/ai/chat/message' import {ChatMessageApi, ChatMessageVO} from '@/api/ai/chat/message'
import { ChatConversationApi, ChatConversationVO } from '@/api/ai/chat/conversation' import {ChatConversationApi, ChatConversationVO} from '@/api/ai/chat/conversation'
import ChatConversationUpdateForm from './components/ChatConversationUpdateForm.vue' import ChatConversationUpdateForm from './components/ChatConversationUpdateForm.vue'
import Role from '@/views/ai/chat/role/index.vue' import Role from '@/views/ai/chat/role/index.vue'
import { formatDate } from '@/utils/formatTime' import {formatDate} from '@/utils/formatTime'
import { useClipboard } from '@vueuse/core' import {useClipboard} from '@vueuse/core'
// markdown // markdown
import { marked } from 'marked' import {marked} from 'marked'
// https://highlightjs.org/ // https://highlightjs.org/
import 'highlight.js/styles/vs2015.min.css' import 'highlight.js/styles/vs2015.min.css'
import hljs from 'highlight.js' import hljs from 'highlight.js'
const route = useRoute() // const route = useRoute() //
const message = useMessage() // const message = useMessage() //
// //
const renderer = { const renderer = {
code(code, language, c) { code(code, language, c) {
const highlightHtml = hljs.highlight(code, { language: language, ignoreIllegals: true }).value const highlightHtml = hljs.highlight(code, {language: language, ignoreIllegals: true}).value
const copyHtml = `<div id="copy" data-copy='${code}' style="position: absolute; right: 10px; top: 5px; color: #fff;cursor: pointer;">复制</div>` const copyHtml = `<div id="copy" data-copy='${code}' style="position: absolute; right: 10px; top: 5px; color: #fff;cursor: pointer;">复制</div>`
return `<pre>${copyHtml}<code class="hljs">${highlightHtml}</code></pre>` return `<pre>${copyHtml}<code class="hljs">${highlightHtml}</code></pre>`
} }
@ -220,8 +226,9 @@ marked.use({
}) })
const conversationList = ref([] as ChatConversationVO[]) const conversationList = ref([] as ChatConversationVO[])
const conversationMap = ref<any>({})
// copy // copy
const { copy } = useClipboard() const {copy} = useClipboard()
const drawer = ref<boolean>(false) // const drawer = ref<boolean>(false) //
const searchName = ref('') // const searchName = ref('') //
@ -261,7 +268,7 @@ const changeConversation = (id: number) => {
/** 更新聊天会话的标题 */ /** 更新聊天会话的标题 */
const updateConversationTitle = async (conversation: ChatConversationVO) => { const updateConversationTitle = async (conversation: ChatConversationVO) => {
// //
const { value } = await ElMessageBox.prompt('修改标题', { const {value} = await ElMessageBox.prompt('修改标题', {
inputPattern: /^[\s\S]*.*\S[\s\S]*$/, // inputPattern: /^[\s\S]*.*\S[\s\S]*$/, //
inputErrorMessage: '标题不能为空', inputErrorMessage: '标题不能为空',
inputValue: conversation.title inputValue: conversation.title
@ -286,7 +293,8 @@ const deleteChatConversation = async (conversation: ChatConversationVO) => {
message.success('会话已删除') message.success('会话已删除')
// //
await getChatConversationList() await getChatConversationList()
} catch {} } catch {
}
} }
const searchConversation = () => { const searchConversation = () => {
@ -532,8 +540,54 @@ const getChatConversationList = async () => {
changeConversation(conversationList.value[0].id) changeConversation(conversationList.value[0].id)
} }
} }
// map
const groupRes = await conversationTimeGroup(conversationList.value)
conversationMap.value = groupRes
} }
const conversationTimeGroup = async (list: ChatConversationVO[]) => {
// (30)
const groupMap = {
'置顶': [],
'今天': [],
'一天前': [],
'三天前': [],
'七天前': [],
'三十天前': []
}
//
const now = Date.now();
//
const oneDay = 24 * 60 * 60 * 1000;
const threeDays = 3 * oneDay;
const sevenDays = 7 * oneDay;
const thirtyDays = 30 * oneDay;
console.log('listlistlist', list)
for (const conversation: ChatConversationVO of list) {
//
if (conversation.pinned) {
groupMap['置顶'].push(conversation)
continue
}
//
const diff = now - conversation.updateTime;
//
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 handleConversationClick = async (id: number) => { const handleConversationClick = async (id: number) => {
// //