feat: openKnowledgeSource for chat

This commit is contained in:
teukkk 2025-06-30 16:05:14 +08:00
parent d8a9c9ccdd
commit 3e4aac3557
6 changed files with 63 additions and 29 deletions

View File

@ -88,11 +88,15 @@ const props = defineProps({
} }
}) })
const emit = defineEmits(['openExecutionDetail']) const emit = defineEmits(['openExecutionDetail', 'openParagraph'])
const ParagraphSourceDialogRef = ref() const ParagraphSourceDialogRef = ref()
const ExecutionDetailDialogRef = ref() const ExecutionDetailDialogRef = ref()
function openParagraph(row: any, id?: string) { function openParagraph(row: any, id?: string) {
if (props.executionIsRightPanel) {
emit('openParagraph')
return
}
ParagraphSourceDialogRef.value.open(row, id) ParagraphSourceDialogRef.value.open(row, id)
} }
function openExecutionDetail(row: any) { function openExecutionDetail(row: any) {

View File

@ -10,26 +10,7 @@
:close-on-press-escape="false" :close-on-press-escape="false"
> >
<div class="mb-8"> <div class="mb-8">
<el-scrollbar> <ParagraphSourceContent :detail="detail"/>
<div class="paragraph-source-height p-16 pb-0">
<el-form label-position="top">
<el-form-item :label="$t('chat.paragraphSource.question')">
<el-input v-model="detail.problem_text" disabled />
</el-form-item>
<el-form-item :label="$t('chat.paragraphSource.optimizationQuestion')">
<el-input v-model="detail.padding_problem_text" disabled />
</el-form-item>
<el-form-item :label="$t('chat.KnowledgeSource.referenceParagraph')">
<div v-if="detail.paragraph_list.length > 0" class="w-full">
<template v-for="(item, index) in detail.paragraph_list" :key="index">
<ParagraphCard :data="item" :content="item.content" :index="index" />
</template>
</div>
<span v-else> - </span>
</el-form-item>
</el-form>
</div>
</el-scrollbar>
</div> </div>
</el-dialog> </el-dialog>
</template> </template>
@ -37,7 +18,7 @@
import { ref, watch, onBeforeUnmount } from 'vue' import { ref, watch, onBeforeUnmount } from 'vue'
import { cloneDeep } from 'lodash' import { cloneDeep } from 'lodash'
import { arraySort } from '@/utils/utils' import { arraySort } from '@/utils/utils'
import ParagraphCard from './component/ParagraphCard.vue' import ParagraphSourceContent from './component/ParagraphSourceContent.vue'
const emit = defineEmits(['refresh']) const emit = defineEmits(['refresh'])
const dialogVisible = ref(false) const dialogVisible = ref(false)

View File

@ -0,0 +1,30 @@
<template>
<el-scrollbar>
<div class="paragraph-source-height p-16 pb-0">
<el-form label-position="top">
<el-form-item :label="$t('chat.paragraphSource.question')">
<el-input :value="props.detail?.problem_text" disabled />
</el-form-item>
<el-form-item :label="$t('chat.paragraphSource.optimizationQuestion')">
<el-input :value="props.detail?.padding_problem_text" disabled />
</el-form-item>
<el-form-item :label="$t('chat.KnowledgeSource.referenceParagraph')">
<div v-if="props.detail?.paragraph_list.length > 0" class="w-full">
<template v-for="(item, index) in props.detail.paragraph_list" :key="index">
<ParagraphCard :data="item" :content="item.content" :index="index" />
</template>
</div>
<span v-else> - </span>
</el-form-item>
</el-form>
</div>
</el-scrollbar>
</template>
<script setup lang="ts">
import ParagraphCard from '@/components/ai-chat/component/ParagraphCard.vue'
const props = defineProps<{
detail?: any
}>()
</script>

View File

@ -45,6 +45,7 @@
:type="application.type" :type="application.type"
:executionIsRightPanel="props.executionIsRightPanel" :executionIsRightPanel="props.executionIsRightPanel"
@open-execution-detail="emit('openExecutionDetail')" @open-execution-detail="emit('openExecutionDetail')"
@openParagraph="emit('openParagraph')"
v-if="showSource(chatRecord) && index === chatRecord.answer_text_list.length - 1" v-if="showSource(chatRecord) && index === chatRecord.answer_text_list.length - 1"
/> />
</el-card> </el-card>
@ -90,7 +91,7 @@ const props = defineProps<{
const { user } = useStore() const { user } = useStore()
const emit = defineEmits(['update:chatRecord', 'openExecutionDetail']) const emit = defineEmits(['update:chatRecord', 'openExecutionDetail', 'openParagraph'])
const showAvatar = computed(() => { const showAvatar = computed(() => {
return user.isEnterprise() ? props.application.show_avatar : true return user.isEnterprise() ? props.application.show_avatar : true

View File

@ -50,6 +50,7 @@
:chat-management="ChatManagement" :chat-management="ChatManagement"
:executionIsRightPanel="props.executionIsRightPanel" :executionIsRightPanel="props.executionIsRightPanel"
@open-execution-detail="emit('openExecutionDetail', chatList[index])" @open-execution-detail="emit('openExecutionDetail', chatList[index])"
@openParagraph="emit('openParagraph', chatList[index])"
></AnswerContent> ></AnswerContent>
</template> </template>
<TransitionContent <TransitionContent
@ -129,7 +130,7 @@ const props = withDefaults(
type: 'ai-chat', type: 'ai-chat',
}, },
) )
const emit = defineEmits(['refresh', 'scroll', 'openExecutionDetail']) const emit = defineEmits(['refresh', 'scroll', 'openExecutionDetail', 'openParagraph'])
const { application, common } = useStore() const { application, common } = useStore()
const isMobile = computed(() => { const isMobile = computed(() => {
return common.isMobile() || mode === 'embed' || mode === 'mobile' return common.isMobile() || mode === 'embed' || mode === 'mobile'

View File

@ -232,17 +232,19 @@
@refresh="refresh" @refresh="refresh"
@scroll="handleScroll" @scroll="handleScroll"
@open-execution-detail="openExecutionDetail" @open-execution-detail="openExecutionDetail"
@openParagraph="openKnowledgeSource"
> >
</AiChat> </AiChat>
</div> </div>
</el-splitter-panel> </el-splitter-panel>
<el-splitter-panel class="execution-detail-panel" v-model:size="rightPanelSize" :resizable="false" collapsible> <el-splitter-panel class="execution-detail-panel" v-model:size="rightPanelSize" :resizable="false" collapsible>
<div class="p-16 flex-between border-b"> <div class="p-16 flex-between border-b">
<h4 class="medium">{{ $t('chat.executionDetails.title') }}</h4> <h4 class="medium">{{ rightPanelTitle }}</h4>
<el-icon size="20" class="cursor" @click="closeExecutionDetail"><Close /></el-icon> <el-icon size="20" class="cursor" @click="closeExecutionDetail"><Close /></el-icon>
</div> </div>
<div class="execution-detail-content" v-loading="executionLoading"> <div class="execution-detail-content" v-loading="rightPanelLoading">
<ExecutionDetailContent :detail="executionDetail" /> <ParagraphSourceContent v-if="rightPanelType === 'knowledgeSource'" :detail="rightPanelDetail" />
<ExecutionDetailContent v-if="rightPanelType === 'executionDetail'" :detail="executionDetail" />
</div> </div>
</el-splitter-panel> </el-splitter-panel>
</el-splitter> </el-splitter>
@ -274,6 +276,7 @@ import ResetPassword from '@/layout/layout-header/avatar/ResetPassword.vue'
import { t } from '@/locales' import { t } from '@/locales'
import type { ResetCurrentUserPasswordRequest } from '@/api/type/user' import type { ResetCurrentUserPasswordRequest } from '@/api/type/user'
import ExecutionDetailContent from '@/components/ai-chat/component/ExecutionDetailContent.vue' import ExecutionDetailContent from '@/components/ai-chat/component/ExecutionDetailContent.vue'
import ParagraphSourceContent from '@/components/ai-chat/component/ParagraphSourceContent.vue'
import { cloneDeep } from 'lodash' import { cloneDeep } from 'lodash'
useResize() useResize()
@ -530,17 +533,31 @@ onMounted(() => {
}) })
const rightPanelSize = ref(0) const rightPanelSize = ref(0)
const rightPanelTitle = ref('')
const rightPanelType = ref('')
const rightPanelLoading = ref(false)
const executionDetail = ref<any[]>([]) const executionDetail = ref<any[]>([])
const executionLoading = ref(false) const rightPanelDetail = ref<any>()
async function openExecutionDetail(row: any) { async function openExecutionDetail(row: any) {
rightPanelSize.value = 400 rightPanelSize.value = 400
rightPanelTitle.value = t('chat.executionDetails.title')
rightPanelType.value = 'executionDetail'
if (row.execution_details) { if (row.execution_details) {
executionDetail.value = cloneDeep(row.execution_details) executionDetail.value = cloneDeep(row.execution_details)
} else { } else {
const res = await chatAPI.getChatRecord(row.chat_id, row.record_id, executionLoading) const res = await chatAPI.getChatRecord(row.chat_id, row.record_id, rightPanelLoading)
executionDetail.value = cloneDeep(res.data.execution_details) executionDetail.value = cloneDeep(res.data.execution_details)
} }
} }
async function openKnowledgeSource (row: any) {
rightPanelTitle.value = t('chat.KnowledgeSource.title')
rightPanelType.value = 'knowledgeSource'
// TODO
rightPanelDetail.value = row
rightPanelSize.value = 400
}
function closeExecutionDetail() { function closeExecutionDetail() {
rightPanelSize.value = 0 rightPanelSize.value = 0
} }