441 lines
15 KiB
JavaScript
441 lines
15 KiB
JavaScript
'use client';
|
||
|
||
import axios from 'axios';
|
||
import { useState, useEffect, useRef } from 'react';
|
||
import { useTranslation } from 'react-i18next';
|
||
import {
|
||
Container,
|
||
Box,
|
||
Tabs,
|
||
Tab,
|
||
IconButton,
|
||
Collapse,
|
||
Dialog,
|
||
DialogContent,
|
||
DialogTitle,
|
||
Typography,
|
||
LinearProgress,
|
||
CircularProgress
|
||
} from '@mui/material';
|
||
import { useTheme } from '@mui/material/styles';
|
||
import ExpandMoreIcon from '@mui/icons-material/ExpandMore';
|
||
import ExpandLessIcon from '@mui/icons-material/ExpandLess';
|
||
import FullscreenIcon from '@mui/icons-material/Fullscreen';
|
||
import CloseIcon from '@mui/icons-material/Close';
|
||
import FileUploader from '@/components/text-split/FileUploader';
|
||
import FileList from '@/components/text-split/components/FileList';
|
||
import DeleteConfirmDialog from '@/components/text-split/components/DeleteConfirmDialog';
|
||
import PdfSettings from '@/components/text-split/PdfSettings';
|
||
import ChunkList from '@/components/text-split/ChunkList';
|
||
import DomainAnalysis from '@/components/text-split/DomainAnalysis';
|
||
import useTaskSettings from '@/hooks/useTaskSettings';
|
||
import { useAtomValue } from 'jotai/index';
|
||
import { selectedModelInfoAtom } from '@/lib/store';
|
||
import useChunks from './useChunks';
|
||
import useQuestionGeneration from './useQuestionGeneration';
|
||
import useDataCleaning from './useDataCleaning';
|
||
import useEvalGeneration from './useEvalGeneration';
|
||
import useFileProcessing from './useFileProcessing';
|
||
import useFileProcessingStatus from '@/hooks/useFileProcessingStatus';
|
||
import { toast } from 'sonner';
|
||
|
||
export default function TextSplitPage({ params }) {
|
||
const { t } = useTranslation();
|
||
const theme = useTheme();
|
||
const { projectId } = params;
|
||
const [activeTab, setActiveTab] = useState(0);
|
||
const [renderedTab, setRenderedTab] = useState(0);
|
||
const [tabSwitching, setTabSwitching] = useState(false);
|
||
const tabSwitchTimerRef = useRef(null);
|
||
const { taskSettings } = useTaskSettings(projectId);
|
||
const [pdfStrategy, setPdfStrategy] = useState('default');
|
||
const [questionFilter, setQuestionFilter] = useState('all'); // 'all', 'generated', 'ungenerated'
|
||
const [selectedViosnModel, setSelectedViosnModel] = useState('');
|
||
const selectedModelInfo = useAtomValue(selectedModelInfoAtom);
|
||
const { taskFileProcessing, task } = useFileProcessingStatus();
|
||
const [currentPage, setCurrentPage] = useState(1);
|
||
const [uploadedFiles, setUploadedFiles] = useState({ data: [], total: 0 });
|
||
const [searchFileName, setSearchFileName] = useState('');
|
||
const [showLoadingBar, setShowLoadingBar] = useState(false);
|
||
|
||
// 娑撳﹣绱堕崠鍝勭厵閻ㄥ嫬鐫嶅鈧?閹舵ê褰旈悩鑸碘偓?
|
||
const [uploaderExpanded, setUploaderExpanded] = useState(true);
|
||
|
||
// 閺傚洨灏為崚妤勩€?FileList)鐏炴洜銇氱€电鐦藉鍡欏Ц閹?
|
||
const [fileListDialogOpen, setFileListDialogOpen] = useState(false);
|
||
|
||
// 娴h法鏁ら懛顏勭暰娑斿“ooks
|
||
const { chunks, tocData, loading, fetchChunks, handleDeleteChunk, handleEditChunk, updateChunks, setLoading } =
|
||
useChunks(projectId, questionFilter);
|
||
|
||
// 閼惧嘲褰囬弬鍥︽閸掓銆?
|
||
const fetchUploadedFiles = async (page = currentPage, fileName = searchFileName) => {
|
||
try {
|
||
setLoading(true);
|
||
const params = new URLSearchParams({
|
||
page: page.toString(),
|
||
size: '10'
|
||
});
|
||
|
||
if (fileName && fileName.trim()) {
|
||
params.append('fileName', fileName.trim());
|
||
}
|
||
|
||
const response = await axios.get(`/api/projects/${projectId}/files?${params}`);
|
||
setUploadedFiles(response.data);
|
||
} catch (error) {
|
||
console.error('Error fetching files:', error);
|
||
toast.error(error.message || '閼惧嘲褰囬弬鍥︽閸掓銆冩径杈Е');
|
||
} finally {
|
||
setLoading(false);
|
||
}
|
||
};
|
||
|
||
// 閸掔娀娅庨弬鍥︽绾喛顓荤€电鐦藉鍡欏Ц閹?
|
||
const [deleteConfirmOpen, setDeleteConfirmOpen] = useState(false);
|
||
const [fileToDelete, setFileToDelete] = useState(null);
|
||
|
||
// 閹垫挸绱戦崚鐘绘珟绾喛顓荤€电鐦藉?
|
||
const openDeleteConfirm = (fileId, fileName) => {
|
||
setFileToDelete({ fileId, fileName });
|
||
setDeleteConfirmOpen(true);
|
||
};
|
||
|
||
// 閸忔娊妫撮崚鐘绘珟绾喛顓荤€电鐦藉?
|
||
const closeDeleteConfirm = () => {
|
||
setDeleteConfirmOpen(false);
|
||
setFileToDelete(null);
|
||
};
|
||
|
||
// 绾喛顓婚崚鐘绘珟閺傚洣娆?
|
||
const confirmDeleteFile = async () => {
|
||
if (!fileToDelete) return;
|
||
|
||
try {
|
||
setLoading(true);
|
||
closeDeleteConfirm();
|
||
|
||
await axios.delete(`/api/projects/${projectId}/files/${fileToDelete.fileId}`);
|
||
await fetchUploadedFiles();
|
||
fetchChunks();
|
||
|
||
toast.success(
|
||
t('textSplit.deleteSuccess', { fileName: fileToDelete.fileName }) || `删除 ${fileToDelete.fileName} 成功`
|
||
);
|
||
} catch (error) {
|
||
console.error('删除文件出错:', error);
|
||
toast.error(error.message || '删除文件失败');
|
||
} finally {
|
||
setLoading(false);
|
||
setFileToDelete(null);
|
||
}
|
||
};
|
||
|
||
const { handleGenerateQuestions } = useQuestionGeneration(projectId, taskSettings);
|
||
const { handleDataCleaning } = useDataCleaning(projectId, taskSettings);
|
||
const { handleGenerateEvalQuestions } = useEvalGeneration(projectId);
|
||
const { handleFileProcessing } = useFileProcessing(projectId);
|
||
|
||
// 文本块数据刷新:初始化 + 文件处理任务状态变化
|
||
useEffect(() => {
|
||
fetchChunks('all');
|
||
}, [fetchChunks, taskFileProcessing]);
|
||
|
||
// 文件列表刷新:文件分页、搜索关键词变化时触发
|
||
useEffect(() => {
|
||
fetchUploadedFiles(currentPage, searchFileName);
|
||
}, [projectId, currentPage, searchFileName]);
|
||
|
||
useEffect(() => {
|
||
let timerId;
|
||
if (loading) {
|
||
timerId = setTimeout(() => setShowLoadingBar(true), 180);
|
||
} else {
|
||
setShowLoadingBar(false);
|
||
}
|
||
return () => {
|
||
if (timerId) clearTimeout(timerId);
|
||
};
|
||
}, [loading]);
|
||
|
||
useEffect(() => {
|
||
return () => {
|
||
if (tabSwitchTimerRef.current) {
|
||
clearTimeout(tabSwitchTimerRef.current);
|
||
}
|
||
};
|
||
}, []);
|
||
|
||
const handleTabChange = (event, newValue) => {
|
||
if (newValue === activeTab) return;
|
||
|
||
setActiveTab(newValue);
|
||
setTabSwitching(true);
|
||
|
||
if (tabSwitchTimerRef.current) {
|
||
clearTimeout(tabSwitchTimerRef.current);
|
||
}
|
||
|
||
const switchContent = () => {
|
||
setRenderedTab(newValue);
|
||
tabSwitchTimerRef.current = null;
|
||
if (typeof window !== 'undefined') {
|
||
window.requestAnimationFrame(() => setTabSwitching(false));
|
||
} else {
|
||
setTabSwitching(false);
|
||
}
|
||
};
|
||
|
||
if (typeof window !== 'undefined') {
|
||
window.requestAnimationFrame(() => {
|
||
tabSwitchTimerRef.current = setTimeout(switchContent, 80);
|
||
});
|
||
} else {
|
||
switchContent();
|
||
}
|
||
};
|
||
|
||
/**
|
||
* 鐎甸€涚瑐娴肩姴鎮楅惃鍕瀮娴犳儼绻樼悰灞筋槱閻?
|
||
*/
|
||
const handleUploadSuccess = async (fileNames, pdfFiles, domainTreeAction) => {
|
||
try {
|
||
await handleFileProcessing(fileNames, pdfStrategy, selectedViosnModel, domainTreeAction);
|
||
location.reload();
|
||
} catch (error) {
|
||
toast.error('File upload failed' + error.message || '');
|
||
}
|
||
};
|
||
|
||
// 閸栧懓顥婇悽鐔稿灇闂傤噣顣介惃鍕槱閻炲棗鍤遍弫?
|
||
const onGenerateQuestions = async chunkIds => {
|
||
await handleGenerateQuestions(chunkIds, selectedModelInfo, fetchChunks);
|
||
};
|
||
|
||
// 閸栧懓顥婇弫鐗堝祦濞撳懏绀傞惃鍕槱閻炲棗鍤遍弫?
|
||
const onDataCleaning = async chunkIds => {
|
||
await handleDataCleaning(chunkIds, selectedModelInfo, fetchChunks);
|
||
};
|
||
|
||
// 閸栧懓顥婇悽鐔稿灇濞村鐦庢0妯兼窗閻ㄥ嫬顦╅悶鍡楀毐閺?
|
||
const onGenerateEvalQuestions = async chunkId => {
|
||
await handleGenerateEvalQuestions(chunkId, selectedModelInfo, () => {
|
||
// 閹存劕濮涢崥搴″煕閺傛澘鍨悰?
|
||
fetchChunks();
|
||
});
|
||
};
|
||
|
||
useEffect(() => {
|
||
const url = new URL(window.location.href);
|
||
if (questionFilter !== 'all') {
|
||
url.searchParams.set('filter', questionFilter);
|
||
} else {
|
||
url.searchParams.delete('filter');
|
||
}
|
||
window.history.replaceState({}, '', url);
|
||
fetchChunks(questionFilter);
|
||
}, [questionFilter]);
|
||
|
||
const handleSelected = array => {
|
||
if (array.length > 0) {
|
||
axios.post(`/api/projects/${projectId}/chunks`, { array }).then(response => {
|
||
updateChunks(response.data);
|
||
});
|
||
} else {
|
||
fetchChunks();
|
||
}
|
||
};
|
||
|
||
return (
|
||
<Container maxWidth="lg" sx={{ mt: 4, mb: 8, position: 'relative' }}>
|
||
{/* 閺傚洣娆㈡稉濠佺炊缂佸嫪娆?*/}
|
||
|
||
<Box
|
||
sx={{ position: 'absolute', top: -18, left: '50%', transform: 'translateX(-50%)', zIndex: 1, display: 'flex' }}
|
||
>
|
||
<IconButton
|
||
onClick={() => setUploaderExpanded(!uploaderExpanded)}
|
||
sx={{
|
||
bgcolor: 'background.paper',
|
||
boxShadow: 1,
|
||
mr: uploaderExpanded ? 1 : 0 // 鐏炴洖绱戦弮鑸靛瘻闁筋喕绠i梻瀵告殌閻愬綊妫跨捄?
|
||
}}
|
||
size="small"
|
||
>
|
||
{uploaderExpanded ? <ExpandLessIcon /> : <ExpandMoreIcon />}
|
||
</IconButton>
|
||
|
||
{/* 閺傚洨灏為崚妤勩€冮幍鈺佺潔閹稿鎸抽敍灞肩矌閸︺劋绗傞柈銊ュ隘閸╃喎鐫嶅鈧弮鑸垫▔缁€?*/}
|
||
{uploaderExpanded && (
|
||
<IconButton
|
||
color="primary"
|
||
onClick={() => setFileListDialogOpen(true)}
|
||
sx={{ bgcolor: 'background.paper', boxShadow: 1 }}
|
||
size="small"
|
||
title={t('textSplit.expandFileList') || '扩展文件列表'}
|
||
>
|
||
<FullscreenIcon />
|
||
</IconButton>
|
||
)}
|
||
</Box>
|
||
|
||
<Collapse in={uploaderExpanded}>
|
||
<FileUploader
|
||
projectId={projectId}
|
||
onUploadSuccess={handleUploadSuccess}
|
||
onFileDeleted={fetchChunks}
|
||
setPageLoading={setLoading}
|
||
sendToPages={handleSelected}
|
||
setPdfStrategy={setPdfStrategy}
|
||
pdfStrategy={pdfStrategy}
|
||
selectedViosnModel={selectedViosnModel}
|
||
setSelectedViosnModel={setSelectedViosnModel}
|
||
taskFileProcessing={taskFileProcessing}
|
||
fileTask={task}
|
||
>
|
||
<PdfSettings
|
||
pdfStrategy={pdfStrategy}
|
||
setPdfStrategy={setPdfStrategy}
|
||
selectedViosnModel={selectedViosnModel}
|
||
setSelectedViosnModel={setSelectedViosnModel}
|
||
/>
|
||
</FileUploader>
|
||
</Collapse>
|
||
|
||
{/* 閺嶅洨顒锋い?*/}
|
||
<Box sx={{ width: '100%', mb: 3 }}>
|
||
<Box sx={{ display: 'flex', alignItems: 'center', mb: 1 }}>
|
||
<Tabs
|
||
value={activeTab}
|
||
onChange={handleTabChange}
|
||
variant="fullWidth"
|
||
sx={{ borderBottom: 1, borderColor: 'divider', flexGrow: 1 }}
|
||
>
|
||
<Tab label={t('textSplit.tabs.smartSplit')} />
|
||
<Tab label={t('textSplit.tabs.domainAnalysis')} />
|
||
</Tabs>
|
||
</Box>
|
||
|
||
{/* 閺呴缚鍏橀崚鍡楀閺嶅洨顒烽崘鍛啇 */}
|
||
{tabSwitching ? (
|
||
<Box
|
||
sx={{
|
||
minHeight: 220,
|
||
display: 'flex',
|
||
alignItems: 'center',
|
||
justifyContent: 'center',
|
||
flexDirection: 'column',
|
||
gap: 1.5
|
||
}}
|
||
>
|
||
<CircularProgress size={26} />
|
||
<Typography variant="body2" color="text.secondary">
|
||
{t('common.loading')}
|
||
</Typography>
|
||
</Box>
|
||
) : (
|
||
<>
|
||
{renderedTab === 0 && (
|
||
<ChunkList
|
||
projectId={projectId}
|
||
chunks={chunks}
|
||
onDelete={handleDeleteChunk}
|
||
onEdit={handleEditChunk}
|
||
onGenerateQuestions={onGenerateQuestions}
|
||
onGenerateEvalQuestions={onGenerateEvalQuestions}
|
||
onDataCleaning={onDataCleaning}
|
||
loading={loading}
|
||
questionFilter={questionFilter}
|
||
setQuestionFilter={setQuestionFilter}
|
||
selectedModel={selectedModelInfo}
|
||
/>
|
||
)}
|
||
|
||
{renderedTab === 1 && <DomainAnalysis projectId={projectId} toc={tocData} loading={loading} />}
|
||
</>
|
||
)}
|
||
</Box>
|
||
|
||
{/* 閸旂姾娴囨稉顓℃寢閻?*/}
|
||
{showLoadingBar && (
|
||
<Box sx={{ position: 'sticky', bottom: 12, zIndex: 5, px: 1 }}>
|
||
<Box
|
||
sx={{
|
||
bgcolor: 'background.paper',
|
||
border: 1,
|
||
borderColor: 'divider',
|
||
borderRadius: 2,
|
||
px: 1.5,
|
||
py: 1,
|
||
boxShadow: 1
|
||
}}
|
||
>
|
||
<Typography variant="caption" color="text.secondary" sx={{ display: 'block', mb: 0.5 }}>
|
||
{t('textSplit.loading')}
|
||
</Typography>
|
||
<LinearProgress />
|
||
</Box>
|
||
</Box>
|
||
)}
|
||
|
||
{/* 婢跺嫮鎮婃稉顓℃寢閻?*/}
|
||
|
||
{/* 閺佺増宓佸〒鍛鏉╂稑瀹抽拏娆戝 */}
|
||
|
||
{/* 閺傚洣娆㈡径鍕倞鏉╂稑瀹抽拏娆戝 */}
|
||
|
||
{/* 閺傚洣娆㈤崚鐘绘珟绾喛顓荤€电鐦藉?*/}
|
||
<DeleteConfirmDialog
|
||
open={deleteConfirmOpen}
|
||
fileName={fileToDelete?.fileName}
|
||
onClose={closeDeleteConfirm}
|
||
onConfirm={confirmDeleteFile}
|
||
/>
|
||
|
||
{/* 閺傚洨灏為崚妤勩€冪€电鐦藉?*/}
|
||
<Dialog
|
||
open={fileListDialogOpen}
|
||
onClose={() => setFileListDialogOpen(false)}
|
||
maxWidth="lg"
|
||
fullWidth
|
||
sx={{ '& .MuiDialog-paper': { bgcolor: 'background.default' } }}
|
||
>
|
||
<DialogTitle sx={{ display: 'flex', justifyContent: 'space-between', alignItems: 'center', px: 3, py: 1 }}>
|
||
<Typography variant="h6">{t('textSplit.fileList')}</Typography>
|
||
<IconButton edge="end" color="inherit" onClick={() => setFileListDialogOpen(false)} aria-label="close">
|
||
<CloseIcon />
|
||
</IconButton>
|
||
</DialogTitle>
|
||
<DialogContent dividers sx={{ p: 3 }}>
|
||
{/* 濮濄倕顦╂径宥囨暏 FileUploader 缂佸嫪娆㈡稉顓犳畱 FileList 闁劌鍨?*/}
|
||
<Box sx={{ minHeight: '80vh' }}>
|
||
{/* 閺傚洣娆㈤崚妤勩€冮崘鍛啇 */}
|
||
<FileList
|
||
theme={theme}
|
||
files={uploadedFiles}
|
||
loading={loading}
|
||
setPageLoading={setLoading}
|
||
sendToFileUploader={array => handleSelected(array)}
|
||
onDeleteFile={(fileId, fileName) => openDeleteConfirm(fileId, fileName)}
|
||
projectId={projectId}
|
||
currentPage={currentPage}
|
||
onPageChange={(page, fileName) => {
|
||
if (fileName !== undefined) {
|
||
// 閹兼粎鍌ㄩ弮鑸垫纯閺傜増鎮崇槐銏犲彠闁款喛鐦濋崪宀勩€夐惍?
|
||
setSearchFileName(fileName);
|
||
setCurrentPage(page);
|
||
} else {
|
||
// 缂堝銆夐弮璺哄涧閺囧瓨鏌婃い鐢电垳
|
||
setCurrentPage(page);
|
||
}
|
||
}}
|
||
onRefresh={fetchUploadedFiles} // 娴肩娀鈧帒鍩涢弬鏉垮毐閺?
|
||
isFullscreen={true} // 閸︺劌顕拠婵囶攱娑擃厾些闂勩倝鐝惔锕傛閸?
|
||
/>
|
||
</Box>
|
||
</DialogContent>
|
||
</Dialog>
|
||
</Container>
|
||
);
|
||
}
|