Files
DataMate/frontend/src/pages/Layout/Sidebar.tsx
Jerry Yan 0c23fd6c23 fix(import): 修复文件上传配置和表单状态管理问题
- 移除手动fileList状态管理,改用Form组件内置字段
- 修复重置状态时的初始值设置,确保hasArchive和splitByLine默认值正确
- 更新文件上传验证逻辑,使用form.getFieldValue获取文件列表
- 修改拖拽上传组件配置,移除不必要的回调函数
- 修复按钮禁用条件判断,使用正确的字段路径检查文件长度
- 移除表单项的initialValue配置,统一在state初始化时设置默认值
- 调整Sidebar组件中的注释标记,修复任务中心弹窗显示逻辑
2026-01-21 00:56:09 +08:00

204 lines
6.1 KiB
TypeScript

import { memo, useEffect, useState } from "react";
import { Button, Drawer, Menu, Popover } from "antd";
import {
CloseOutlined,
MenuOutlined,
SettingOutlined,
} from "@ant-design/icons";
import { ClipboardList, Sparkles, X } from "lucide-react";
import { menuItems } from "@/pages/Layout/menu";
import { NavLink, useLocation, useNavigate } from "react-router";
import TaskUpload from "./TaskUpload";
import SettingsPage from "../SettingsPage/SettingsPage";
import { useAppSelector, useAppDispatch } from "@/store/hooks";
import { showSettings, hideSettings } from "@/store/slices/settingsSlice";
const AsiderAndHeaderLayout = () => {
const { pathname } = useLocation();
const navigate = useNavigate();
const [activeItem, setActiveItem] = useState<string>("");
const [sidebarOpen, setSidebarOpen] = useState(true);
const [taskCenterVisible, setTaskCenterVisible] = useState(false);
const settingVisible = useAppSelector((state) => state.settings.visible);
const dispatch = useAppDispatch();
// Initialize active item based on current pathname
const initActiveItem = () => {
for (let index = 0; index < menuItems.length; index++) {
const element = menuItems[index];
if (element.children) {
element.children.forEach((subItem) => {
if (pathname.includes(subItem.id)) {
setActiveItem(subItem.id);
return;
}
});
} else if (pathname.includes(element.id)) {
setActiveItem(element.id);
return;
}
}
console.log(pathname);
};
useEffect(() => {
initActiveItem();
}, [pathname]);
useEffect(() => {
const handleShowTaskPopover = (event: CustomEvent) => {
const { show } = event.detail;
setTaskCenterVisible(show);
};
window.addEventListener(
"show:task-popover",
handleShowTaskPopover as EventListener
);
return () => {
window.removeEventListener(
"show:task-popover",
handleShowTaskPopover as EventListener
);
};
}, []);
return (
<div
className={`${
sidebarOpen ? "w-64" : "w-20"
} bg-white border-r border-gray-200 transition-all duration-300 flex flex-col relative`}
>
{/* Header */}
<div className="flex items-center justify-between p-4 border-b border-gray-200">
{sidebarOpen && (
<NavLink to="/" className="flex items-center gap-2 cursor-pointer">
<img src="/assets/logo.png" alt="logo" className="h-12 w-40" />
</NavLink>
)}
<span
className="cursor-pointer hover:text-blue-500"
onClick={() => setSidebarOpen(!sidebarOpen)}
>
{sidebarOpen ? <CloseOutlined /> : <MenuOutlined className="ml-4" />}
</span>
</div>
{/* Navigation */}
<div className="flex-1">
<Menu
mode="inline"
inlineCollapsed={!sidebarOpen}
items={menuItems.map((item) => ({
key: item.id,
label: item.title,
icon: item.icon ? <item.icon className="w-4 h-4" /> : null,
children: item.children
? item.children.map((subItem) => ({
key: subItem.id,
label: subItem.title,
icon: subItem.icon ? (
<subItem.icon className="w-4 h-4" />
) : null,
}))
: undefined,
}))}
selectedKeys={[activeItem]}
defaultOpenKeys={["synthesis"]}
onClick={({ key }) => {
setActiveItem(key);
navigate(`/data/${key}`);
}}
/>
</div>
{/* Footer */}
<div className="p-4 border-t border-gray-200">
{sidebarOpen ? (
<div className="space-y-2">
<Popover
forceRender
title={
<div className="flex items-center justify-between gap-2 border-b border-gray-200 pb-2 mb-2">
<h4 className="font-bold"></h4>
<X
onClick={() => setTaskCenterVisible(false)}
className="cursor-pointer w-4 h-4 text-gray-500 hover:text-gray-900"
/>
</div>
}
open={taskCenterVisible}
content={<TaskUpload />}
trigger="click"
destroyOnHidden={false}
>
<Button block onClick={() => setTaskCenterVisible(true)}>
</Button>
</Popover>
<Button
block
onClick={() => {
dispatch(showSettings());
}}
>
</Button>
</div>
) : (
<div className="space-y-2">
<div className="relative">
<Popover
forceRender
title="任务中心"
open={taskCenterVisible}
content={<TaskUpload />}
trigger="click"
destroyOnHidden={false}
>
<Button
block
onClick={() => setTaskCenterVisible(true)}
icon={<ClipboardList className="w-4 h-4" />}
></Button>
</Popover>
</div>
<Button
block
onClick={() => {
dispatch(showSettings());
}}
>
<SettingOutlined />
</Button>
</div>
)}
</div>
<Drawer
title="设置"
placement="bottom"
width="100%"
height="100%"
open={settingVisible}
onClose={() => dispatch(hideSettings())}
bodyStyle={{ padding: 0 }}
destroyOnHidden={true}
>
<SettingsPage></SettingsPage>
</Drawer>
{/* 添加遮罩层,点击外部区域时关闭 */}
{taskCenterVisible && (
<div
className="fixed inset-0 z-40"
onClick={() => {
setTaskCenterVisible(false);
}}
/>
)}
</div>
);
};
export default memo(AsiderAndHeaderLayout);