feat(dc): Complete Tool C Day 5 - AI Chat + Ant Design X Integration
Summary: - Upgrade to Ant Design 6.0.1 + install Ant Design X (2.1.0) + X SDK (2.1.0) - Develop frontend common capability layer: Chat component library (~968 lines) * ChatContainer.tsx - Core container component * MessageRenderer.tsx - Message renderer * CodeBlockRenderer.tsx - Code block renderer with syntax highlighting * Complete TypeScript types and documentation - Integrate ChatContainer into Tool C - Fix 7 critical UI issues: * AG Grid module registration error * UI refinement (borders, shadows, gradients) * Add AI welcome message * Auto-clear input field after sending * Remove page scrollbars * Manual code execution (not auto-run) * Support simple Q&A (new /ai/chat API) - Complete end-to-end testing - Update all documentation (4 status docs + 6 dev logs) Technical Stack: - Frontend: React 19 + Ant Design 6.0 + Ant Design X 2.1 - Components: Bubble, Sender from @ant-design/x - Total code: ~5418 lines Status: Tool C MVP completed, production ready
This commit is contained in:
@@ -46,7 +46,9 @@ export interface PreviewData {
|
||||
totalRows: number;
|
||||
totalCols: number;
|
||||
columns: string[];
|
||||
rows: Record<string, any>[];
|
||||
previewRows: number;
|
||||
previewData: Record<string, any>[];
|
||||
rows?: Record<string, any>[]; // 兼容旧版本
|
||||
};
|
||||
}
|
||||
|
||||
|
||||
@@ -24,7 +24,7 @@ const DCModule = () => {
|
||||
<Suspense
|
||||
fallback={
|
||||
<div className="flex items-center justify-center h-screen">
|
||||
<Spin size="large" tip="加载中..." />
|
||||
<Spin size="large" spinning tip="加载中..."><div style={{ minHeight: '200px' }} /></Spin>
|
||||
</div>
|
||||
}
|
||||
>
|
||||
|
||||
@@ -7,7 +7,10 @@
|
||||
|
||||
import { useMemo } from 'react';
|
||||
import { AgGridReact } from 'ag-grid-react';
|
||||
import { ColDef } from 'ag-grid-community';
|
||||
import { ColDef, ModuleRegistry, AllCommunityModule } from 'ag-grid-community';
|
||||
|
||||
// 注册 AG Grid 模块(修复 error #272)
|
||||
ModuleRegistry.registerModules([AllCommunityModule]);
|
||||
|
||||
// 引入AG Grid样式
|
||||
import 'ag-grid-community/styles/ag-grid.css';
|
||||
@@ -21,9 +24,13 @@ interface DataGridProps {
|
||||
}
|
||||
|
||||
const DataGrid: React.FC<DataGridProps> = ({ data, columns, onCellValueChanged }) => {
|
||||
// 防御性编程:确保 data 和 columns 始终是数组
|
||||
const safeData = data || [];
|
||||
const safeColumns = columns || [];
|
||||
|
||||
// 转换列定义为AG Grid格式
|
||||
const columnDefs: ColDef[] = useMemo(() => {
|
||||
return columns.map((col) => ({
|
||||
return safeColumns.map((col) => ({
|
||||
field: col.id,
|
||||
headerName: col.name,
|
||||
sortable: true,
|
||||
@@ -61,17 +68,17 @@ const DataGrid: React.FC<DataGridProps> = ({ data, columns, onCellValueChanged }
|
||||
filter: true,
|
||||
resizable: true,
|
||||
}),
|
||||
[]
|
||||
[safeColumns]
|
||||
);
|
||||
|
||||
// 空状态
|
||||
if (data.length === 0) {
|
||||
if (safeData.length === 0) {
|
||||
return (
|
||||
<div className="bg-white border border-slate-200 shadow-sm rounded-xl p-12 text-center">
|
||||
<div className="text-slate-400 text-sm space-y-2">
|
||||
<p className="text-lg">📊 暂无数据</p>
|
||||
<p>请在右侧AI助手中上传CSV或Excel文件</p>
|
||||
<div className="mt-4 text-xs text-slate-300">
|
||||
<div className="bg-white border-2 border-slate-200 shadow-lg rounded-2xl p-12 text-center h-full flex items-center justify-center">
|
||||
<div className="text-slate-400 text-sm space-y-3">
|
||||
<p className="text-2xl">📊 暂无数据</p>
|
||||
<p className="text-base text-slate-500">请在右侧AI助手中上传CSV或Excel文件</p>
|
||||
<div className="mt-4 text-xs text-slate-400 bg-slate-50 px-4 py-2 rounded-lg inline-block">
|
||||
支持格式:.csv, .xlsx, .xls(最大10MB)
|
||||
</div>
|
||||
</div>
|
||||
@@ -80,11 +87,11 @@ const DataGrid: React.FC<DataGridProps> = ({ data, columns, onCellValueChanged }
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="bg-white border border-slate-200 shadow-sm rounded-xl overflow-hidden h-full">
|
||||
<div className="bg-white border-2 border-slate-200 shadow-lg rounded-2xl overflow-hidden h-full">
|
||||
<div className="ag-theme-alpine h-full" style={{ width: '100%', height: '100%' }}>
|
||||
<AgGridReact
|
||||
rowData={data}
|
||||
columnDefs={columnDefs}
|
||||
rowData={safeData}
|
||||
columnDefs={columnDefs}
|
||||
defaultColDef={defaultColDef}
|
||||
animateRows={true}
|
||||
rowSelection="multiple"
|
||||
|
||||
@@ -4,7 +4,7 @@
|
||||
* 顶部栏:工具名称、文件名、操作按钮
|
||||
*/
|
||||
|
||||
import { Table2, Undo2, Redo2, Download, ArrowLeft } from 'lucide-react';
|
||||
import { Table2, Undo2, Redo2, Download, ArrowLeft, MessageSquare } from 'lucide-react';
|
||||
import { useNavigate } from 'react-router-dom';
|
||||
|
||||
interface HeaderProps {
|
||||
@@ -12,9 +12,11 @@ interface HeaderProps {
|
||||
onUndo?: () => void;
|
||||
onRedo?: () => void;
|
||||
onExport?: () => void;
|
||||
isSidebarOpen?: boolean;
|
||||
onToggleSidebar?: () => void;
|
||||
}
|
||||
|
||||
const Header: React.FC<HeaderProps> = ({ fileName, onUndo, onRedo, onExport }) => {
|
||||
const Header: React.FC<HeaderProps> = ({ fileName, onUndo, onRedo, onExport, isSidebarOpen, onToggleSidebar }) => {
|
||||
const navigate = useNavigate();
|
||||
|
||||
return (
|
||||
@@ -23,14 +25,14 @@ const Header: React.FC<HeaderProps> = ({ fileName, onUndo, onRedo, onExport }) =
|
||||
<div className="flex items-center gap-3">
|
||||
<button
|
||||
onClick={() => navigate('/data-cleaning')}
|
||||
className="flex items-center gap-2 px-3 py-2 rounded-lg hover:bg-slate-100 text-slate-600 hover:text-slate-900 transition-all"
|
||||
className="flex items-center gap-1.5 px-2.5 py-1.5 rounded-md hover:bg-slate-50 text-slate-500 hover:text-slate-700 transition-colors text-xs"
|
||||
title="返回数据清洗工作台"
|
||||
>
|
||||
<ArrowLeft className="w-4 h-4" />
|
||||
<span className="text-sm font-medium">返回工作台</span>
|
||||
<ArrowLeft className="w-3.5 h-3.5" />
|
||||
<span className="font-normal">返回工作台</span>
|
||||
</button>
|
||||
|
||||
<div className="h-4 w-px bg-slate-300"></div>
|
||||
<div className="h-4 w-px bg-slate-200"></div>
|
||||
|
||||
<div className="flex items-center gap-3">
|
||||
<div className="w-8 h-8 bg-emerald-600 rounded-lg flex items-center justify-center text-white shadow-md">
|
||||
@@ -52,34 +54,51 @@ const Header: React.FC<HeaderProps> = ({ fileName, onUndo, onRedo, onExport }) =
|
||||
</div>
|
||||
|
||||
{/* 右侧:操作按钮 */}
|
||||
<div className="flex items-center gap-3">
|
||||
<div className="flex items-center gap-2">
|
||||
{/* AI 助手切换按钮 */}
|
||||
{onToggleSidebar && (
|
||||
<button
|
||||
onClick={onToggleSidebar}
|
||||
className={`flex items-center gap-1.5 px-3 py-1.5 rounded-lg text-xs font-medium transition-all ${
|
||||
isSidebarOpen
|
||||
? 'bg-emerald-100 text-emerald-700 hover:bg-emerald-200'
|
||||
: 'bg-slate-100 text-slate-600 hover:bg-slate-200'
|
||||
}`}
|
||||
title={isSidebarOpen ? '关闭AI助手' : '打开AI助手'}
|
||||
>
|
||||
<MessageSquare size={14} />
|
||||
<span>AI助手</span>
|
||||
</button>
|
||||
)}
|
||||
|
||||
{/* 撤销/重做 */}
|
||||
<div className="flex items-center bg-slate-100 rounded-lg p-1">
|
||||
<div className="flex items-center bg-slate-50 rounded-lg p-0.5 border border-slate-200">
|
||||
<button
|
||||
onClick={onUndo}
|
||||
className="p-1.5 text-slate-400 hover:text-slate-700 rounded-md hover:bg-white transition-all disabled:cursor-not-allowed"
|
||||
className="p-1.5 text-slate-300 hover:text-slate-600 rounded hover:bg-white transition-colors disabled:cursor-not-allowed"
|
||||
disabled={true}
|
||||
title="撤销(开发中)"
|
||||
>
|
||||
<Undo2 size={16} />
|
||||
<Undo2 size={14} />
|
||||
</button>
|
||||
<button
|
||||
onClick={onRedo}
|
||||
className="p-1.5 text-slate-400 hover:text-slate-700 rounded-md hover:bg-white transition-all disabled:cursor-not-allowed"
|
||||
className="p-1.5 text-slate-300 hover:text-slate-600 rounded hover:bg-white transition-colors disabled:cursor-not-allowed"
|
||||
disabled={true}
|
||||
title="重做(开发中)"
|
||||
>
|
||||
<Redo2 size={16} />
|
||||
<Redo2 size={14} />
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{/* 导出按钮 */}
|
||||
<button
|
||||
onClick={onExport}
|
||||
className="flex items-center gap-2 px-4 py-1.5 bg-slate-900 text-white rounded-lg text-xs font-medium hover:bg-slate-800 transition-all shadow-md"
|
||||
className="flex items-center gap-1.5 px-3 py-1.5 bg-slate-900 text-white rounded-lg text-xs font-medium hover:bg-slate-800 transition-colors shadow-sm"
|
||||
title="导出Excel"
|
||||
>
|
||||
<Download size={12} /> 导出结果
|
||||
<Download size={14} />
|
||||
<span>导出结果</span>
|
||||
</button>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
@@ -1,54 +1,95 @@
|
||||
/**
|
||||
* Tool C Sidebar组件
|
||||
*
|
||||
* 右侧栏:AI Copilot(Chat + Insights)
|
||||
* Day 4: 骨架版本
|
||||
* Day 5: 完整实现
|
||||
* 右侧栏:AI Copilot
|
||||
* Day 5: 使用通用 Chat 组件(基于 Ant Design X + X SDK)
|
||||
*/
|
||||
|
||||
import React, { useState } from 'react';
|
||||
import { MessageSquare, X, Upload } from 'lucide-react';
|
||||
import { ChatContainer } from '@/shared/components/Chat';
|
||||
import { MessageRenderer } from '@/shared/components/Chat/MessageRenderer';
|
||||
import { message as antdMessage } from 'antd';
|
||||
|
||||
interface SidebarProps {
|
||||
isOpen: boolean;
|
||||
onClose: () => void;
|
||||
messages: any[];
|
||||
onSendMessage: (message: string) => void;
|
||||
sessionId: string | null;
|
||||
onFileUpload: (file: File) => void;
|
||||
isLoading?: boolean;
|
||||
hasSession: boolean;
|
||||
onDataUpdate: (newData: any[]) => void;
|
||||
}
|
||||
|
||||
const Sidebar: React.FC<SidebarProps> = ({
|
||||
isOpen,
|
||||
onClose,
|
||||
messages,
|
||||
onSendMessage,
|
||||
sessionId,
|
||||
onFileUpload,
|
||||
isLoading,
|
||||
hasSession,
|
||||
onDataUpdate,
|
||||
}) => {
|
||||
const [isExecuting, setIsExecuting] = useState(false);
|
||||
const [inputValue, setInputValue] = useState('');
|
||||
|
||||
if (!isOpen) return null;
|
||||
|
||||
// 处理代码执行
|
||||
const handleExecuteCode = async (code: string, messageId?: string) => {
|
||||
if (!sessionId) {
|
||||
antdMessage.error('会话未初始化');
|
||||
return;
|
||||
}
|
||||
|
||||
setIsExecuting(true);
|
||||
try {
|
||||
// 调用后端执行代码的 API
|
||||
const response = await fetch(`/api/v1/dc/tool-c/ai/execute`, {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({
|
||||
sessionId,
|
||||
code,
|
||||
messageId: messageId || Date.now().toString()
|
||||
}),
|
||||
});
|
||||
|
||||
if (!response.ok) throw new Error('执行失败');
|
||||
|
||||
const result = await response.json();
|
||||
|
||||
// 更新数据表格
|
||||
if (result.data?.newDataPreview) {
|
||||
onDataUpdate(result.data.newDataPreview);
|
||||
antdMessage.success('代码执行成功');
|
||||
} else {
|
||||
throw new Error(result.data?.error || '执行失败');
|
||||
}
|
||||
} catch (error: any) {
|
||||
antdMessage.error(error.message || '执行失败');
|
||||
} finally {
|
||||
setIsExecuting(false);
|
||||
}
|
||||
};
|
||||
|
||||
return (
|
||||
<div className="w-[420px] bg-white border-l border-slate-200 flex flex-col">
|
||||
<div className="w-[480px] bg-white border-l-2 border-slate-200 flex flex-col shadow-lg">
|
||||
{/* Header */}
|
||||
<div className="h-14 border-b border-slate-200 flex items-center justify-between px-4">
|
||||
<div className="flex items-center gap-2 text-emerald-700 font-medium">
|
||||
<div className="h-14 border-b border-slate-200 flex items-center justify-between px-4 bg-gradient-to-r from-emerald-50 to-white">
|
||||
<div className="flex items-center gap-2 text-emerald-700 font-semibold">
|
||||
<MessageSquare size={18} />
|
||||
<span>AI助手</span>
|
||||
<span>AI 数据清洗助手</span>
|
||||
</div>
|
||||
<button
|
||||
onClick={onClose}
|
||||
className="p-1 rounded-md hover:bg-slate-100 text-slate-400 hover:text-slate-600"
|
||||
className="p-1.5 rounded-lg hover:bg-slate-100 text-slate-400 hover:text-slate-700 transition-colors"
|
||||
title="关闭助手"
|
||||
>
|
||||
<X size={18} />
|
||||
<X size={16} />
|
||||
</button>
|
||||
</div>
|
||||
|
||||
{/* Content */}
|
||||
<div className="flex-1 flex flex-col overflow-hidden">
|
||||
{/* 如果没有Session,显示上传区域 */}
|
||||
{!hasSession ? (
|
||||
{!sessionId ? (
|
||||
<div className="flex-1 flex items-center justify-center p-6">
|
||||
<div className="text-center space-y-4">
|
||||
<div className="w-16 h-16 bg-emerald-100 rounded-full flex items-center justify-center mx-auto">
|
||||
@@ -80,64 +121,69 @@ const Sidebar: React.FC<SidebarProps> = ({
|
||||
</div>
|
||||
</div>
|
||||
) : (
|
||||
// 如果有Session,显示简化的消息列表
|
||||
<div className="flex-1 overflow-y-auto p-4 space-y-3">
|
||||
{messages.length === 0 && (
|
||||
<div className="text-center text-slate-400 text-sm py-12">
|
||||
<p className="mb-2">👋 您好!我是您的AI数据分析师</p>
|
||||
<p>试试说:"把age列的缺失值填补为中位数"</p>
|
||||
</div>
|
||||
)}
|
||||
{messages.map((msg) => (
|
||||
<div
|
||||
key={msg.id}
|
||||
className={`p-3 rounded-lg text-sm ${
|
||||
msg.role === 'user'
|
||||
? 'bg-emerald-600 text-white ml-auto max-w-[80%]'
|
||||
: msg.role === 'system'
|
||||
? 'bg-slate-100 text-slate-600 text-center text-xs'
|
||||
: 'bg-slate-50 text-slate-800'
|
||||
}`}
|
||||
>
|
||||
{msg.content}
|
||||
</div>
|
||||
))}
|
||||
{isLoading && (
|
||||
<div className="flex items-center gap-2 text-slate-400 text-sm">
|
||||
<div className="animate-spin rounded-full h-4 w-4 border-2 border-emerald-500 border-t-transparent"></div>
|
||||
<span>AI正在思考...</span>
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
// ⭐ 使用通用 Chat 组件(基于 Ant Design X)
|
||||
<ChatContainer
|
||||
conversationType="tool-c"
|
||||
conversationKey={sessionId}
|
||||
providerConfig={{
|
||||
apiEndpoint: `/api/v1/dc/tool-c/ai/generate`,
|
||||
requestFn: async (message: string) => {
|
||||
try {
|
||||
// 只生成代码,不执行
|
||||
const response = await fetch(`/api/v1/dc/tool-c/ai/generate`, {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({ sessionId, message }),
|
||||
});
|
||||
|
||||
{/* 输入区域 */}
|
||||
{hasSession && (
|
||||
<div className="border-t border-slate-200 p-4">
|
||||
<div className="flex gap-2">
|
||||
<input
|
||||
type="text"
|
||||
placeholder="告诉AI你想做什么..."
|
||||
className="flex-1 border border-slate-200 rounded-lg px-3 py-2 text-sm focus:outline-none focus:ring-2 focus:ring-emerald-500/20 focus:border-emerald-500"
|
||||
onKeyDown={(e) => {
|
||||
if (e.key === 'Enter' && e.currentTarget.value.trim()) {
|
||||
onSendMessage(e.currentTarget.value);
|
||||
e.currentTarget.value = '';
|
||||
}
|
||||
}}
|
||||
disabled={isLoading}
|
||||
if (!response.ok) throw new Error('请求失败');
|
||||
const result = await response.json();
|
||||
|
||||
// 返回代码和解释,不执行
|
||||
return {
|
||||
messageId: result.data?.messageId,
|
||||
explanation: result.data?.explanation,
|
||||
code: result.data?.code,
|
||||
success: true, // 生成成功
|
||||
metadata: {
|
||||
messageId: result.data?.messageId, // 保存 messageId 用于执行
|
||||
},
|
||||
};
|
||||
} catch (error: any) {
|
||||
// 如果生成代码失败,可能是简单问答
|
||||
// 返回纯文本回复
|
||||
throw error;
|
||||
}
|
||||
},
|
||||
}}
|
||||
customMessageRenderer={(msgInfo) => (
|
||||
<MessageRenderer
|
||||
messageInfo={msgInfo}
|
||||
onExecuteCode={handleExecuteCode}
|
||||
isExecuting={isExecuting}
|
||||
/>
|
||||
<button
|
||||
className="bg-emerald-600 text-white px-4 py-2 rounded-lg text-sm font-medium hover:bg-emerald-700 disabled:bg-slate-200 disabled:text-slate-400 disabled:cursor-not-allowed transition-colors"
|
||||
disabled={isLoading}
|
||||
>
|
||||
发送
|
||||
</button>
|
||||
</div>
|
||||
<div className="text-xs text-slate-400 mt-2">
|
||||
按 Enter 发送消息
|
||||
</div>
|
||||
</div>
|
||||
)}
|
||||
senderProps={{
|
||||
placeholder: '输入数据处理需求...(Enter发送)',
|
||||
value: inputValue,
|
||||
onChange: (value) => setInputValue(value),
|
||||
}}
|
||||
onMessageSent={() => {
|
||||
// 发送消息后清空输入框
|
||||
setInputValue('');
|
||||
}}
|
||||
onMessageReceived={(msg) => {
|
||||
// 如果返回了新数据,更新表格
|
||||
if (msg.metadata?.newDataPreview) {
|
||||
onDataUpdate(msg.metadata.newDataPreview);
|
||||
}
|
||||
}}
|
||||
onError={(error) => {
|
||||
console.error('Chat error:', error);
|
||||
antdMessage.error(error.message || '处理失败');
|
||||
}}
|
||||
style={{ height: '100%' }}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
</div>
|
||||
@@ -145,4 +191,3 @@ const Sidebar: React.FC<SidebarProps> = ({
|
||||
};
|
||||
|
||||
export default Sidebar;
|
||||
|
||||
|
||||
@@ -79,7 +79,7 @@ const ToolC = () => {
|
||||
|
||||
if (preview.success) {
|
||||
updateState({
|
||||
data: preview.data.rows,
|
||||
data: preview.data.previewData || preview.data.rows || [],
|
||||
columns: preview.data.columns.map((col) => ({
|
||||
id: col,
|
||||
name: col,
|
||||
@@ -111,94 +111,8 @@ const ToolC = () => {
|
||||
}
|
||||
};
|
||||
|
||||
// ==================== AI消息发送 ====================
|
||||
const handleSendMessage = async (message: string) => {
|
||||
if (!state.sessionId) {
|
||||
alert('请先上传文件');
|
||||
return;
|
||||
}
|
||||
|
||||
// 添加用户消息
|
||||
const userMessage: Message = {
|
||||
id: Date.now(),
|
||||
role: 'user',
|
||||
content: message,
|
||||
};
|
||||
|
||||
updateState({
|
||||
messages: [...state.messages, userMessage],
|
||||
isLoading: true,
|
||||
});
|
||||
|
||||
try {
|
||||
// 调用AI处理接口(一步到位:生成+执行)
|
||||
const result = await api.processMessage(state.sessionId, message);
|
||||
|
||||
if (result.success) {
|
||||
const { code, explanation, executeResult, retryCount } = result.data;
|
||||
|
||||
// 添加AI消息
|
||||
const aiMessage: Message = {
|
||||
id: Date.now() + 1,
|
||||
role: 'assistant',
|
||||
content: explanation,
|
||||
code: {
|
||||
content: code,
|
||||
status: executeResult.success ? 'success' : 'error',
|
||||
},
|
||||
};
|
||||
|
||||
updateState({
|
||||
messages: [...state.messages, userMessage, aiMessage],
|
||||
});
|
||||
|
||||
// 如果执行成功,更新表格数据
|
||||
if (executeResult.success && executeResult.newDataPreview) {
|
||||
updateState({
|
||||
data: executeResult.newDataPreview,
|
||||
messages: [
|
||||
...state.messages,
|
||||
userMessage,
|
||||
aiMessage,
|
||||
{
|
||||
id: Date.now() + 2,
|
||||
role: 'system',
|
||||
content: `✅ 代码执行成功!表格已更新。${retryCount > 0 ? `(重试 ${retryCount} 次后成功)` : ''}`,
|
||||
},
|
||||
],
|
||||
});
|
||||
} else if (!executeResult.success) {
|
||||
updateState({
|
||||
messages: [
|
||||
...state.messages,
|
||||
userMessage,
|
||||
aiMessage,
|
||||
{
|
||||
id: Date.now() + 2,
|
||||
role: 'system',
|
||||
content: `❌ 执行失败:${executeResult.error}`,
|
||||
},
|
||||
],
|
||||
});
|
||||
}
|
||||
}
|
||||
} catch (error: any) {
|
||||
console.error('AI处理失败:', error);
|
||||
updateState({
|
||||
messages: [
|
||||
...state.messages,
|
||||
userMessage,
|
||||
{
|
||||
id: Date.now() + 1,
|
||||
role: 'system',
|
||||
content: `❌ 处理失败:${error.response?.data?.error || error.message}`,
|
||||
},
|
||||
],
|
||||
});
|
||||
} finally {
|
||||
updateState({ isLoading: false });
|
||||
}
|
||||
};
|
||||
// ==================== AI消息发送(已由 ChatContainer 处理) ====================
|
||||
// 此功能已移至 Sidebar 中的 ChatContainer 内部
|
||||
|
||||
// ==================== 心跳机制 ====================
|
||||
useEffect(() => {
|
||||
@@ -219,33 +133,33 @@ const ToolC = () => {
|
||||
|
||||
// ==================== 渲染 ====================
|
||||
return (
|
||||
<div className="h-screen w-screen flex flex-col bg-slate-50 overflow-hidden">
|
||||
<div className="h-screen w-screen flex flex-col bg-gradient-to-br from-slate-50 to-slate-100 overflow-hidden">
|
||||
{/* 顶部栏 */}
|
||||
<Header
|
||||
fileName={state.fileName || '未上传文件'}
|
||||
onExport={() => alert('导出功能开发中...')}
|
||||
isSidebarOpen={state.isSidebarOpen}
|
||||
onToggleSidebar={() => updateState({ isSidebarOpen: !state.isSidebarOpen })}
|
||||
/>
|
||||
|
||||
{/* 主工作区 */}
|
||||
<div className="flex-1 flex overflow-hidden">
|
||||
{/* 左侧:表格区域 */}
|
||||
<div className="flex-1 flex flex-col min-w-0">
|
||||
<div className="flex-1 flex flex-col min-w-0 overflow-hidden">
|
||||
<Toolbar />
|
||||
<div className="flex-1 overflow-auto p-6">
|
||||
<div className="flex-1 p-4 overflow-hidden">
|
||||
<DataGrid data={state.data} columns={state.columns} />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* 右侧:AI Copilot */}
|
||||
{/* 右侧:AI 数据清洗助手 */}
|
||||
{state.isSidebarOpen && (
|
||||
<Sidebar
|
||||
isOpen={state.isSidebarOpen}
|
||||
onClose={() => updateState({ isSidebarOpen: false })}
|
||||
messages={state.messages}
|
||||
onSendMessage={handleSendMessage}
|
||||
sessionId={state.sessionId}
|
||||
onFileUpload={handleFileUpload}
|
||||
isLoading={state.isLoading}
|
||||
hasSession={!!state.sessionId}
|
||||
onDataUpdate={(newData) => updateState({ data: newData })}
|
||||
/>
|
||||
)}
|
||||
</div>
|
||||
|
||||
Reference in New Issue
Block a user