176 lines
5.9 KiB
TypeScript
176 lines
5.9 KiB
TypeScript
/*
|
|
* @Author: ZhaoYing
|
|
* @Date: 2026-02-03 14:10:20
|
|
* @Last Modified by: ZhaoYing
|
|
* @Last Modified time: 2026-04-14 16:55:26
|
|
*/
|
|
import { type FC, useEffect, useState, useRef } from 'react'
|
|
import { useParams, useNavigate } from 'react-router-dom';
|
|
import { useTranslation } from 'react-i18next';
|
|
import { App, Row, Col, Tooltip, Space, Button, Flex } from 'antd'
|
|
|
|
import PageHeader from '@/components/Layout/PageHeader'
|
|
import { getOntologyClassList, deleteOntologyClass } from '@/api/ontology'
|
|
import type { OntologyClassData, OntologyClassModalRef, OntologyClassExtractModalRef, OntologyClassItem } from '@/views/Ontology/types'
|
|
import RbCard from '@/components/RbCard';
|
|
import OntologyClassModal from '../components/OntologyClassModal'
|
|
import SearchInput from '@/components/SearchInput';
|
|
import OntologyClassExtractModal from '../components/OntologyClassExtractModal'
|
|
import BodyWrapper from '@/components/Empty/BodyWrapper'
|
|
import Tag from '@/components/Tag'
|
|
import { useI18n } from '@/store/locale'
|
|
|
|
/**
|
|
* Ontology detail page component
|
|
* Displays and manages classes within a specific ontology scene
|
|
*/
|
|
const Detail: FC = () => {
|
|
// Hooks
|
|
const { t } = useTranslation();
|
|
const navigate = useNavigate()
|
|
const { id } = useParams()
|
|
const { modal, message } = App.useApp()
|
|
const { language } = useI18n()
|
|
|
|
// Refs
|
|
const ontologyClassModalRef = useRef<OntologyClassModalRef>(null)
|
|
const ontologyClassExtractModalRef = useRef<OntologyClassExtractModalRef>(null)
|
|
|
|
// State
|
|
const [query, setQuery] = useState<{
|
|
class_name?: string;
|
|
}>({});
|
|
const [loading, setLoading] = useState(false)
|
|
const [data, setData] = useState<OntologyClassData>({} as OntologyClassData)
|
|
|
|
// Fetch data when component mounts or dependencies change
|
|
useEffect(() => {
|
|
getData()
|
|
}, [id, query])
|
|
|
|
useEffect(() => {
|
|
document.title = `${data.scene_name} - ${t('memoryBear')}`;
|
|
}, [data.scene_name, language])
|
|
|
|
/**
|
|
* Fetch ontology class list data
|
|
*/
|
|
const getData = () => {
|
|
if (!id) return;
|
|
setLoading(true)
|
|
getOntologyClassList({
|
|
...query,
|
|
scene_id: id
|
|
})
|
|
.then(res => {
|
|
setData(res as OntologyClassData)
|
|
})
|
|
.finally(() => {
|
|
setLoading(false)
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Delete an ontology class with confirmation
|
|
* @param item - The class item to delete
|
|
*/
|
|
const handleDelete = (item: OntologyClassItem) => {
|
|
modal.confirm({
|
|
title: t('common.confirmDeleteDesc', { name: item.class_name }),
|
|
okText: t('common.delete'),
|
|
cancelText: t('common.cancel'),
|
|
okType: 'danger',
|
|
onOk: () => {
|
|
deleteOntologyClass(item.class_id)
|
|
.then(() => {
|
|
getData();
|
|
message.success(t('common.deleteSuccess'))
|
|
})
|
|
}
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Open modal to add a new class
|
|
*/
|
|
const handleAdd = () => {
|
|
ontologyClassModalRef.current?.handleOpen(data.scene_id)
|
|
}
|
|
|
|
/**
|
|
* Open modal to extract classes using LLM
|
|
*/
|
|
const handleExtract = () => {
|
|
ontologyClassExtractModalRef.current?.handleOpen(data)
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<Flex vertical className="rb:h-screen!">
|
|
<PageHeader
|
|
title={<Space>
|
|
{data.scene_name}
|
|
{data.is_system_default ? <Tag color="warning">{t('common.default')}</Tag> : undefined}
|
|
<Tooltip title={data.scene_description}>
|
|
<div className="rb:size-4 rb:bg-cover rb:bg-[url('@/assets/images/common/question.svg')]"></div>
|
|
</Tooltip>
|
|
</Space>}
|
|
extra={<Space size={12}>
|
|
{data.is_system_default ? undefined : (<Space>
|
|
<Button type="primary" ghost className="rb:h-6! rb:px-2! rb:leading-5.5!" onClick={handleAdd}>+ {t('ontology.addClass')}</Button>
|
|
<Button className="rb:h-6! rb:px-2! rb:leading-5.5!" type="primary" onClick={handleExtract}>+ {t('ontology.extract')}</Button>
|
|
</Space>)}
|
|
<Flex align="center" className="rb:leading-5 rb:text-[14px] rb:text-[#5B6167] rb:font-regular rb:cursor-pointer" onClick={() => navigate(-1)}>
|
|
<div
|
|
className="rb:mr-2 rb:size-4 rb:cursor-pointer rb:bg-cover rb:bg-[url('@/assets/images/logout_grey.svg')]"
|
|
></div>
|
|
{t('common.return')}
|
|
</Flex>
|
|
</Space>}
|
|
/>
|
|
|
|
<div className="rb:flex-1 rb:p-3 rb:overflow-hidden">
|
|
<Row gutter={12} className="rb:mb-4">
|
|
<Col span={6} offset={18}>
|
|
<SearchInput
|
|
placeholder={t('ontology.classSearchPlaceholder')}
|
|
onSearch={(value) => setQuery({ class_name: value })}
|
|
className="rb:w-full!"
|
|
/>
|
|
</Col>
|
|
</Row>
|
|
<BodyWrapper loading={loading} empty={!data.items?.length}>
|
|
<Row gutter={[16, 16]} className="rb:max-h-[calc(100%-48px)] rb:overflow-y-auto">
|
|
{data.items?.map(item => (
|
|
<Col key={item.class_id} span={6}>
|
|
<RbCard
|
|
title={item.class_name}
|
|
extra={data.is_system_default ? undefined : (<div
|
|
className="rb:size-5 rb:cursor-pointer rb:bg-cover rb:bg-[url('@/assets/images/common/delete.svg')] rb:hover:bg-[url('@/assets/images/common/delete_hover.svg')]"
|
|
onClick={() => handleDelete(item)}
|
|
></div>)}
|
|
>
|
|
<Tooltip title={item.class_description}>
|
|
<div className="rb:h-10 rb:text-[#5B6167] rb:leading-5 rb:font-regular rb:wrap-break-word rb:line-clamp-2">{item.class_description}</div>
|
|
</Tooltip>
|
|
</RbCard>
|
|
</Col>
|
|
))}
|
|
</Row>
|
|
</BodyWrapper>
|
|
</div>
|
|
</Flex>
|
|
|
|
<OntologyClassModal
|
|
ref={ontologyClassModalRef}
|
|
refresh={getData}
|
|
/>
|
|
<OntologyClassExtractModal
|
|
ref={ontologyClassExtractModalRef}
|
|
refresh={getData}
|
|
/>
|
|
</>
|
|
)
|
|
}
|
|
|
|
export default Detail |