DB-GPT/web/pages/construct/knowledge/index.tsx
Dreammy23 471689ba20
feat(web): Unified frontend code style (#1923)
Co-authored-by: Fangyin Cheng <staneyffer@gmail.com>
Co-authored-by: 谨欣 <echo.cmy@antgroup.com>
Co-authored-by: 严志勇 <yanzhiyong@tiansuixiansheng.com>
Co-authored-by: yanzhiyong <932374019@qq.com>
2024-08-30 14:03:06 +08:00

289 lines
9.9 KiB
TypeScript

import { ChatContext } from '@/app/chat-context';
import { apiInterceptors, delSpace, getSpaceConfig, getSpaceList, newDialogue } from '@/client/api';
import DocPanel from '@/components/knowledge/doc-panel';
import DocTypeForm from '@/components/knowledge/doc-type-form';
import DocUploadForm from '@/components/knowledge/doc-upload-form';
import Segmentation from '@/components/knowledge/segmentation';
import SpaceForm from '@/components/knowledge/space-form';
import BlurredCard, { ChatButton, InnerDropdown } from '@/new-components/common/blurredCard';
import ConstructLayout from '@/new-components/layout/Construct';
import { File, ISpace, IStorage, StepChangeParams } from '@/types/knowledge';
import { PlusOutlined, ReadOutlined, SearchOutlined, WarningOutlined } from '@ant-design/icons';
import { Button, Input, Modal, Spin, Steps, Tag } from 'antd';
import classNames from 'classnames';
import { debounce } from 'lodash';
import moment from 'moment';
import { useRouter } from 'next/router';
import { useContext, useEffect, useState } from 'react';
import { useTranslation } from 'react-i18next';
const Knowledge = () => {
const { setCurrentDialogInfo } = useContext(ChatContext);
const [spaceList, setSpaceList] = useState<Array<ISpace> | null>([]);
const [isAddShow, setIsAddShow] = useState<boolean>(false);
const [isPanelShow, setIsPanelShow] = useState<boolean>(false);
const [currentSpace, setCurrentSpace] = useState<ISpace>();
const [activeStep, setActiveStep] = useState<number>(0);
const [spaceName, setSpaceName] = useState<string>('');
const [files, setFiles] = useState<Array<File>>([]);
const [docType, setDocType] = useState<string>('');
const [addStatus, setAddStatus] = useState<string>('');
const [loading, setLoading] = useState<boolean>(false);
const [spaceConfig, setSpaceConfig] = useState<IStorage | null>(null);
const { t } = useTranslation();
const addKnowledgeSteps = [
{ title: t('Knowledge_Space_Config') },
{ title: t('Choose_a_Datasource_type') },
{ title: t('Upload') },
{ title: t('Segmentation') },
];
const router = useRouter();
async function getSpaces(params?: any) {
setLoading(true);
const [_, data] = await apiInterceptors(getSpaceList({ ...params }));
setLoading(false);
setSpaceList(data);
}
async function getSpaceConfigs() {
const [_, data] = await apiInterceptors(getSpaceConfig());
if (!data) return null;
setSpaceConfig(data.storage);
}
useEffect(() => {
getSpaces();
getSpaceConfigs();
}, []);
const handleChat = async (space: ISpace) => {
const [_, data] = await apiInterceptors(
newDialogue({
chat_mode: 'chat_knowledge',
}),
);
// 知识库对话都默认私有知识库应用下
if (data?.conv_uid) {
setCurrentDialogInfo?.({
chat_scene: data.chat_mode,
app_code: data.chat_mode,
});
localStorage.setItem(
'cur_dialog_info',
JSON.stringify({
chat_scene: data.chat_mode,
app_code: data.chat_mode,
}),
);
router.push(`/chat?scene=chat_knowledge&id=${data?.conv_uid}&knowledge_id=${space.name}`);
}
};
const handleStepChange = ({ label, spaceName, docType, files }: StepChangeParams) => {
if (label === 'finish') {
setIsAddShow(false);
getSpaces();
setSpaceName('');
setDocType('');
setAddStatus('finish');
localStorage.removeItem('cur_space_id');
} else if (label === 'forward') {
activeStep === 0 && getSpaces();
setActiveStep(step => step + 1);
} else {
setActiveStep(step => step - 1);
}
files && setFiles(files);
spaceName && setSpaceName(spaceName);
docType && setDocType(docType);
};
function onAddDoc(spaceName: string) {
setSpaceName(spaceName);
setActiveStep(1);
setIsAddShow(true);
setAddStatus('start');
}
const showDeleteConfirm = (space: ISpace) => {
Modal.confirm({
title: t('Tips'),
icon: <WarningOutlined />,
content: `${t('Del_Knowledge_Tips')}?`,
okText: 'Yes',
okType: 'danger',
cancelText: 'No',
async onOk() {
await apiInterceptors(delSpace({ name: space?.name }));
getSpaces();
},
});
};
const onSearch = async (e: any) => {
getSpaces({ name: e.target.value });
};
return (
<ConstructLayout>
<Spin spinning={loading}>
<div className='page-body p-4 md:p-6 h-[90vh] overflow-auto'>
{/* <Button
type="primary"
className="flex items-center"
icon={<PlusOutlined />}
onClick={() => {
setIsAddShow(true);
}}
>
Create
</Button> */}
<div className='flex justify-between items-center mb-6'>
<div className='flex items-center gap-4'>
<Input
variant='filled'
prefix={<SearchOutlined />}
placeholder={t('please_enter_the_keywords')}
onChange={debounce(onSearch, 300)}
allowClear
className='w-[230px] h-[40px] border-1 border-white backdrop-filter backdrop-blur-lg bg-white bg-opacity-30 dark:border-[#6f7f95] dark:bg-[#6f7f95] dark:bg-opacity-60'
/>
</div>
<div className='flex items-center gap-4'>
<Button
className='border-none text-white bg-button-gradient'
icon={<PlusOutlined />}
onClick={() => {
setIsAddShow(true);
}}
>
{t('create_knowledge')}
</Button>
</div>
</div>
<div className='flex flex-wrap mt-4 mx-[-8px]'>
{spaceList?.map((space: ISpace) => (
<BlurredCard
onClick={() => {
setCurrentSpace(space);
setIsPanelShow(true);
localStorage.setItem('cur_space_id', JSON.stringify(space.id));
}}
description={space.desc}
name={space.name}
key={space.id}
logo={
space.domain_type === 'FinancialReport'
? '/models/fin_report.jpg'
: space.vector_type === 'KnowledgeGraph'
? '/models/knowledge-graph.png'
: space.vector_type === 'FullText'
? '/models/knowledge-full-text.jpg'
: '/models/knowledge-default.jpg'
}
RightTop={
<InnerDropdown
menu={{
items: [
{
key: 'del',
label: (
<span className='text-red-400' onClick={() => showDeleteConfirm(space)}>
{t('Delete')}
</span>
),
},
],
}}
/>
}
rightTopHover={false}
Tags={
<div className='flex item-center'>
<Tag>
<span className='flex items-center gap-1'>
<ReadOutlined className='mt-[1px]' />
{space.docs}
</span>
</Tag>
<Tag>
<span className='flex items-center gap-1'>{space.domain_type || 'Normal'}</span>
</Tag>
{space.vector_type ? (
<Tag>
<span className='flex items-center gap-1'>{space.vector_type}</span>
</Tag>
) : null}
</div>
}
LeftBottom={
<div className='flex gap-2'>
<span>{space.owner}</span>
<span></span>
{space?.gmt_modified && <span>{moment(space?.gmt_modified).fromNow() + ' ' + t('update')}</span>}
</div>
}
RightBottom={
<ChatButton
text={t('start_chat')}
onClick={() => {
handleChat(space);
}}
/>
}
/>
))}
</div>
</div>
<Modal
className='h-5/6 overflow-hidden'
open={isPanelShow}
width={'70%'}
onCancel={() => setIsPanelShow(false)}
footer={null}
destroyOnClose={true}
>
<DocPanel space={currentSpace!} onAddDoc={onAddDoc} onDeleteDoc={getSpaces} addStatus={addStatus} />
</Modal>
<Modal
title={t('New_knowledge_base')}
centered
open={isAddShow}
destroyOnClose={true}
onCancel={() => {
setIsAddShow(false);
}}
width={1000}
afterClose={() => {
setActiveStep(0);
getSpaces();
}}
footer={null}
>
<Steps current={activeStep} items={addKnowledgeSteps} />
{activeStep === 0 && <SpaceForm handleStepChange={handleStepChange} spaceConfig={spaceConfig} />}
{activeStep === 1 && <DocTypeForm handleStepChange={handleStepChange} />}
<DocUploadForm
className={classNames({ hidden: activeStep !== 2 })}
spaceName={spaceName}
docType={docType}
handleStepChange={handleStepChange}
/>
{activeStep === 3 && (
<Segmentation
spaceName={spaceName}
docType={docType}
uploadFiles={files}
handleStepChange={handleStepChange}
/>
)}
</Modal>
</Spin>
</ConstructLayout>
);
};
export default Knowledge;