diff --git a/README.md b/README.md
index 7dbf7eb..b3b029f 100644
--- a/README.md
+++ b/README.md
@@ -1,73 +1,81 @@
-# React + TypeScript + Vite
+# 知习 Admin
-This template provides a minimal setup to get React working in Vite with HMR and some ESLint rules.
+企业知识管理中台 — 后台管理系统前端。
-Currently, two official plugins are available:
+## 技术栈
-- [@vitejs/plugin-react](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react) uses [Oxc](https://oxc.rs)
-- [@vitejs/plugin-react-swc](https://github.com/vitejs/vite-plugin-react/blob/main/packages/plugin-react-swc) uses [SWC](https://swc.rs/)
+- React 19 + TypeScript 6
+- Vite 8
+- Ant Design 5 + @ant-design/pro-components 2.8
+- @tanstack/react-query 5
+- ECharts 5 + echarts-for-react 3
+- dayjs
+- react-router-dom v7
-## React Compiler
+## 项目结构
-The React Compiler is not enabled on this template because of its impact on dev & build performances. To add it, see [this documentation](https://react.dev/learn/react-compiler/installation).
-
-## Expanding the ESLint configuration
-
-If you are developing a production application, we recommend updating the configuration to enable type-aware lint rules:
-
-```js
-export default defineConfig([
- globalIgnores(['dist']),
- {
- files: ['**/*.{ts,tsx}'],
- extends: [
- // Other configs...
-
- // Remove tseslint.configs.recommended and replace with this
- tseslint.configs.recommendedTypeChecked,
- // Alternatively, use this for stricter rules
- tseslint.configs.strictTypeChecked,
- // Optionally, add this for stylistic rules
- tseslint.configs.stylisticTypeChecked,
-
- // Other configs...
- ],
- languageOptions: {
- parserOptions: {
- project: ['./tsconfig.node.json', './tsconfig.app.json'],
- tsconfigRootDir: import.meta.dirname,
- },
- // other options...
- },
- },
-])
+```
+src/
+├── components/ # 共享组件
+│ ├── AuditLogTable # 审计日志 ProTable
+│ ├── ConfirmDangerModal # 高危操作确认弹窗
+│ ├── DetailDrawer # 侧滑详情抽屉
+│ ├── EChartsChartContainer # 图表容器
+│ ├── EmptyState # 空状态占位
+│ ├── MetricCard # 指标卡
+│ ├── StatusTag # 状态标签
+│ ├── AuthGuard # 登录鉴权守卫
+│ ├── PageLoading # 路由懒加载 Spin
+│ └── PermissionGuard # 角色权限守卫
+├── config/
+│ └── menu.tsx # 菜单树与角色过滤
+├── constants/
+│ └── roles.ts # 角色标签/颜色/层级
+├── contexts/
+│ └── AuthContext.tsx # 认证上下文
+├── hooks/
+│ └── use-auth-query.ts # React Query 认证 hooks
+├── layouts/
+│ └── AdminLayout.tsx # 管理后台布局(侧边栏/顶栏/面包屑)
+├── pages/
+│ ├── Dashboard # 数据看板(指标卡 + 图表 + 审计日志)
+│ ├── UserManagement # 管理员 CRUD
+│ ├── Login # 登录页
+│ ├── Placeholder # 占位页
+│ ├── 403/404/500 # 错误页
+├── routes/
+│ └── index.tsx # 路由配置(懒加载 + 角色要求)
+├── services/
+│ ├── admin-api.ts # 类型化 API 函数
+│ ├── http-client.ts # Fetch 封装(401 自动刷新队列)
+│ ├── token-store.ts # Token 本地存储
+│ └── mock-data.ts # DEV 模式 Mock 数据
+└── types/
+ ├── admin.ts # AdminUser, DashboardStats, AuditLog 等
+ └── api.ts # PaginatedResult, ApiError
```
-You can also install [eslint-plugin-react-x](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-x) and [eslint-plugin-react-dom](https://github.com/Rel1cx/eslint-react/tree/main/packages/plugins/eslint-plugin-react-dom) for React-specific lint rules:
+## 启动
-```js
-// eslint.config.js
-import reactX from 'eslint-plugin-react-x'
-import reactDom from 'eslint-plugin-react-dom'
-
-export default defineConfig([
- globalIgnores(['dist']),
- {
- files: ['**/*.{ts,tsx}'],
- extends: [
- // Other configs...
- // Enable lint rules for React
- reactX.configs['recommended-typescript'],
- // Enable lint rules for React DOM
- reactDom.configs.recommended,
- ],
- languageOptions: {
- parserOptions: {
- project: ['./tsconfig.node.json', './tsconfig.app.json'],
- tsconfigRootDir: import.meta.dirname,
- },
- // other options...
- },
- },
-])
+```bash
+npm install
+npm run dev # 开发模式,默认 http://localhost:5173
+npm run build # 生产构建
```
+
+## 认证流程
+
+1. `AuthGuard` 检查登录状态,未登录跳转 `/login`
+2. `PermissionGuard` 检查角色权限,不足跳转 `/403`
+3. HTTP 客户端自动处理 401 → refresh token → 重试请求
+4. refresh token 失败时强制退出到 `/login`
+
+## 角色体系
+
+| 角色 | 说明 | 权限范围 |
+|------|------|---------|
+| SUPER_ADMIN | 超级管理员 | 全部权限,可管理其他管理员 |
+| ADMIN | 管理员 | 用户管理、仪表盘、审计日志 |
+| OPERATIONS | 运营人员 | 仪表盘、审计日志(只读) |
+| DEVELOPER | 开发者 | 仪表盘(只读) |
+| READONLY | 只读用户 | 仪表盘(只读) |
diff --git a/index.html b/index.html
index 02cef4f..2ea0e3b 100644
--- a/index.html
+++ b/index.html
@@ -4,7 +4,7 @@
-
admin-projects-tmp
+ 知习 Admin
diff --git a/package-lock.json b/package-lock.json
index fec39a3..5c4515e 100644
--- a/package-lock.json
+++ b/package-lock.json
@@ -8,11 +8,20 @@
"name": "admin-projects-tmp",
"version": "0.0.0",
"dependencies": {
+ "@ant-design/icons": "^6.2.3",
+ "@ant-design/pro-components": "^2.8.10",
+ "@tanstack/react-query": "^5.100.11",
+ "antd": "^5.29.3",
+ "dayjs": "^1.11.20",
+ "echarts": "^6.1.0",
+ "echarts-for-react": "^3.0.6",
"react": "^19.2.6",
- "react-dom": "^19.2.6"
+ "react-dom": "^19.2.6",
+ "react-router-dom": "^7.15.1"
},
"devDependencies": {
"@eslint/js": "^10.0.1",
+ "@tailwindcss/vite": "^4.3.0",
"@types/node": "^24.12.3",
"@types/react": "^19.2.14",
"@types/react-dom": "^19.2.3",
@@ -21,11 +30,695 @@
"eslint-plugin-react-hooks": "^7.1.1",
"eslint-plugin-react-refresh": "^0.5.2",
"globals": "^17.6.0",
+ "tailwindcss": "^4.3.0",
"typescript": "~6.0.2",
"typescript-eslint": "^8.59.2",
"vite": "^8.0.12"
}
},
+ "node_modules/@ant-design/colors": {
+ "version": "8.0.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-8.0.1.tgz",
+ "integrity": "sha512-foPVl0+SWIslGUtD/xBr1p9U4AKzPhNYEseXYRRo5QSzGACYZrQbe11AYJbYfAWnWSpGBx6JjBmSeugUsD9vqQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^3.0.0"
+ }
+ },
+ "node_modules/@ant-design/cssinjs": {
+ "version": "1.24.0",
+ "resolved": "https://registry.npmmirror.com/@ant-design/cssinjs/-/cssinjs-1.24.0.tgz",
+ "integrity": "sha512-K4cYrJBsgvL+IoozUXYjbT6LHHNt+19a9zkvpBPxLjFHas1UpPM2A5MlhROb0BT8N8WoavM5VsP9MeSeNK/3mg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.11.1",
+ "@emotion/hash": "^0.8.0",
+ "@emotion/unitless": "^0.7.5",
+ "classnames": "^2.3.1",
+ "csstype": "^3.1.3",
+ "rc-util": "^5.35.0",
+ "stylis": "^4.3.4"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/cssinjs-utils": {
+ "version": "1.1.3",
+ "resolved": "https://registry.npmmirror.com/@ant-design/cssinjs-utils/-/cssinjs-utils-1.1.3.tgz",
+ "integrity": "sha512-nOoQMLW1l+xR1Co8NFVYiP8pZp3VjIIzqV6D6ShYF2ljtdwWJn5WSsH+7kvCktXL/yhEtWURKOfH5Xz/gzlwsg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/cssinjs": "^1.21.0",
+ "@babel/runtime": "^7.23.2",
+ "rc-util": "^5.38.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@ant-design/fast-color": {
+ "version": "3.0.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-3.0.1.tgz",
+ "integrity": "sha512-esKJegpW4nckh0o6kV3Tkb7NPIZYbPnnFxmQDUmL08ukXZAvV85TZBr70eGuke/CIArLaP6aw8lt9KILjnWuOw==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@ant-design/icons": {
+ "version": "6.2.3",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-6.2.3.tgz",
+ "integrity": "sha512-Pl3aoAtxQeKryYnt6VvDJtOxMOtA8wrRSACe/pTjOAIG3fdHrWm6Ivb4ku9tsFjYroSXBKirvuxG4QkwBXD9gg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^8.0.1",
+ "@ant-design/icons-svg": "^4.4.2",
+ "@rc-component/util": "^1.10.1",
+ "clsx": "^2.1.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/icons-svg": {
+ "version": "4.4.2",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons-svg/-/icons-svg-4.4.2.tgz",
+ "integrity": "sha512-vHbT+zJEVzllwP+CM+ul7reTEfBR0vgxFe7+lREAsAA7YGsYpboiq2sQNeQeRvh09GfQgs/GyFEvZpJ9cLXpXA==",
+ "license": "MIT"
+ },
+ "node_modules/@ant-design/pro-card": {
+ "version": "2.10.0",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-card/-/pro-card-2.10.0.tgz",
+ "integrity": "sha512-sLONn1odmE0Wkbse8pol4WiaEzBV8JU5s3FAMflPpycfUcbSaa1ktXzQ7LCo2SAvOS7gkfmpFjBPtrfbigKh4g==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/cssinjs": "^1.21.1",
+ "@ant-design/icons": "^5.0.0",
+ "@ant-design/pro-provider": "2.16.2",
+ "@ant-design/pro-utils": "2.18.0",
+ "@babel/runtime": "^7.18.0",
+ "classnames": "^2.3.2",
+ "rc-resize-observer": "^1.0.0",
+ "rc-util": "^5.4.0"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-card/node_modules/@ant-design/colors": {
+ "version": "7.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-7.2.1.tgz",
+ "integrity": "sha512-lCHDcEzieu4GA3n8ELeZ5VQ8pKQAWcGGLRTQ50aQM2iqPpq2evTxER84jfdPvsPAtEcZ7m44NI45edFMo8oOYQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6"
+ }
+ },
+ "node_modules/@ant-design/pro-card/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@ant-design/pro-card/node_modules/@ant-design/icons": {
+ "version": "5.6.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-5.6.1.tgz",
+ "integrity": "sha512-0/xS39c91WjPAZOWsvi1//zjx6kAp4kxWwctR6kuU6p133w8RU0D2dSCvZC19uQyharg/sAvYxGYWl01BbZZfg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^7.0.0",
+ "@ant-design/icons-svg": "^4.4.0",
+ "@babel/runtime": "^7.24.8",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.31.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-components": {
+ "version": "2.8.10",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-components/-/pro-components-2.8.10.tgz",
+ "integrity": "sha512-QHnnIXdmC5GTAtm6i8eeJy5yT9npPlFyxpDm+duiDrTRKRFaAQBduArxlH3DA/hoRCCypzPONxfK9BQNIhIyZA==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/pro-card": "2.10.0",
+ "@ant-design/pro-descriptions": "2.6.10",
+ "@ant-design/pro-field": "3.1.0",
+ "@ant-design/pro-form": "2.32.0",
+ "@ant-design/pro-layout": "7.22.7",
+ "@ant-design/pro-list": "2.6.10",
+ "@ant-design/pro-provider": "2.16.2",
+ "@ant-design/pro-skeleton": "2.2.1",
+ "@ant-design/pro-table": "3.21.0",
+ "@ant-design/pro-utils": "2.18.0",
+ "@babel/runtime": "^7.16.3"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0",
+ "react-dom": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-descriptions": {
+ "version": "2.6.10",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-descriptions/-/pro-descriptions-2.6.10.tgz",
+ "integrity": "sha512-+4MbiOfumnWlW0Awm4m8JML5o3lR649FD24AaivCmr8BQvIAAXdTITnDMXEg8BqvdP4KOvNsStZrvYfqoev33A==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/pro-field": "3.1.0",
+ "@ant-design/pro-form": "2.32.0",
+ "@ant-design/pro-provider": "2.16.2",
+ "@ant-design/pro-skeleton": "2.2.1",
+ "@ant-design/pro-utils": "2.18.0",
+ "@babel/runtime": "^7.18.0",
+ "rc-resize-observer": "^0.2.3",
+ "rc-util": "^5.0.6"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-descriptions/node_modules/rc-resize-observer": {
+ "version": "0.2.6",
+ "resolved": "https://registry.npmmirror.com/rc-resize-observer/-/rc-resize-observer-0.2.6.tgz",
+ "integrity": "sha512-YX6nYnd6fk7zbuvT6oSDMKiZjyngjHoy+fz+vL3Tez38d/G5iGdaDJa2yE7345G6sc4Mm1IGRUIwclvltddhmA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "^2.2.1",
+ "rc-util": "^5.0.0",
+ "resize-observer-polyfill": "^1.5.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@ant-design/pro-field": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-field/-/pro-field-3.1.0.tgz",
+ "integrity": "sha512-+Dgp31WjD+iwg9KIRAMgNkfQivkJKMcYBrIBmho1e8ep/O0HgWSp48g70tBIWi/Lfem/Ky2schF7O8XCFouczw==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/icons": "^5.0.0",
+ "@ant-design/pro-provider": "2.16.2",
+ "@ant-design/pro-utils": "2.18.0",
+ "@babel/runtime": "^7.18.0",
+ "@chenshuai2144/sketch-color": "^1.0.8",
+ "classnames": "^2.3.2",
+ "dayjs": "^1.11.10",
+ "lodash": "^4.17.21",
+ "lodash-es": "^4.17.21",
+ "rc-util": "^5.4.0",
+ "swr": "^2.0.0"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-field/node_modules/@ant-design/colors": {
+ "version": "7.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-7.2.1.tgz",
+ "integrity": "sha512-lCHDcEzieu4GA3n8ELeZ5VQ8pKQAWcGGLRTQ50aQM2iqPpq2evTxER84jfdPvsPAtEcZ7m44NI45edFMo8oOYQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6"
+ }
+ },
+ "node_modules/@ant-design/pro-field/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@ant-design/pro-field/node_modules/@ant-design/icons": {
+ "version": "5.6.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-5.6.1.tgz",
+ "integrity": "sha512-0/xS39c91WjPAZOWsvi1//zjx6kAp4kxWwctR6kuU6p133w8RU0D2dSCvZC19uQyharg/sAvYxGYWl01BbZZfg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^7.0.0",
+ "@ant-design/icons-svg": "^4.4.0",
+ "@babel/runtime": "^7.24.8",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.31.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-form": {
+ "version": "2.32.0",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-form/-/pro-form-2.32.0.tgz",
+ "integrity": "sha512-GZnVAMeYv+YHJb17lJ7rX5PYuQPvEA6EotQnPbHi9tGLN3PfexcAd21rqzuO+OrulU2x7TEMDIxtY9MzvvOGbg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/icons": "^5.0.0",
+ "@ant-design/pro-field": "3.1.0",
+ "@ant-design/pro-provider": "2.16.2",
+ "@ant-design/pro-utils": "2.18.0",
+ "@babel/runtime": "^7.18.0",
+ "@chenshuai2144/sketch-color": "^1.0.7",
+ "@umijs/use-params": "^1.0.9",
+ "classnames": "^2.3.2",
+ "dayjs": "^1.11.10",
+ "lodash": "^4.17.21",
+ "lodash-es": "^4.17.21",
+ "rc-resize-observer": "^1.1.0",
+ "rc-util": "^5.0.6"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "rc-field-form": ">=1.22.0",
+ "react": ">=17.0.0",
+ "react-dom": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-form/node_modules/@ant-design/colors": {
+ "version": "7.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-7.2.1.tgz",
+ "integrity": "sha512-lCHDcEzieu4GA3n8ELeZ5VQ8pKQAWcGGLRTQ50aQM2iqPpq2evTxER84jfdPvsPAtEcZ7m44NI45edFMo8oOYQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6"
+ }
+ },
+ "node_modules/@ant-design/pro-form/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@ant-design/pro-form/node_modules/@ant-design/icons": {
+ "version": "5.6.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-5.6.1.tgz",
+ "integrity": "sha512-0/xS39c91WjPAZOWsvi1//zjx6kAp4kxWwctR6kuU6p133w8RU0D2dSCvZC19uQyharg/sAvYxGYWl01BbZZfg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^7.0.0",
+ "@ant-design/icons-svg": "^4.4.0",
+ "@babel/runtime": "^7.24.8",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.31.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-layout": {
+ "version": "7.22.7",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-layout/-/pro-layout-7.22.7.tgz",
+ "integrity": "sha512-fvmtNA1r9SaasVIQIQt611VSlNxtVxDbQ3e+1GhYQza3tVJi/3gCZuDyfMfTnbLmf3PaW/YvLkn7MqDbzAzoLA==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/cssinjs": "^1.21.1",
+ "@ant-design/icons": "^5.0.0",
+ "@ant-design/pro-provider": "2.16.2",
+ "@ant-design/pro-utils": "2.18.0",
+ "@babel/runtime": "^7.18.0",
+ "@umijs/route-utils": "^4.0.0",
+ "@umijs/use-params": "^1.0.9",
+ "classnames": "^2.3.2",
+ "lodash": "^4.17.21",
+ "lodash-es": "^4.17.21",
+ "path-to-regexp": "8.2.0",
+ "rc-resize-observer": "^1.1.0",
+ "rc-util": "^5.0.6",
+ "swr": "^2.0.0",
+ "warning": "^4.0.3"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0",
+ "react-dom": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-layout/node_modules/@ant-design/colors": {
+ "version": "7.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-7.2.1.tgz",
+ "integrity": "sha512-lCHDcEzieu4GA3n8ELeZ5VQ8pKQAWcGGLRTQ50aQM2iqPpq2evTxER84jfdPvsPAtEcZ7m44NI45edFMo8oOYQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6"
+ }
+ },
+ "node_modules/@ant-design/pro-layout/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@ant-design/pro-layout/node_modules/@ant-design/icons": {
+ "version": "5.6.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-5.6.1.tgz",
+ "integrity": "sha512-0/xS39c91WjPAZOWsvi1//zjx6kAp4kxWwctR6kuU6p133w8RU0D2dSCvZC19uQyharg/sAvYxGYWl01BbZZfg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^7.0.0",
+ "@ant-design/icons-svg": "^4.4.0",
+ "@babel/runtime": "^7.24.8",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.31.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-list": {
+ "version": "2.6.10",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-list/-/pro-list-2.6.10.tgz",
+ "integrity": "sha512-xSWwnqCr+hPEYR4qY7nFUaxO5RQBxNlFaPNmobP2i+Im31slk9JuAusgWeIYO0mNhLJuLbxd8CCma2AZij3fBQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/cssinjs": "^1.21.1",
+ "@ant-design/icons": "^5.0.0",
+ "@ant-design/pro-card": "2.10.0",
+ "@ant-design/pro-field": "3.1.0",
+ "@ant-design/pro-table": "3.21.0",
+ "@ant-design/pro-utils": "2.18.0",
+ "@babel/runtime": "^7.18.0",
+ "classnames": "^2.3.2",
+ "dayjs": "^1.11.10",
+ "rc-resize-observer": "^1.0.0",
+ "rc-util": "^4.19.0"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0",
+ "react-dom": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-list/node_modules/@ant-design/colors": {
+ "version": "7.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-7.2.1.tgz",
+ "integrity": "sha512-lCHDcEzieu4GA3n8ELeZ5VQ8pKQAWcGGLRTQ50aQM2iqPpq2evTxER84jfdPvsPAtEcZ7m44NI45edFMo8oOYQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6"
+ }
+ },
+ "node_modules/@ant-design/pro-list/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@ant-design/pro-list/node_modules/@ant-design/icons": {
+ "version": "5.6.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-5.6.1.tgz",
+ "integrity": "sha512-0/xS39c91WjPAZOWsvi1//zjx6kAp4kxWwctR6kuU6p133w8RU0D2dSCvZC19uQyharg/sAvYxGYWl01BbZZfg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^7.0.0",
+ "@ant-design/icons-svg": "^4.4.0",
+ "@babel/runtime": "^7.24.8",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.31.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-list/node_modules/@ant-design/icons/node_modules/rc-util": {
+ "version": "5.44.4",
+ "resolved": "https://registry.npmmirror.com/rc-util/-/rc-util-5.44.4.tgz",
+ "integrity": "sha512-resueRJzmHG9Q6rI/DfK6Kdv9/Lfls05vzMs1Sk3M2P+3cJa+MakaZyWY8IPfehVuhPJFKrIY1IK4GqbiaiY5w==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.3",
+ "react-is": "^18.2.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@ant-design/pro-list/node_modules/rc-util": {
+ "version": "4.21.1",
+ "resolved": "https://registry.npmmirror.com/rc-util/-/rc-util-4.21.1.tgz",
+ "integrity": "sha512-Z+vlkSQVc1l8O2UjR3WQ+XdWlhj5q9BMQNLk2iOBch75CqPfrJyGtcWMcnhRlNuDu0Ndtt4kLVO8JI8BrABobg==",
+ "license": "MIT",
+ "dependencies": {
+ "add-dom-event-listener": "^1.1.0",
+ "prop-types": "^15.5.10",
+ "react-is": "^16.12.0",
+ "react-lifecycles-compat": "^3.0.4",
+ "shallowequal": "^1.1.0"
+ }
+ },
+ "node_modules/@ant-design/pro-list/node_modules/rc-util/node_modules/react-is": {
+ "version": "16.13.1",
+ "resolved": "https://registry.npmmirror.com/react-is/-/react-is-16.13.1.tgz",
+ "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==",
+ "license": "MIT"
+ },
+ "node_modules/@ant-design/pro-provider": {
+ "version": "2.16.2",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-provider/-/pro-provider-2.16.2.tgz",
+ "integrity": "sha512-0KmCH1EaOND787Jz6VRMYtLNZmqfT0JPjdUfxhyOxFfnBRfrjyfZgIa6CQoAJLEUMWv57PccWS8wRHVUUk2Yiw==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/cssinjs": "^1.21.1",
+ "@babel/runtime": "^7.18.0",
+ "@ctrl/tinycolor": "^3.4.0",
+ "dayjs": "^1.11.10",
+ "rc-util": "^5.0.1",
+ "swr": "^2.0.0"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0",
+ "react-dom": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-skeleton": {
+ "version": "2.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-skeleton/-/pro-skeleton-2.2.1.tgz",
+ "integrity": "sha512-3M2jNOZQZWEDR8pheY00OkHREfb0rquvFZLCa6DypGmiksiuuYuR9Y4iA82ZF+mva2FmpHekdwbje/GpbxqBeg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.0"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0",
+ "react-dom": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-table": {
+ "version": "3.21.0",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-table/-/pro-table-3.21.0.tgz",
+ "integrity": "sha512-sI81d3FYRv5sXamUc+M5CsHZ9CchuUQgOAPzo5H4oPAVL5h+mkYGRsBzPsxQX7khTNpWjrAtPoRm5ipx3vvWog==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/cssinjs": "^1.21.1",
+ "@ant-design/icons": "^5.0.0",
+ "@ant-design/pro-card": "2.10.0",
+ "@ant-design/pro-field": "3.1.0",
+ "@ant-design/pro-form": "2.32.0",
+ "@ant-design/pro-provider": "2.16.2",
+ "@ant-design/pro-utils": "2.18.0",
+ "@babel/runtime": "^7.18.0",
+ "@dnd-kit/core": "^6.0.8",
+ "@dnd-kit/modifiers": "^6.0.1",
+ "@dnd-kit/sortable": "^7.0.2",
+ "@dnd-kit/utilities": "^3.2.1",
+ "classnames": "^2.3.2",
+ "dayjs": "^1.11.10",
+ "lodash": "^4.17.21",
+ "lodash-es": "^4.17.21",
+ "rc-resize-observer": "^1.0.0",
+ "rc-util": "^5.0.1"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "rc-field-form": ">=1.22.0",
+ "react": ">=17.0.0",
+ "react-dom": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-table/node_modules/@ant-design/colors": {
+ "version": "7.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-7.2.1.tgz",
+ "integrity": "sha512-lCHDcEzieu4GA3n8ELeZ5VQ8pKQAWcGGLRTQ50aQM2iqPpq2evTxER84jfdPvsPAtEcZ7m44NI45edFMo8oOYQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6"
+ }
+ },
+ "node_modules/@ant-design/pro-table/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@ant-design/pro-table/node_modules/@ant-design/icons": {
+ "version": "5.6.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-5.6.1.tgz",
+ "integrity": "sha512-0/xS39c91WjPAZOWsvi1//zjx6kAp4kxWwctR6kuU6p133w8RU0D2dSCvZC19uQyharg/sAvYxGYWl01BbZZfg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^7.0.0",
+ "@ant-design/icons-svg": "^4.4.0",
+ "@babel/runtime": "^7.24.8",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.31.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-utils": {
+ "version": "2.18.0",
+ "resolved": "https://registry.npmmirror.com/@ant-design/pro-utils/-/pro-utils-2.18.0.tgz",
+ "integrity": "sha512-8+ikyrN8L8a8Ph4oeHTOJEiranTj18+9+WHCHjKNdEfukI7Rjn8xpYdLJWb2AUJkb9d4eoAqjd5+k+7w81Df0w==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/icons": "^5.0.0",
+ "@ant-design/pro-provider": "2.16.2",
+ "@babel/runtime": "^7.18.0",
+ "classnames": "^2.3.2",
+ "dayjs": "^1.11.10",
+ "lodash": "^4.17.21",
+ "lodash-es": "^4.17.21",
+ "rc-util": "^5.0.6",
+ "safe-stable-stringify": "^2.4.3",
+ "swr": "^2.0.0"
+ },
+ "peerDependencies": {
+ "antd": "^4.24.15 || ^5.11.2",
+ "react": ">=17.0.0",
+ "react-dom": ">=17.0.0"
+ }
+ },
+ "node_modules/@ant-design/pro-utils/node_modules/@ant-design/colors": {
+ "version": "7.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-7.2.1.tgz",
+ "integrity": "sha512-lCHDcEzieu4GA3n8ELeZ5VQ8pKQAWcGGLRTQ50aQM2iqPpq2evTxER84jfdPvsPAtEcZ7m44NI45edFMo8oOYQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6"
+ }
+ },
+ "node_modules/@ant-design/pro-utils/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@ant-design/pro-utils/node_modules/@ant-design/icons": {
+ "version": "5.6.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-5.6.1.tgz",
+ "integrity": "sha512-0/xS39c91WjPAZOWsvi1//zjx6kAp4kxWwctR6kuU6p133w8RU0D2dSCvZC19uQyharg/sAvYxGYWl01BbZZfg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^7.0.0",
+ "@ant-design/icons-svg": "^4.4.0",
+ "@babel/runtime": "^7.24.8",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.31.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/@ant-design/react-slick": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmmirror.com/@ant-design/react-slick/-/react-slick-1.1.2.tgz",
+ "integrity": "sha512-EzlvzE6xQUBrZuuhSAFTdsr4P2bBBHGZwKFemEfq8gIGyIQCxalYfZW/T2ORbtQx5rU69o+WycP3exY/7T1hGA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.4",
+ "classnames": "^2.2.5",
+ "json2mq": "^0.2.0",
+ "resize-observer-polyfill": "^1.5.1",
+ "throttle-debounce": "^5.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0"
+ }
+ },
"node_modules/@babel/code-frame": {
"version": "7.29.0",
"resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.29.0.tgz",
@@ -219,6 +912,15 @@
"node": ">=6.0.0"
}
},
+ "node_modules/@babel/runtime": {
+ "version": "7.29.2",
+ "resolved": "https://registry.npmmirror.com/@babel/runtime/-/runtime-7.29.2.tgz",
+ "integrity": "sha512-JiDShH45zKHWyGe4ZNVRrCjBz8Nh9TMmZG1kh4QTK8hCBTWBi8Da+i7s1fJw7/lYpM4ccepSNfqzZ/QvABBi5g==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=6.9.0"
+ }
+ },
"node_modules/@babel/template": {
"version": "7.28.6",
"resolved": "https://registry.npmjs.org/@babel/template/-/template-7.28.6.tgz",
@@ -267,29 +969,94 @@
"node": ">=6.9.0"
}
},
- "node_modules/@emnapi/core": {
- "version": "1.10.0",
- "resolved": "https://registry.npmjs.org/@emnapi/core/-/core-1.10.0.tgz",
- "integrity": "sha512-yq6OkJ4p82CAfPl0u9mQebQHKPJkY7WrIuk205cTYnYe+k2Z8YBh11FrbRG/H6ihirqcacOgl2BIO8oyMQLeXw==",
- "dev": true,
+ "node_modules/@chenshuai2144/sketch-color": {
+ "version": "1.0.9",
+ "resolved": "https://registry.npmmirror.com/@chenshuai2144/sketch-color/-/sketch-color-1.0.9.tgz",
+ "integrity": "sha512-obzSy26cb7Pm7OprWyVpgMpIlrZpZ0B7vbrU0RMbvRg0YAI890S5Xy02Aj1Nhl4+KTbi1lVYHt6HQP8Hm9s+1w==",
"license": "MIT",
- "optional": true,
- "peer": true,
"dependencies": {
- "@emnapi/wasi-threads": "1.2.1",
- "tslib": "^2.4.0"
+ "reactcss": "^1.2.3",
+ "tinycolor2": "^1.4.2"
+ },
+ "peerDependencies": {
+ "react": ">=16.12.0"
}
},
- "node_modules/@emnapi/runtime": {
- "version": "1.10.0",
- "resolved": "https://registry.npmjs.org/@emnapi/runtime/-/runtime-1.10.0.tgz",
- "integrity": "sha512-ewvYlk86xUoGI0zQRNq/mC+16R1QeDlKQy21Ki3oSYXNgLb45GV1P6A0M+/s6nyCuNDqe5VpaY84BzXGwVbwFA==",
- "dev": true,
+ "node_modules/@ctrl/tinycolor": {
+ "version": "3.6.1",
+ "resolved": "https://registry.npmmirror.com/@ctrl/tinycolor/-/tinycolor-3.6.1.tgz",
+ "integrity": "sha512-SITSV6aIXsuVNV3f3O0f2n/cgyEDWoSqtZMYiAmcsYHydcKrOz3gUxB/iXd/Qf08+IZX4KpgNbvUdMBmWz+kcA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=10"
+ }
+ },
+ "node_modules/@dnd-kit/accessibility": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmmirror.com/@dnd-kit/accessibility/-/accessibility-3.1.1.tgz",
+ "integrity": "sha512-2P+YgaXF+gRsIihwwY1gCsQSYnu9Zyj2py8kY5fFvUM1qm2WA2u639R6YNVfU4GWr+ZM5mqEsfHZZLoRONbemw==",
+ "license": "MIT",
+ "dependencies": {
+ "tslib": "^2.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.8.0"
+ }
+ },
+ "node_modules/@dnd-kit/core": {
+ "version": "6.3.1",
+ "resolved": "https://registry.npmmirror.com/@dnd-kit/core/-/core-6.3.1.tgz",
+ "integrity": "sha512-xkGBRQQab4RLwgXxoqETICr6S5JlogafbhNsidmrkVv2YRs5MLwpjoF2qpiGjQt8S9AoxtIV603s0GIUpY5eYQ==",
"license": "MIT",
- "optional": true,
"peer": true,
"dependencies": {
- "tslib": "^2.4.0"
+ "@dnd-kit/accessibility": "^3.1.1",
+ "@dnd-kit/utilities": "^3.2.2",
+ "tslib": "^2.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.8.0",
+ "react-dom": ">=16.8.0"
+ }
+ },
+ "node_modules/@dnd-kit/modifiers": {
+ "version": "6.0.1",
+ "resolved": "https://registry.npmmirror.com/@dnd-kit/modifiers/-/modifiers-6.0.1.tgz",
+ "integrity": "sha512-rbxcsg3HhzlcMHVHWDuh9LCjpOVAgqbV78wLGI8tziXY3+qcMQ61qVXIvNKQFuhj75dSfD+o+PYZQ/NUk2A23A==",
+ "license": "MIT",
+ "dependencies": {
+ "@dnd-kit/utilities": "^3.2.1",
+ "tslib": "^2.0.0"
+ },
+ "peerDependencies": {
+ "@dnd-kit/core": "^6.0.6",
+ "react": ">=16.8.0"
+ }
+ },
+ "node_modules/@dnd-kit/sortable": {
+ "version": "7.0.2",
+ "resolved": "https://registry.npmmirror.com/@dnd-kit/sortable/-/sortable-7.0.2.tgz",
+ "integrity": "sha512-wDkBHHf9iCi1veM834Gbk1429bd4lHX4RpAwT0y2cHLf246GAvU2sVw/oxWNpPKQNQRQaeGXhAVgrOl1IT+iyA==",
+ "license": "MIT",
+ "dependencies": {
+ "@dnd-kit/utilities": "^3.2.0",
+ "tslib": "^2.0.0"
+ },
+ "peerDependencies": {
+ "@dnd-kit/core": "^6.0.7",
+ "react": ">=16.8.0"
+ }
+ },
+ "node_modules/@dnd-kit/utilities": {
+ "version": "3.2.2",
+ "resolved": "https://registry.npmmirror.com/@dnd-kit/utilities/-/utilities-3.2.2.tgz",
+ "integrity": "sha512-+MKAJEOfaBe5SmV6t34p80MMKhjvUz0vRrvVJbPT0WElzaOJ/1xs+D+KDv+tD/NE5ujfrChEcshd4fLn0wpiqg==",
+ "license": "MIT",
+ "dependencies": {
+ "tslib": "^2.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.8.0"
}
},
"node_modules/@emnapi/wasi-threads": {
@@ -303,6 +1070,18 @@
"tslib": "^2.4.0"
}
},
+ "node_modules/@emotion/hash": {
+ "version": "0.8.0",
+ "resolved": "https://registry.npmmirror.com/@emotion/hash/-/hash-0.8.0.tgz",
+ "integrity": "sha512-kBJtf7PH6aWwZ6fka3zQ0p6SBYzx4fl1LoZXE2RrnYST9Xljm7WfKJrU4g/Xr3Beg72MLrp1AWNUmuYJTL7Cow==",
+ "license": "MIT"
+ },
+ "node_modules/@emotion/unitless": {
+ "version": "0.7.5",
+ "resolved": "https://registry.npmmirror.com/@emotion/unitless/-/unitless-0.7.5.tgz",
+ "integrity": "sha512-OWORNpfjMsSSUBVrRBVGECkhWcULOAJz9ZW8uK9qgxD+87M7jHRcvh/A96XXNhXTLmKcoYSQtBEX7lHMO7YRwg==",
+ "license": "MIT"
+ },
"node_modules/@eslint-community/eslint-utils": {
"version": "4.9.1",
"resolved": "https://registry.npmjs.org/@eslint-community/eslint-utils/-/eslint-utils-4.9.1.tgz",
@@ -576,6 +1355,179 @@
"url": "https://github.com/sponsors/Boshen"
}
},
+ "node_modules/@rc-component/async-validator": {
+ "version": "5.1.0",
+ "resolved": "https://registry.npmmirror.com/@rc-component/async-validator/-/async-validator-5.1.0.tgz",
+ "integrity": "sha512-n4HcR5siNUXRX23nDizbZBQPO0ZM/5oTtmKZ6/eqL0L2bo747cklFdZGRN2f+c9qWGICwDzrhW0H7tE9PptdcA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.4"
+ },
+ "engines": {
+ "node": ">=14.x"
+ }
+ },
+ "node_modules/@rc-component/color-picker": {
+ "version": "2.0.1",
+ "resolved": "https://registry.npmmirror.com/@rc-component/color-picker/-/color-picker-2.0.1.tgz",
+ "integrity": "sha512-WcZYwAThV/b2GISQ8F+7650r5ZZJ043E57aVBFkQ+kSY4C6wdofXgB0hBx+GPGpIU0Z81eETNoDUJMr7oy/P8Q==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6",
+ "@babel/runtime": "^7.23.6",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.38.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@rc-component/color-picker/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@rc-component/context": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmmirror.com/@rc-component/context/-/context-1.4.0.tgz",
+ "integrity": "sha512-kFcNxg9oLRMoL3qki0OMxK+7g5mypjgaaJp/pkOis/6rVxma9nJBF/8kCIuTYHUQNr0ii7MxqE33wirPZLJQ2w==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "rc-util": "^5.27.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@rc-component/mini-decimal": {
+ "version": "1.1.3",
+ "resolved": "https://registry.npmmirror.com/@rc-component/mini-decimal/-/mini-decimal-1.1.3.tgz",
+ "integrity": "sha512-bk/FJ09fLf+NLODMAFll6CfYrHPBioTedhW6lxDBuuWucJEqFUd4l/D/5JgIi3dina6sYahB8iuPAZTNz2pMxw==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.0"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/@rc-component/mutate-observer": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmmirror.com/@rc-component/mutate-observer/-/mutate-observer-1.1.0.tgz",
+ "integrity": "sha512-QjrOsDXQusNwGZPf4/qRQasg7UFEj06XiCJ8iuiq/Io7CrHrgVi6Uuetw60WAMG1799v+aM8kyc+1L/GBbHSlw==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.0",
+ "classnames": "^2.3.2",
+ "rc-util": "^5.24.4"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@rc-component/portal": {
+ "version": "1.1.2",
+ "resolved": "https://registry.npmmirror.com/@rc-component/portal/-/portal-1.1.2.tgz",
+ "integrity": "sha512-6f813C0IsasTZms08kfA8kPAGxbbkYToa8ALaiDIGGECU4i9hj8Plgbx0sNJDrey3EtHO30hmdaxtT0138xZcg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.0",
+ "classnames": "^2.3.2",
+ "rc-util": "^5.24.4"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@rc-component/qrcode": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmmirror.com/@rc-component/qrcode/-/qrcode-1.1.1.tgz",
+ "integrity": "sha512-LfLGNymzKdUPjXUbRP+xOhIWY4jQ+YMj5MmWAcgcAq1Ij8XP7tRmAXqyuv96XvLUBE/5cA8hLFl9eO1JQMujrA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@rc-component/tour": {
+ "version": "1.15.1",
+ "resolved": "https://registry.npmmirror.com/@rc-component/tour/-/tour-1.15.1.tgz",
+ "integrity": "sha512-Tr2t7J1DKZUpfJuDZWHxyxWpfmj8EZrqSgyMZ+BCdvKZ6r1UDsfU46M/iWAAFBy961Ssfom2kv5f3UcjIL2CmQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.0",
+ "@rc-component/portal": "^1.0.0-9",
+ "@rc-component/trigger": "^2.0.0",
+ "classnames": "^2.3.2",
+ "rc-util": "^5.24.4"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@rc-component/trigger": {
+ "version": "2.3.1",
+ "resolved": "https://registry.npmmirror.com/@rc-component/trigger/-/trigger-2.3.1.tgz",
+ "integrity": "sha512-ORENF39PeXTzM+gQEshuk460Z8N4+6DkjpxlpE7Q3gYy1iBpLrx0FOJz3h62ryrJZ/3zCAUIkT1Pb/8hHWpb3A==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.23.2",
+ "@rc-component/portal": "^1.1.0",
+ "classnames": "^2.3.2",
+ "rc-motion": "^2.0.0",
+ "rc-resize-observer": "^1.3.1",
+ "rc-util": "^5.44.0"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/@rc-component/util": {
+ "version": "1.11.1",
+ "resolved": "https://registry.npmmirror.com/@rc-component/util/-/util-1.11.1.tgz",
+ "integrity": "sha512-awVlI3ub2vqfqkYxOBc/uQ0efm3jw0wcrhtO/YWLyZfxiKXczKwNbVuhlnyxytDt7H9pbbVQiqr+O6MLATtRYg==",
+ "license": "MIT",
+ "dependencies": {
+ "is-mobile": "^5.0.0",
+ "react-is": "^18.2.0"
+ },
+ "peerDependencies": {
+ "react": ">=18.0.0",
+ "react-dom": ">=18.0.0"
+ }
+ },
"node_modules/@rolldown/binding-android-arm64": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/@rolldown/binding-android-arm64/-/binding-android-arm64-1.0.1.tgz",
@@ -840,6 +1792,304 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/@tailwindcss/node": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/node/-/node-4.3.0.tgz",
+ "integrity": "sha512-aFb4gUhFOgdh9AXo4IzBEOzBkkAxm9VigwDJnMIYv3lcfXCJVesNfbEaBl4BNgVRyid92AmdviqwBUBRKSeY3g==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@jridgewell/remapping": "^2.3.5",
+ "enhanced-resolve": "^5.21.0",
+ "jiti": "^2.6.1",
+ "lightningcss": "1.32.0",
+ "magic-string": "^0.30.21",
+ "source-map-js": "^1.2.1",
+ "tailwindcss": "4.3.0"
+ }
+ },
+ "node_modules/@tailwindcss/oxide": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide/-/oxide-4.3.0.tgz",
+ "integrity": "sha512-F7HZGBeN9I0/AuuJS5PwcD8xayx5ri5GhjYUDBEVYUkexyA/giwbDNjRVrxSezE3T250OU2K/wp/ltWx3UOefg==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">= 20"
+ },
+ "optionalDependencies": {
+ "@tailwindcss/oxide-android-arm64": "4.3.0",
+ "@tailwindcss/oxide-darwin-arm64": "4.3.0",
+ "@tailwindcss/oxide-darwin-x64": "4.3.0",
+ "@tailwindcss/oxide-freebsd-x64": "4.3.0",
+ "@tailwindcss/oxide-linux-arm-gnueabihf": "4.3.0",
+ "@tailwindcss/oxide-linux-arm64-gnu": "4.3.0",
+ "@tailwindcss/oxide-linux-arm64-musl": "4.3.0",
+ "@tailwindcss/oxide-linux-x64-gnu": "4.3.0",
+ "@tailwindcss/oxide-linux-x64-musl": "4.3.0",
+ "@tailwindcss/oxide-wasm32-wasi": "4.3.0",
+ "@tailwindcss/oxide-win32-arm64-msvc": "4.3.0",
+ "@tailwindcss/oxide-win32-x64-msvc": "4.3.0"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-android-arm64": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-android-arm64/-/oxide-android-arm64-4.3.0.tgz",
+ "integrity": "sha512-TJPiq67tKlLuObP6RkwvVGDoxCMBVtDgKkLfa/uyj7/FyxvQwHS+UOnVrXXgbEsfUaMgiVvC4KbJnRr26ho4Ng==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "android"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-darwin-arm64": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-darwin-arm64/-/oxide-darwin-arm64-4.3.0.tgz",
+ "integrity": "sha512-oMN/WZRb+SO37BmUElEgeEWuU8E/HXRkiODxJxLe1UTHVXLrdVSgfaJV7pSlhRGMSOiXLuxTIjfsF3wYvz8cgQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-darwin-x64": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-darwin-x64/-/oxide-darwin-x64-4.3.0.tgz",
+ "integrity": "sha512-N6CUmu4a6bKVADfw77p+iw6Yd9Q3OBhe0veaDX+QazfuVYlQsHfDgxBrsjQ/IW+zywL8mTrNd0SdJT/zgtvMdA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "darwin"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-freebsd-x64": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-freebsd-x64/-/oxide-freebsd-x64-4.3.0.tgz",
+ "integrity": "sha512-zDL5hBkQdH5C6MpqbK3gQAgP80tsMwSI26vjOzjJtNCMUo0lFgOItzHKBIupOZNQxt3ouPH7RPhvNhiTfCe5CQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "freebsd"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-linux-arm-gnueabihf": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-linux-arm-gnueabihf/-/oxide-linux-arm-gnueabihf-4.3.0.tgz",
+ "integrity": "sha512-R06HdNi7A7OEoMsf6d4tjZ71RCWnZQPHj2mnotSFURjNLdBC+cIgXQ7l81CqeoiQftjf6OOblxXMInMgN2VzMA==",
+ "cpu": [
+ "arm"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-linux-arm64-gnu": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-linux-arm64-gnu/-/oxide-linux-arm64-gnu-4.3.0.tgz",
+ "integrity": "sha512-qTJHELX8jetjhRQHCLilkVLmybpzNQAtaI/gaoVoidn/ufbNDbAo8KlK2J+yPoc8wQxvDxCmh/5lr8nC1+lTbg==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-linux-arm64-musl": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-linux-arm64-musl/-/oxide-linux-arm64-musl-4.3.0.tgz",
+ "integrity": "sha512-Z6sukiQsngnWO+l39X4pPbiWT81IC+PLKF+PHxIlyZbGNb9MODfYlXEVlFvej5BOZInWX01kVyzeLvHsXhfczQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-linux-x64-gnu": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-linux-x64-gnu/-/oxide-linux-x64-gnu-4.3.0.tgz",
+ "integrity": "sha512-DRNdQRpSGzRGfARVuVkxvM8Q12nh19l4BF/G7zGA1oe+9wcC6saFBHTISrpIcKzhiXtSrlSrluCfvMuledoCTQ==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-linux-x64-musl": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-linux-x64-musl/-/oxide-linux-x64-musl-4.3.0.tgz",
+ "integrity": "sha512-Z0IADbDo8bh6I7h2IQMx601AdXBLfFpEdUotft86evd/8ZPflZe9COPO8Q1vw+pfLWIUo9zN/JGZvwuAJqduqg==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "linux"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-wasm32-wasi": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-wasm32-wasi/-/oxide-wasm32-wasi-4.3.0.tgz",
+ "integrity": "sha512-HNZGOUxEmElksYR7S6sC5jTeNGpobAsy9u7Gu0AskJ8/20FR9GqebUyB+HBcU/ax6BHuiuJi+Oda4B+YX6H1yA==",
+ "bundleDependencies": [
+ "@napi-rs/wasm-runtime",
+ "@emnapi/core",
+ "@emnapi/runtime",
+ "@tybys/wasm-util",
+ "@emnapi/wasi-threads",
+ "tslib"
+ ],
+ "cpu": [
+ "wasm32"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "dependencies": {
+ "@emnapi/core": "^1.10.0",
+ "@emnapi/runtime": "^1.10.0",
+ "@emnapi/wasi-threads": "^1.2.1",
+ "@napi-rs/wasm-runtime": "^1.1.4",
+ "@tybys/wasm-util": "^0.10.1",
+ "tslib": "^2.8.1"
+ },
+ "engines": {
+ "node": ">=14.0.0"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-win32-arm64-msvc": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-win32-arm64-msvc/-/oxide-win32-arm64-msvc-4.3.0.tgz",
+ "integrity": "sha512-Pe+RPVTi1T+qymuuRpcdvwSVZjnll/f7n8gBxMMh3xLTctMDKqpdfGimbMyioqtLhUYZxdJ9wGNhV7MKHvgZsQ==",
+ "cpu": [
+ "arm64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/oxide-win32-x64-msvc": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/oxide-win32-x64-msvc/-/oxide-win32-x64-msvc-4.3.0.tgz",
+ "integrity": "sha512-Mvrf2kXW/yeW/OTezZlCGOirXRcUuLIBx/5Y12BaPM7wJoryG6dfS/NJL8aBPqtTEx/Vm4T4vKzFUcKDT+TKUA==",
+ "cpu": [
+ "x64"
+ ],
+ "dev": true,
+ "license": "MIT",
+ "optional": true,
+ "os": [
+ "win32"
+ ],
+ "engines": {
+ "node": ">= 20"
+ }
+ },
+ "node_modules/@tailwindcss/vite": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/@tailwindcss/vite/-/vite-4.3.0.tgz",
+ "integrity": "sha512-t6J3OrB5Fc0ExuhohouH0fWUGMYL6PTLhW+E7zIk/pdbnJARZDCwjBznFnkh5ynRnIRSI4YjtTH0t6USjJISrw==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@tailwindcss/node": "4.3.0",
+ "@tailwindcss/oxide": "4.3.0",
+ "tailwindcss": "4.3.0"
+ },
+ "peerDependencies": {
+ "vite": "^5.2.0 || ^6 || ^7 || ^8"
+ }
+ },
+ "node_modules/@tanstack/query-core": {
+ "version": "5.100.11",
+ "resolved": "https://registry.npmmirror.com/@tanstack/query-core/-/query-core-5.100.11.tgz",
+ "integrity": "sha512-lmE0994apShXPj8CUxgx4ch5yUJhE9k/+tVwihBvPOyerACWdBocfFg24t8+0RhtlTd7tEgchDkhlCxNssvDxw==",
+ "license": "MIT",
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/tannerlinsley"
+ }
+ },
+ "node_modules/@tanstack/react-query": {
+ "version": "5.100.11",
+ "resolved": "https://registry.npmmirror.com/@tanstack/react-query/-/react-query-5.100.11.tgz",
+ "integrity": "sha512-J0f9s5x3LE1450nNNfYx+e/n0DMa0uOBdFJUy5r0RvmsXd4nB/n0rbHtHI1vYXhikNFan+wf51p6Tmp4c8ucrg==",
+ "license": "MIT",
+ "dependencies": {
+ "@tanstack/query-core": "5.100.11"
+ },
+ "funding": {
+ "type": "github",
+ "url": "https://github.com/sponsors/tannerlinsley"
+ },
+ "peerDependencies": {
+ "react": "^18 || ^19"
+ }
+ },
"node_modules/@tybys/wasm-util": {
"version": "0.10.2",
"resolved": "https://registry.npmjs.org/@tybys/wasm-util/-/wasm-util-0.10.2.tgz",
@@ -1148,6 +2398,21 @@
"url": "https://opencollective.com/typescript-eslint"
}
},
+ "node_modules/@umijs/route-utils": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmmirror.com/@umijs/route-utils/-/route-utils-4.0.3.tgz",
+ "integrity": "sha512-zPEcYhl1cSfkSRDzzGgoD1mDvGjxoOTJFvkn55srfgdQ3NZe2ZMCScCU6DEnOxuKP1XDVf8pqyqCDVd2+RCQIw==",
+ "license": "MIT"
+ },
+ "node_modules/@umijs/use-params": {
+ "version": "1.0.9",
+ "resolved": "https://registry.npmmirror.com/@umijs/use-params/-/use-params-1.0.9.tgz",
+ "integrity": "sha512-QlN0RJSBVQBwLRNxbxjQ5qzqYIGn+K7USppMoIOVlf7fxXHsnQZ2bEsa6Pm74bt6DVQxpUE8HqvdStn6Y9FV1w==",
+ "license": "MIT",
+ "peerDependencies": {
+ "react": "*"
+ }
+ },
"node_modules/@vitejs/plugin-react": {
"version": "6.0.2",
"resolved": "https://registry.npmjs.org/@vitejs/plugin-react/-/plugin-react-6.0.2.tgz",
@@ -1198,6 +2463,15 @@
"acorn": "^6.0.0 || ^7.0.0 || ^8.0.0"
}
},
+ "node_modules/add-dom-event-listener": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmmirror.com/add-dom-event-listener/-/add-dom-event-listener-1.1.0.tgz",
+ "integrity": "sha512-WCxx1ixHT0GQU9hb0KI/mhgRQhnU+U3GvwY6ZvVjYq8rsihIGoaIOUbY0yMPBxLH5MDtr0kz3fisWGNcbWW7Jw==",
+ "license": "MIT",
+ "dependencies": {
+ "object-assign": "4.x"
+ }
+ },
"node_modules/ajv": {
"version": "6.15.0",
"resolved": "https://registry.npmjs.org/ajv/-/ajv-6.15.0.tgz",
@@ -1215,6 +2489,113 @@
"url": "https://github.com/sponsors/epoberezkin"
}
},
+ "node_modules/antd": {
+ "version": "5.29.3",
+ "resolved": "https://registry.npmmirror.com/antd/-/antd-5.29.3.tgz",
+ "integrity": "sha512-3DdbGCa9tWAJGcCJ6rzR8EJFsv2CtyEbkVabZE14pfgUHfCicWCj0/QzQVLDYg8CPfQk9BH7fHCoTXHTy7MP/A==",
+ "license": "MIT",
+ "peer": true,
+ "dependencies": {
+ "@ant-design/colors": "^7.2.1",
+ "@ant-design/cssinjs": "^1.23.0",
+ "@ant-design/cssinjs-utils": "^1.1.3",
+ "@ant-design/fast-color": "^2.0.6",
+ "@ant-design/icons": "^5.6.1",
+ "@ant-design/react-slick": "~1.1.2",
+ "@babel/runtime": "^7.26.0",
+ "@rc-component/color-picker": "~2.0.1",
+ "@rc-component/mutate-observer": "^1.1.0",
+ "@rc-component/qrcode": "~1.1.0",
+ "@rc-component/tour": "~1.15.1",
+ "@rc-component/trigger": "^2.3.0",
+ "classnames": "^2.5.1",
+ "copy-to-clipboard": "^3.3.3",
+ "dayjs": "^1.11.11",
+ "rc-cascader": "~3.34.0",
+ "rc-checkbox": "~3.5.0",
+ "rc-collapse": "~3.9.0",
+ "rc-dialog": "~9.6.0",
+ "rc-drawer": "~7.3.0",
+ "rc-dropdown": "~4.2.1",
+ "rc-field-form": "~2.7.1",
+ "rc-image": "~7.12.0",
+ "rc-input": "~1.8.0",
+ "rc-input-number": "~9.5.0",
+ "rc-mentions": "~2.20.0",
+ "rc-menu": "~9.16.1",
+ "rc-motion": "^2.9.5",
+ "rc-notification": "~5.6.4",
+ "rc-pagination": "~5.1.0",
+ "rc-picker": "~4.11.3",
+ "rc-progress": "~4.0.0",
+ "rc-rate": "~2.13.1",
+ "rc-resize-observer": "^1.4.3",
+ "rc-segmented": "~2.7.0",
+ "rc-select": "~14.16.8",
+ "rc-slider": "~11.1.9",
+ "rc-steps": "~6.0.1",
+ "rc-switch": "~4.1.0",
+ "rc-table": "~7.54.0",
+ "rc-tabs": "~15.7.0",
+ "rc-textarea": "~1.10.2",
+ "rc-tooltip": "~6.4.0",
+ "rc-tree": "~5.13.1",
+ "rc-tree-select": "~5.27.0",
+ "rc-upload": "~4.11.0",
+ "rc-util": "^5.44.4",
+ "scroll-into-view-if-needed": "^3.1.0",
+ "throttle-debounce": "^5.0.2"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/ant-design"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/antd/node_modules/@ant-design/colors": {
+ "version": "7.2.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/colors/-/colors-7.2.1.tgz",
+ "integrity": "sha512-lCHDcEzieu4GA3n8ELeZ5VQ8pKQAWcGGLRTQ50aQM2iqPpq2evTxER84jfdPvsPAtEcZ7m44NI45edFMo8oOYQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/fast-color": "^2.0.6"
+ }
+ },
+ "node_modules/antd/node_modules/@ant-design/fast-color": {
+ "version": "2.0.6",
+ "resolved": "https://registry.npmmirror.com/@ant-design/fast-color/-/fast-color-2.0.6.tgz",
+ "integrity": "sha512-y2217gk4NqL35giHl72o6Zzqji9O7vHh9YmhUVkPtAOpoTCH4uWxo/pr4VE8t0+ChEPs0qo4eJRC5Q1eXWo3vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7"
+ },
+ "engines": {
+ "node": ">=8.x"
+ }
+ },
+ "node_modules/antd/node_modules/@ant-design/icons": {
+ "version": "5.6.1",
+ "resolved": "https://registry.npmmirror.com/@ant-design/icons/-/icons-5.6.1.tgz",
+ "integrity": "sha512-0/xS39c91WjPAZOWsvi1//zjx6kAp4kxWwctR6kuU6p133w8RU0D2dSCvZC19uQyharg/sAvYxGYWl01BbZZfg==",
+ "license": "MIT",
+ "dependencies": {
+ "@ant-design/colors": "^7.0.0",
+ "@ant-design/icons-svg": "^4.4.0",
+ "@babel/runtime": "^7.24.8",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.31.1"
+ },
+ "engines": {
+ "node": ">=8"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
"node_modules/balanced-match": {
"version": "4.0.4",
"resolved": "https://registry.npmjs.org/balanced-match/-/balanced-match-4.0.4.tgz",
@@ -1307,6 +2688,27 @@
],
"license": "CC-BY-4.0"
},
+ "node_modules/classnames": {
+ "version": "2.5.1",
+ "resolved": "https://registry.npmmirror.com/classnames/-/classnames-2.5.1.tgz",
+ "integrity": "sha512-saHYOzhIQs6wy2sVxTM6bUDsQO4F50V9RQ22qBpEdCW+I+/Wmke2HOl6lS6dTpdxVhb88/I6+Hs+438c3lfUow==",
+ "license": "MIT"
+ },
+ "node_modules/clsx": {
+ "version": "2.1.1",
+ "resolved": "https://registry.npmmirror.com/clsx/-/clsx-2.1.1.tgz",
+ "integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
+ "node_modules/compute-scroll-into-view": {
+ "version": "3.1.1",
+ "resolved": "https://registry.npmmirror.com/compute-scroll-into-view/-/compute-scroll-into-view-3.1.1.tgz",
+ "integrity": "sha512-VRhuHOLoKYOy4UbilLbUzbYg93XLjv2PncJC50EuTWPA3gaja1UjBsUP/D/9/juV3vQFr6XBEzn9KCAHdUvOHw==",
+ "license": "MIT"
+ },
"node_modules/convert-source-map": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/convert-source-map/-/convert-source-map-2.0.0.tgz",
@@ -1314,6 +2716,28 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/cookie": {
+ "version": "1.1.1",
+ "resolved": "https://registry.npmmirror.com/cookie/-/cookie-1.1.1.tgz",
+ "integrity": "sha512-ei8Aos7ja0weRpFzJnEA9UHJ/7XQmqglbRwnf2ATjcB9Wq874VKH9kfjjirM6UhU2/E5fFYadylyhFldcqSidQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=18"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/express"
+ }
+ },
+ "node_modules/copy-to-clipboard": {
+ "version": "3.3.3",
+ "resolved": "https://registry.npmmirror.com/copy-to-clipboard/-/copy-to-clipboard-3.3.3.tgz",
+ "integrity": "sha512-2KV8NhB5JqC3ky0r9PMCAZKbUHSwtEo4CwCs0KXgruG43gX5PMqDEBbVU4OUzw2MuAWUfsuFmWvEKG5QRfSnJA==",
+ "license": "MIT",
+ "dependencies": {
+ "toggle-selection": "^1.0.6"
+ }
+ },
"node_modules/cross-spawn": {
"version": "7.0.6",
"resolved": "https://registry.npmjs.org/cross-spawn/-/cross-spawn-7.0.6.tgz",
@@ -1333,7 +2757,12 @@
"version": "3.2.3",
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz",
"integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==",
- "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/dayjs": {
+ "version": "1.11.20",
+ "resolved": "https://registry.npmmirror.com/dayjs/-/dayjs-1.11.20.tgz",
+ "integrity": "sha512-YbwwqR/uYpeoP4pu043q+LTDLFBLApUP6VxRihdfNTqu4ubqMlGDLd6ErXhEgsyvY0K6nCs7nggYumAN+9uEuQ==",
"license": "MIT"
},
"node_modules/debug": {
@@ -1361,6 +2790,15 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/dequal": {
+ "version": "2.0.3",
+ "resolved": "https://registry.npmmirror.com/dequal/-/dequal-2.0.3.tgz",
+ "integrity": "sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ }
+ },
"node_modules/detect-libc": {
"version": "2.1.2",
"resolved": "https://registry.npmjs.org/detect-libc/-/detect-libc-2.1.2.tgz",
@@ -1371,6 +2809,36 @@
"node": ">=8"
}
},
+ "node_modules/echarts": {
+ "version": "6.1.0",
+ "resolved": "https://registry.npmmirror.com/echarts/-/echarts-6.1.0.tgz",
+ "integrity": "sha512-q0yaFPggC9FUdsWH4blavRWFmxdrIodbkoKNAjJudAI6CA9gNPxHtV2RcZNEepZVlk4yvBYkOkbk6HIVpIyHZA==",
+ "license": "Apache-2.0",
+ "dependencies": {
+ "tslib": "2.3.0",
+ "zrender": "6.1.0"
+ }
+ },
+ "node_modules/echarts-for-react": {
+ "version": "3.0.6",
+ "resolved": "https://registry.npmmirror.com/echarts-for-react/-/echarts-for-react-3.0.6.tgz",
+ "integrity": "sha512-4zqLgTGWS3JvkQDXjzkR1k1CHRdpd6by0988TWMJgnvDytegWLbeP/VNZmMa+0VJx2eD7Y632bi2JquXDgiGJg==",
+ "license": "MIT",
+ "dependencies": {
+ "fast-deep-equal": "^3.1.3",
+ "size-sensor": "^1.0.1"
+ },
+ "peerDependencies": {
+ "echarts": "^3.0.0 || ^4.0.0 || ^5.0.0 || ^6.0.0",
+ "react": "^15.0.0 || >=16.0.0"
+ }
+ },
+ "node_modules/echarts/node_modules/tslib": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmmirror.com/tslib/-/tslib-2.3.0.tgz",
+ "integrity": "sha512-N82ooyxVNm6h1riLCoyS9e3fuJ3AMG2zIZs2Gd1ATcSFjSA23Q0fzjjZeh0jbJvWVDZ0cJT8yaNNaaXHzueNjg==",
+ "license": "0BSD"
+ },
"node_modules/electron-to-chromium": {
"version": "1.5.360",
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.360.tgz",
@@ -1378,6 +2846,20 @@
"dev": true,
"license": "ISC"
},
+ "node_modules/enhanced-resolve": {
+ "version": "5.21.6",
+ "resolved": "https://registry.npmmirror.com/enhanced-resolve/-/enhanced-resolve-5.21.6.tgz",
+ "integrity": "sha512-aNnGCvbJ/RIyWo1IuhNdVjnNF+EjH9wpzpNHt+ci/m9He9LJvUN8wrCcXjp9cWsGNAuvSpVFTx/vraAFQ8qGjQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "graceful-fs": "^4.2.4",
+ "tapable": "^2.3.3"
+ },
+ "engines": {
+ "node": ">=10.13.0"
+ }
+ },
"node_modules/escalade": {
"version": "3.2.0",
"resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
@@ -1588,7 +3070,6 @@
"version": "3.1.3",
"resolved": "https://registry.npmjs.org/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz",
"integrity": "sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q==",
- "dev": true,
"license": "MIT"
},
"node_modules/fast-json-stable-stringify": {
@@ -1725,6 +3206,13 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/graceful-fs": {
+ "version": "4.2.11",
+ "resolved": "https://registry.npmmirror.com/graceful-fs/-/graceful-fs-4.2.11.tgz",
+ "integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==",
+ "dev": true,
+ "license": "ISC"
+ },
"node_modules/hermes-estree": {
"version": "0.25.1",
"resolved": "https://registry.npmjs.org/hermes-estree/-/hermes-estree-0.25.1.tgz",
@@ -1785,6 +3273,12 @@
"node": ">=0.10.0"
}
},
+ "node_modules/is-mobile": {
+ "version": "5.0.0",
+ "resolved": "https://registry.npmmirror.com/is-mobile/-/is-mobile-5.0.0.tgz",
+ "integrity": "sha512-Tz/yndySvLAEXh+Uk8liFCxOwVH6YutuR74utvOcu7I9Di+DwM0mtdPVZNaVvvBUM2OXxne/NhOs1zAO7riusQ==",
+ "license": "MIT"
+ },
"node_modules/isexe": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/isexe/-/isexe-2.0.0.tgz",
@@ -1792,11 +3286,20 @@
"dev": true,
"license": "ISC"
},
+ "node_modules/jiti": {
+ "version": "2.7.0",
+ "resolved": "https://registry.npmmirror.com/jiti/-/jiti-2.7.0.tgz",
+ "integrity": "sha512-AC/7JofJvZGrrneWNaEnJeOLUx+JlGt7tNa0wZiRPT4MY1wmfKjt2+6O2p2uz2+skll8OZZmJMNqeke7kKbNgQ==",
+ "dev": true,
+ "license": "MIT",
+ "bin": {
+ "jiti": "lib/jiti-cli.mjs"
+ }
+ },
"node_modules/js-tokens": {
"version": "4.0.0",
"resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
"integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
- "dev": true,
"license": "MIT"
},
"node_modules/jsesc": {
@@ -1833,6 +3336,15 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/json2mq": {
+ "version": "0.2.0",
+ "resolved": "https://registry.npmmirror.com/json2mq/-/json2mq-0.2.0.tgz",
+ "integrity": "sha512-SzoRg7ux5DWTII9J2qkrZrqV1gt+rTaoufMxEzXbS26Uid0NwaJd123HcoB80TgubEppxxIGdNxCx50fEoEWQA==",
+ "license": "MIT",
+ "dependencies": {
+ "string-convert": "^0.2.0"
+ }
+ },
"node_modules/json5": {
"version": "2.2.3",
"resolved": "https://registry.npmjs.org/json5/-/json5-2.2.3.tgz",
@@ -2147,6 +3659,30 @@
"url": "https://github.com/sponsors/sindresorhus"
}
},
+ "node_modules/lodash": {
+ "version": "4.18.1",
+ "resolved": "https://registry.npmmirror.com/lodash/-/lodash-4.18.1.tgz",
+ "integrity": "sha512-dMInicTPVE8d1e5otfwmmjlxkZoUpiVLwyeTdUsi/Caj/gfzzblBcCE5sRHV/AsjuCmxWrte2TNGSYuCeCq+0Q==",
+ "license": "MIT"
+ },
+ "node_modules/lodash-es": {
+ "version": "4.18.1",
+ "resolved": "https://registry.npmmirror.com/lodash-es/-/lodash-es-4.18.1.tgz",
+ "integrity": "sha512-J8xewKD/Gk22OZbhpOVSwcs60zhd95ESDwezOFuA3/099925PdHJ7OFHNTGtajL3AlZkykD32HykiMo+BIBI8A==",
+ "license": "MIT"
+ },
+ "node_modules/loose-envify": {
+ "version": "1.4.0",
+ "resolved": "https://registry.npmmirror.com/loose-envify/-/loose-envify-1.4.0.tgz",
+ "integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==",
+ "license": "MIT",
+ "dependencies": {
+ "js-tokens": "^3.0.0 || ^4.0.0"
+ },
+ "bin": {
+ "loose-envify": "cli.js"
+ }
+ },
"node_modules/lru-cache": {
"version": "5.1.1",
"resolved": "https://registry.npmjs.org/lru-cache/-/lru-cache-5.1.1.tgz",
@@ -2157,6 +3693,16 @@
"yallist": "^3.0.2"
}
},
+ "node_modules/magic-string": {
+ "version": "0.30.21",
+ "resolved": "https://registry.npmmirror.com/magic-string/-/magic-string-0.30.21.tgz",
+ "integrity": "sha512-vd2F4YUyEXKGcLHoq+TEyCjxueSeHnFxyyjNp80yg0XV4vUhnDer/lvvlqM/arB5bXQN5K2/3oinyCRyx8T2CQ==",
+ "dev": true,
+ "license": "MIT",
+ "dependencies": {
+ "@jridgewell/sourcemap-codec": "^1.5.5"
+ }
+ },
"node_modules/minimatch": {
"version": "10.2.5",
"resolved": "https://registry.npmjs.org/minimatch/-/minimatch-10.2.5.tgz",
@@ -2213,6 +3759,15 @@
"dev": true,
"license": "MIT"
},
+ "node_modules/object-assign": {
+ "version": "4.1.1",
+ "resolved": "https://registry.npmmirror.com/object-assign/-/object-assign-4.1.1.tgz",
+ "integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=0.10.0"
+ }
+ },
"node_modules/optionator": {
"version": "0.9.4",
"resolved": "https://registry.npmjs.org/optionator/-/optionator-0.9.4.tgz",
@@ -2283,6 +3838,15 @@
"node": ">=8"
}
},
+ "node_modules/path-to-regexp": {
+ "version": "8.2.0",
+ "resolved": "https://registry.npmmirror.com/path-to-regexp/-/path-to-regexp-8.2.0.tgz",
+ "integrity": "sha512-TdrF7fW9Rphjq4RjrW0Kp2AW0Ahwu9sRGTkS6bvDi0SCwZlEZYmcfDbEsTz8RVk0EHIS/Vd1bv3JhG+1xZuAyQ==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=16"
+ }
+ },
"node_modules/picocolors": {
"version": "1.1.1",
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
@@ -2343,6 +3907,23 @@
"node": ">= 0.8.0"
}
},
+ "node_modules/prop-types": {
+ "version": "15.8.1",
+ "resolved": "https://registry.npmmirror.com/prop-types/-/prop-types-15.8.1.tgz",
+ "integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==",
+ "license": "MIT",
+ "dependencies": {
+ "loose-envify": "^1.4.0",
+ "object-assign": "^4.1.1",
+ "react-is": "^16.13.1"
+ }
+ },
+ "node_modules/prop-types/node_modules/react-is": {
+ "version": "16.13.1",
+ "resolved": "https://registry.npmmirror.com/react-is/-/react-is-16.13.1.tgz",
+ "integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==",
+ "license": "MIT"
+ },
"node_modules/punycode": {
"version": "2.3.1",
"resolved": "https://registry.npmjs.org/punycode/-/punycode-2.3.1.tgz",
@@ -2353,6 +3934,613 @@
"node": ">=6"
}
},
+ "node_modules/rc-cascader": {
+ "version": "3.34.0",
+ "resolved": "https://registry.npmmirror.com/rc-cascader/-/rc-cascader-3.34.0.tgz",
+ "integrity": "sha512-KpXypcvju9ptjW9FaN2NFcA2QH9E9LHKq169Y0eWtH4e/wHQ5Wh5qZakAgvb8EKZ736WZ3B0zLLOBsrsja5Dag==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.25.7",
+ "classnames": "^2.3.1",
+ "rc-select": "~14.16.2",
+ "rc-tree": "~5.13.0",
+ "rc-util": "^5.43.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-checkbox": {
+ "version": "3.5.0",
+ "resolved": "https://registry.npmmirror.com/rc-checkbox/-/rc-checkbox-3.5.0.tgz",
+ "integrity": "sha512-aOAQc3E98HteIIsSqm6Xk2FPKIER6+5vyEFMZfo73TqM+VVAIqOkHoPjgKLqSNtVLWScoaM7vY2ZrGEheI79yg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "^2.3.2",
+ "rc-util": "^5.25.2"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-collapse": {
+ "version": "3.9.0",
+ "resolved": "https://registry.npmmirror.com/rc-collapse/-/rc-collapse-3.9.0.tgz",
+ "integrity": "sha512-swDdz4QZ4dFTo4RAUMLL50qP0EY62N2kvmk2We5xYdRwcRn8WcYtuetCJpwpaCbUfUt5+huLpVxhvmnK+PHrkA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "2.x",
+ "rc-motion": "^2.3.4",
+ "rc-util": "^5.27.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-dialog": {
+ "version": "9.6.0",
+ "resolved": "https://registry.npmmirror.com/rc-dialog/-/rc-dialog-9.6.0.tgz",
+ "integrity": "sha512-ApoVi9Z8PaCQg6FsUzS8yvBEQy0ZL2PkuvAgrmohPkN3okps5WZ5WQWPc1RNuiOKaAYv8B97ACdsFU5LizzCqg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "@rc-component/portal": "^1.0.0-8",
+ "classnames": "^2.2.6",
+ "rc-motion": "^2.3.0",
+ "rc-util": "^5.21.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-drawer": {
+ "version": "7.3.0",
+ "resolved": "https://registry.npmmirror.com/rc-drawer/-/rc-drawer-7.3.0.tgz",
+ "integrity": "sha512-DX6CIgiBWNpJIMGFO8BAISFkxiuKitoizooj4BDyee8/SnBn0zwO2FHrNDpqqepj0E/TFTDpmEBCyFuTgC7MOg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.23.9",
+ "@rc-component/portal": "^1.1.1",
+ "classnames": "^2.2.6",
+ "rc-motion": "^2.6.1",
+ "rc-util": "^5.38.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-dropdown": {
+ "version": "4.2.1",
+ "resolved": "https://registry.npmmirror.com/rc-dropdown/-/rc-dropdown-4.2.1.tgz",
+ "integrity": "sha512-YDAlXsPv3I1n42dv1JpdM7wJ+gSUBfeyPK59ZpBD9jQhK9jVuxpjj3NmWQHOBceA1zEPVX84T2wbdb2SD0UjmA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.3",
+ "@rc-component/trigger": "^2.0.0",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.44.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.11.0",
+ "react-dom": ">=16.11.0"
+ }
+ },
+ "node_modules/rc-field-form": {
+ "version": "2.7.1",
+ "resolved": "https://registry.npmmirror.com/rc-field-form/-/rc-field-form-2.7.1.tgz",
+ "integrity": "sha512-vKeSifSJ6HoLaAB+B8aq/Qgm8a3dyxROzCtKNCsBQgiverpc4kWDQihoUwzUj+zNWJOykwSY4dNX3QrGwtVb9A==",
+ "license": "MIT",
+ "peer": true,
+ "dependencies": {
+ "@babel/runtime": "^7.18.0",
+ "@rc-component/async-validator": "^5.0.3",
+ "rc-util": "^5.32.2"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-image": {
+ "version": "7.12.0",
+ "resolved": "https://registry.npmmirror.com/rc-image/-/rc-image-7.12.0.tgz",
+ "integrity": "sha512-cZ3HTyyckPnNnUb9/DRqduqzLfrQRyi+CdHjdqgsyDpI3Ln5UX1kXnAhPBSJj9pVRzwRFgqkN7p9b6HBDjmu/Q==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.11.2",
+ "@rc-component/portal": "^1.0.2",
+ "classnames": "^2.2.6",
+ "rc-dialog": "~9.6.0",
+ "rc-motion": "^2.6.2",
+ "rc-util": "^5.34.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-input": {
+ "version": "1.8.0",
+ "resolved": "https://registry.npmmirror.com/rc-input/-/rc-input-1.8.0.tgz",
+ "integrity": "sha512-KXvaTbX+7ha8a/k+eg6SYRVERK0NddX8QX7a7AnRvUa/rEH0CNMlpcBzBkhI0wp2C8C4HlMoYl8TImSN+fuHKA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.11.1",
+ "classnames": "^2.2.1",
+ "rc-util": "^5.18.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/rc-input-number": {
+ "version": "9.5.0",
+ "resolved": "https://registry.npmmirror.com/rc-input-number/-/rc-input-number-9.5.0.tgz",
+ "integrity": "sha512-bKaEvB5tHebUURAEXw35LDcnRZLq3x1k7GxfAqBMzmpHkDGzjAtnUL8y4y5N15rIFIg5IJgwr211jInl3cipag==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "@rc-component/mini-decimal": "^1.0.1",
+ "classnames": "^2.2.5",
+ "rc-input": "~1.8.0",
+ "rc-util": "^5.40.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-mentions": {
+ "version": "2.20.0",
+ "resolved": "https://registry.npmmirror.com/rc-mentions/-/rc-mentions-2.20.0.tgz",
+ "integrity": "sha512-w8HCMZEh3f0nR8ZEd466ATqmXFCMGMN5UFCzEUL0bM/nGw/wOS2GgRzKBcm19K++jDyuWCOJOdgcKGXU3fXfbQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.22.5",
+ "@rc-component/trigger": "^2.0.0",
+ "classnames": "^2.2.6",
+ "rc-input": "~1.8.0",
+ "rc-menu": "~9.16.0",
+ "rc-textarea": "~1.10.0",
+ "rc-util": "^5.34.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-menu": {
+ "version": "9.16.1",
+ "resolved": "https://registry.npmmirror.com/rc-menu/-/rc-menu-9.16.1.tgz",
+ "integrity": "sha512-ghHx6/6Dvp+fw8CJhDUHFHDJ84hJE3BXNCzSgLdmNiFErWSOaZNsihDAsKq9ByTALo/xkNIwtDFGIl6r+RPXBg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "@rc-component/trigger": "^2.0.0",
+ "classnames": "2.x",
+ "rc-motion": "^2.4.3",
+ "rc-overflow": "^1.3.1",
+ "rc-util": "^5.27.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-motion": {
+ "version": "2.9.5",
+ "resolved": "https://registry.npmmirror.com/rc-motion/-/rc-motion-2.9.5.tgz",
+ "integrity": "sha512-w+XTUrfh7ArbYEd2582uDrEhmBHwK1ZENJiSJVb7uRxdE7qJSYjbO2eksRXmndqyKqKoYPc9ClpPh5242mV1vA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.11.1",
+ "classnames": "^2.2.1",
+ "rc-util": "^5.44.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-notification": {
+ "version": "5.6.4",
+ "resolved": "https://registry.npmmirror.com/rc-notification/-/rc-notification-5.6.4.tgz",
+ "integrity": "sha512-KcS4O6B4qzM3KH7lkwOB7ooLPZ4b6J+VMmQgT51VZCeEcmghdeR4IrMcFq0LG+RPdnbe/ArT086tGM8Snimgiw==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "2.x",
+ "rc-motion": "^2.9.0",
+ "rc-util": "^5.20.1"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-overflow": {
+ "version": "1.5.0",
+ "resolved": "https://registry.npmmirror.com/rc-overflow/-/rc-overflow-1.5.0.tgz",
+ "integrity": "sha512-Lm/v9h0LymeUYJf0x39OveU52InkdRXqnn2aYXfWmo8WdOonIKB2kfau+GF0fWq6jPgtdO9yMqveGcK6aIhJmg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.11.1",
+ "classnames": "^2.2.1",
+ "rc-resize-observer": "^1.0.0",
+ "rc-util": "^5.37.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-pagination": {
+ "version": "5.1.0",
+ "resolved": "https://registry.npmmirror.com/rc-pagination/-/rc-pagination-5.1.0.tgz",
+ "integrity": "sha512-8416Yip/+eclTFdHXLKTxZvn70duYVGTvUUWbckCCZoIl3jagqke3GLsFrMs0bsQBikiYpZLD9206Ej4SOdOXQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "^2.3.2",
+ "rc-util": "^5.38.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-picker": {
+ "version": "4.11.3",
+ "resolved": "https://registry.npmmirror.com/rc-picker/-/rc-picker-4.11.3.tgz",
+ "integrity": "sha512-MJ5teb7FlNE0NFHTncxXQ62Y5lytq6sh5nUw0iH8OkHL/TjARSEvSHpr940pWgjGANpjCwyMdvsEV55l5tYNSg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.24.7",
+ "@rc-component/trigger": "^2.0.0",
+ "classnames": "^2.2.1",
+ "rc-overflow": "^1.3.2",
+ "rc-resize-observer": "^1.4.0",
+ "rc-util": "^5.43.0"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "date-fns": ">= 2.x",
+ "dayjs": ">= 1.x",
+ "luxon": ">= 3.x",
+ "moment": ">= 2.x",
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ },
+ "peerDependenciesMeta": {
+ "date-fns": {
+ "optional": true
+ },
+ "dayjs": {
+ "optional": true
+ },
+ "luxon": {
+ "optional": true
+ },
+ "moment": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/rc-progress": {
+ "version": "4.0.0",
+ "resolved": "https://registry.npmmirror.com/rc-progress/-/rc-progress-4.0.0.tgz",
+ "integrity": "sha512-oofVMMafOCokIUIBnZLNcOZFsABaUw8PPrf1/y0ZBvKZNpOiu5h4AO9vv11Sw0p4Hb3D0yGWuEattcQGtNJ/aw==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "^2.2.6",
+ "rc-util": "^5.16.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-rate": {
+ "version": "2.13.1",
+ "resolved": "https://registry.npmmirror.com/rc-rate/-/rc-rate-2.13.1.tgz",
+ "integrity": "sha512-QUhQ9ivQ8Gy7mtMZPAjLbxBt5y9GRp65VcUyGUMF3N3fhiftivPHdpuDIaWIMOTEprAjZPC08bls1dQB+I1F2Q==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "^2.2.5",
+ "rc-util": "^5.0.1"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-resize-observer": {
+ "version": "1.4.3",
+ "resolved": "https://registry.npmmirror.com/rc-resize-observer/-/rc-resize-observer-1.4.3.tgz",
+ "integrity": "sha512-YZLjUbyIWox8E9i9C3Tm7ia+W7euPItNWSPX5sCcQTYbnwDb5uNpnLHQCG1f22oZWUhLw4Mv2tFmeWe68CDQRQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.20.7",
+ "classnames": "^2.2.1",
+ "rc-util": "^5.44.1",
+ "resize-observer-polyfill": "^1.5.1"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-segmented": {
+ "version": "2.7.1",
+ "resolved": "https://registry.npmmirror.com/rc-segmented/-/rc-segmented-2.7.1.tgz",
+ "integrity": "sha512-izj1Nw/Dw2Vb7EVr+D/E9lUTkBe+kKC+SAFSU9zqr7WV2W5Ktaa9Gc7cB2jTqgk8GROJayltaec+DBlYKc6d+g==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.11.1",
+ "classnames": "^2.2.1",
+ "rc-motion": "^2.4.4",
+ "rc-util": "^5.17.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.0.0",
+ "react-dom": ">=16.0.0"
+ }
+ },
+ "node_modules/rc-select": {
+ "version": "14.16.8",
+ "resolved": "https://registry.npmmirror.com/rc-select/-/rc-select-14.16.8.tgz",
+ "integrity": "sha512-NOV5BZa1wZrsdkKaiK7LHRuo5ZjZYMDxPP6/1+09+FB4KoNi8jcG1ZqLE3AVCxEsYMBe65OBx71wFoHRTP3LRg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "@rc-component/trigger": "^2.1.1",
+ "classnames": "2.x",
+ "rc-motion": "^2.0.1",
+ "rc-overflow": "^1.3.1",
+ "rc-util": "^5.16.1",
+ "rc-virtual-list": "^3.5.2"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": "*",
+ "react-dom": "*"
+ }
+ },
+ "node_modules/rc-slider": {
+ "version": "11.1.9",
+ "resolved": "https://registry.npmmirror.com/rc-slider/-/rc-slider-11.1.9.tgz",
+ "integrity": "sha512-h8IknhzSh3FEM9u8ivkskh+Ef4Yo4JRIY2nj7MrH6GQmrwV6mcpJf5/4KgH5JaVI1H3E52yCdpOlVyGZIeph5A==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "^2.2.5",
+ "rc-util": "^5.36.0"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-steps": {
+ "version": "6.0.1",
+ "resolved": "https://registry.npmmirror.com/rc-steps/-/rc-steps-6.0.1.tgz",
+ "integrity": "sha512-lKHL+Sny0SeHkQKKDJlAjV5oZ8DwCdS2hFhAkIjuQt1/pB81M0cA0ErVFdHq9+jmPmFw1vJB2F5NBzFXLJxV+g==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.16.7",
+ "classnames": "^2.2.3",
+ "rc-util": "^5.16.1"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-switch": {
+ "version": "4.1.0",
+ "resolved": "https://registry.npmmirror.com/rc-switch/-/rc-switch-4.1.0.tgz",
+ "integrity": "sha512-TI8ufP2Az9oEbvyCeVE4+90PDSljGyuwix3fV58p7HV2o4wBnVToEyomJRVyTaZeqNPAp+vqeo4Wnj5u0ZZQBg==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.21.0",
+ "classnames": "^2.2.1",
+ "rc-util": "^5.30.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-table": {
+ "version": "7.54.0",
+ "resolved": "https://registry.npmmirror.com/rc-table/-/rc-table-7.54.0.tgz",
+ "integrity": "sha512-/wDTkki6wBTjwylwAGjpLKYklKo9YgjZwAU77+7ME5mBoS32Q4nAwoqhA2lSge6fobLW3Tap6uc5xfwaL2p0Sw==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "@rc-component/context": "^1.4.0",
+ "classnames": "^2.2.5",
+ "rc-resize-observer": "^1.1.0",
+ "rc-util": "^5.44.3",
+ "rc-virtual-list": "^3.14.2"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-tabs": {
+ "version": "15.7.0",
+ "resolved": "https://registry.npmmirror.com/rc-tabs/-/rc-tabs-15.7.0.tgz",
+ "integrity": "sha512-ZepiE+6fmozYdWf/9gVp7k56PKHB1YYoDsKeQA1CBlJ/POIhjkcYiv0AGP0w2Jhzftd3AVvZP/K+V+Lpi2ankA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.11.2",
+ "classnames": "2.x",
+ "rc-dropdown": "~4.2.0",
+ "rc-menu": "~9.16.0",
+ "rc-motion": "^2.6.2",
+ "rc-resize-observer": "^1.0.0",
+ "rc-util": "^5.34.1"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-textarea": {
+ "version": "1.10.2",
+ "resolved": "https://registry.npmmirror.com/rc-textarea/-/rc-textarea-1.10.2.tgz",
+ "integrity": "sha512-HfaeXiaSlpiSp0I/pvWpecFEHpVysZ9tpDLNkxQbMvMz6gsr7aVZ7FpWP9kt4t7DB+jJXesYS0us1uPZnlRnwQ==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "^2.2.1",
+ "rc-input": "~1.8.0",
+ "rc-resize-observer": "^1.0.0",
+ "rc-util": "^5.27.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-tooltip": {
+ "version": "6.4.0",
+ "resolved": "https://registry.npmmirror.com/rc-tooltip/-/rc-tooltip-6.4.0.tgz",
+ "integrity": "sha512-kqyivim5cp8I5RkHmpsp1Nn/Wk+1oeloMv9c7LXNgDxUpGm+RbXJGL+OPvDlcRnx9DBeOe4wyOIl4OKUERyH1g==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.11.2",
+ "@rc-component/trigger": "^2.0.0",
+ "classnames": "^2.3.1",
+ "rc-util": "^5.44.3"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-tree": {
+ "version": "5.13.1",
+ "resolved": "https://registry.npmmirror.com/rc-tree/-/rc-tree-5.13.1.tgz",
+ "integrity": "sha512-FNhIefhftobCdUJshO7M8uZTA9F4OPGVXqGfZkkD/5soDeOhwO06T/aKTrg0WD8gRg/pyfq+ql3aMymLHCTC4A==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.10.1",
+ "classnames": "2.x",
+ "rc-motion": "^2.0.1",
+ "rc-util": "^5.16.1",
+ "rc-virtual-list": "^3.5.1"
+ },
+ "engines": {
+ "node": ">=10.x"
+ },
+ "peerDependencies": {
+ "react": "*",
+ "react-dom": "*"
+ }
+ },
+ "node_modules/rc-tree-select": {
+ "version": "5.27.0",
+ "resolved": "https://registry.npmmirror.com/rc-tree-select/-/rc-tree-select-5.27.0.tgz",
+ "integrity": "sha512-2qTBTzwIT7LRI1o7zLyrCzmo5tQanmyGbSaGTIf7sYimCklAToVVfpMC6OAldSKolcnjorBYPNSKQqJmN3TCww==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.25.7",
+ "classnames": "2.x",
+ "rc-select": "~14.16.2",
+ "rc-tree": "~5.13.0",
+ "rc-util": "^5.43.0"
+ },
+ "peerDependencies": {
+ "react": "*",
+ "react-dom": "*"
+ }
+ },
+ "node_modules/rc-upload": {
+ "version": "4.11.0",
+ "resolved": "https://registry.npmmirror.com/rc-upload/-/rc-upload-4.11.0.tgz",
+ "integrity": "sha512-ZUyT//2JAehfHzjWowqROcwYJKnZkIUGWaTE/VogVrepSl7AFNbQf4+zGfX4zl9Vrj/Jm8scLO0R6UlPDKK4wA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.3",
+ "classnames": "^2.2.5",
+ "rc-util": "^5.2.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-util": {
+ "version": "5.44.4",
+ "resolved": "https://registry.npmmirror.com/rc-util/-/rc-util-5.44.4.tgz",
+ "integrity": "sha512-resueRJzmHG9Q6rI/DfK6Kdv9/Lfls05vzMs1Sk3M2P+3cJa+MakaZyWY8IPfehVuhPJFKrIY1IK4GqbiaiY5w==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.18.3",
+ "react-is": "^18.2.0"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
+ "node_modules/rc-virtual-list": {
+ "version": "3.19.2",
+ "resolved": "https://registry.npmmirror.com/rc-virtual-list/-/rc-virtual-list-3.19.2.tgz",
+ "integrity": "sha512-Ys6NcjwGkuwkeaWBDqfI3xWuZ7rDiQXlH1o2zLfFzATfEgXcqpk8CkgMfbJD81McqjcJVez25a3kPxCR807evA==",
+ "license": "MIT",
+ "dependencies": {
+ "@babel/runtime": "^7.20.0",
+ "classnames": "^2.2.6",
+ "rc-resize-observer": "^1.0.0",
+ "rc-util": "^5.36.0"
+ },
+ "engines": {
+ "node": ">=8.x"
+ },
+ "peerDependencies": {
+ "react": ">=16.9.0",
+ "react-dom": ">=16.9.0"
+ }
+ },
"node_modules/react": {
"version": "19.2.6",
"resolved": "https://registry.npmjs.org/react/-/react-19.2.6.tgz",
@@ -2368,6 +4556,7 @@
"resolved": "https://registry.npmjs.org/react-dom/-/react-dom-19.2.6.tgz",
"integrity": "sha512-0prMI+hvBbPjsWnxDLxlCGyM8PN6UuWjEUCYmZhO67xIV9Xasa/r/vDnq+Xyq4Lo27g8QSbO5YzARu0D1Sps3g==",
"license": "MIT",
+ "peer": true,
"dependencies": {
"scheduler": "^0.27.0"
},
@@ -2375,6 +4564,71 @@
"react": "^19.2.6"
}
},
+ "node_modules/react-is": {
+ "version": "18.3.1",
+ "resolved": "https://registry.npmmirror.com/react-is/-/react-is-18.3.1.tgz",
+ "integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==",
+ "license": "MIT"
+ },
+ "node_modules/react-lifecycles-compat": {
+ "version": "3.0.4",
+ "resolved": "https://registry.npmmirror.com/react-lifecycles-compat/-/react-lifecycles-compat-3.0.4.tgz",
+ "integrity": "sha512-fBASbA6LnOU9dOU2eW7aQ8xmYBSXUIWr+UmF9b1efZBazGNO+rcXT/icdKnYm2pTwcRylVUYwW7H1PHfLekVzA==",
+ "license": "MIT"
+ },
+ "node_modules/react-router": {
+ "version": "7.15.1",
+ "resolved": "https://registry.npmmirror.com/react-router/-/react-router-7.15.1.tgz",
+ "integrity": "sha512-R8rl9HhgikFYoPJymnUtPXWbnDb3oget6lQnfIoupbt61aT9aOhRkDsY2XRhZRyX1Z/8a5sL74fXmFNm3NRK5A==",
+ "license": "MIT",
+ "dependencies": {
+ "cookie": "^1.0.1",
+ "set-cookie-parser": "^2.6.0"
+ },
+ "engines": {
+ "node": ">=20.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=18",
+ "react-dom": ">=18"
+ },
+ "peerDependenciesMeta": {
+ "react-dom": {
+ "optional": true
+ }
+ }
+ },
+ "node_modules/react-router-dom": {
+ "version": "7.15.1",
+ "resolved": "https://registry.npmmirror.com/react-router-dom/-/react-router-dom-7.15.1.tgz",
+ "integrity": "sha512-AzF62gjY6U9rkMq4RfP/r2EVtQ7DMfNMjyOp/flLTCrtRylLiK4wT4pSq6O8rOXZ2eXdZYJPEYe+ifomiv+Igg==",
+ "license": "MIT",
+ "dependencies": {
+ "react-router": "7.15.1"
+ },
+ "engines": {
+ "node": ">=20.0.0"
+ },
+ "peerDependencies": {
+ "react": ">=18",
+ "react-dom": ">=18"
+ }
+ },
+ "node_modules/reactcss": {
+ "version": "1.2.3",
+ "resolved": "https://registry.npmmirror.com/reactcss/-/reactcss-1.2.3.tgz",
+ "integrity": "sha512-KiwVUcFu1RErkI97ywr8nvx8dNOpT03rbnma0SSalTYjkrPYaEajR4a/MRt6DZ46K6arDRbWMNHF+xH7G7n/8A==",
+ "license": "MIT",
+ "dependencies": {
+ "lodash": "^4.0.1"
+ }
+ },
+ "node_modules/resize-observer-polyfill": {
+ "version": "1.5.1",
+ "resolved": "https://registry.npmmirror.com/resize-observer-polyfill/-/resize-observer-polyfill-1.5.1.tgz",
+ "integrity": "sha512-LwZrotdHOo12nQuZlHEmtuXdqGoOD0OhaxopaNFxWzInpEgaLWoVuAMbTzixuosCx2nEG58ngzW3vxdWoxIgdg==",
+ "license": "MIT"
+ },
"node_modules/rolldown": {
"version": "1.0.1",
"resolved": "https://registry.npmjs.org/rolldown/-/rolldown-1.0.1.tgz",
@@ -2409,12 +4663,30 @@
"@rolldown/binding-win32-x64-msvc": "1.0.1"
}
},
+ "node_modules/safe-stable-stringify": {
+ "version": "2.5.0",
+ "resolved": "https://registry.npmmirror.com/safe-stable-stringify/-/safe-stable-stringify-2.5.0.tgz",
+ "integrity": "sha512-b3rppTKm9T+PsVCBEOUR46GWI7fdOs00VKZ1+9c1EWDaDMvjQc6tUwuFyIprgGgTcWoVHSKrU8H31ZHA2e0RHA==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=10"
+ }
+ },
"node_modules/scheduler": {
"version": "0.27.0",
"resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.27.0.tgz",
"integrity": "sha512-eNv+WrVbKu1f3vbYJT/xtiF5syA5HPIMtf9IgY/nKg0sWqzAUEvqY/xm7OcZc/qafLx/iO9FgOmeSAp4v5ti/Q==",
"license": "MIT"
},
+ "node_modules/scroll-into-view-if-needed": {
+ "version": "3.1.0",
+ "resolved": "https://registry.npmmirror.com/scroll-into-view-if-needed/-/scroll-into-view-if-needed-3.1.0.tgz",
+ "integrity": "sha512-49oNpRjWRvnU8NyGVmUaYG4jtTkNonFZI86MmGRDqBphEK2EXT9gdEUoQPZhuBM8yWHxCWbobltqYO5M4XrUvQ==",
+ "license": "MIT",
+ "dependencies": {
+ "compute-scroll-into-view": "^3.0.2"
+ }
+ },
"node_modules/semver": {
"version": "6.3.1",
"resolved": "https://registry.npmjs.org/semver/-/semver-6.3.1.tgz",
@@ -2425,6 +4697,18 @@
"semver": "bin/semver.js"
}
},
+ "node_modules/set-cookie-parser": {
+ "version": "2.7.2",
+ "resolved": "https://registry.npmmirror.com/set-cookie-parser/-/set-cookie-parser-2.7.2.tgz",
+ "integrity": "sha512-oeM1lpU/UvhTxw+g3cIfxXHyJRc/uidd3yK1P242gzHds0udQBYzs3y8j4gCCW+ZJ7ad0yctld8RYO+bdurlvw==",
+ "license": "MIT"
+ },
+ "node_modules/shallowequal": {
+ "version": "1.1.0",
+ "resolved": "https://registry.npmmirror.com/shallowequal/-/shallowequal-1.1.0.tgz",
+ "integrity": "sha512-y0m1JoUZSlPAjXVtPPW70aZWfIL/dSP7AFkRnniLCrK/8MDKog3TySTBmckD+RObVxH0v4Tox67+F14PdED2oQ==",
+ "license": "MIT"
+ },
"node_modules/shebang-command": {
"version": "2.0.0",
"resolved": "https://registry.npmjs.org/shebang-command/-/shebang-command-2.0.0.tgz",
@@ -2448,6 +4732,12 @@
"node": ">=8"
}
},
+ "node_modules/size-sensor": {
+ "version": "1.0.3",
+ "resolved": "https://registry.npmmirror.com/size-sensor/-/size-sensor-1.0.3.tgz",
+ "integrity": "sha512-+k9mJ2/rQMiRmQUcjn+qznch260leIXY8r4FyYKKyRBO/s5UoeMAHGkCJyE1R/4wrIhTJONfyloY55SkE7ve3A==",
+ "license": "ISC"
+ },
"node_modules/source-map-js": {
"version": "1.2.1",
"resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
@@ -2458,6 +4748,67 @@
"node": ">=0.10.0"
}
},
+ "node_modules/string-convert": {
+ "version": "0.2.1",
+ "resolved": "https://registry.npmmirror.com/string-convert/-/string-convert-0.2.1.tgz",
+ "integrity": "sha512-u/1tdPl4yQnPBjnVrmdLo9gtuLvELKsAoRapekWggdiQNvvvum+jYF329d84NAa660KQw7pB2n36KrIKVoXa3A==",
+ "license": "MIT"
+ },
+ "node_modules/stylis": {
+ "version": "4.4.0",
+ "resolved": "https://registry.npmmirror.com/stylis/-/stylis-4.4.0.tgz",
+ "integrity": "sha512-5Z9ZpRzfuH6l/UAvCPAPUo3665Nk2wLaZU3x+TLHKVzIz33+sbJqbtrYoC3KD4/uVOr2Zp+L0LySezP9OHV9yA==",
+ "license": "MIT"
+ },
+ "node_modules/swr": {
+ "version": "2.4.1",
+ "resolved": "https://registry.npmmirror.com/swr/-/swr-2.4.1.tgz",
+ "integrity": "sha512-2CC6CiKQtEwaEeNiqWTAw9PGykW8SR5zZX8MZk6TeAvEAnVS7Visz8WzphqgtQ8v2xz/4Q5K+j+SeMaKXeeQIA==",
+ "license": "MIT",
+ "dependencies": {
+ "dequal": "^2.0.3",
+ "use-sync-external-store": "^1.6.0"
+ },
+ "peerDependencies": {
+ "react": "^16.11.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
+ }
+ },
+ "node_modules/tailwindcss": {
+ "version": "4.3.0",
+ "resolved": "https://registry.npmmirror.com/tailwindcss/-/tailwindcss-4.3.0.tgz",
+ "integrity": "sha512-y6nxMGB1nMW9R6k96e5gdIFzcfL/gTJRNaqGes1YvkLnPVXzWgbqFF2yLC0T8G774n24cx3Pe8XrKoniCOAH+Q==",
+ "dev": true,
+ "license": "MIT"
+ },
+ "node_modules/tapable": {
+ "version": "2.3.3",
+ "resolved": "https://registry.npmmirror.com/tapable/-/tapable-2.3.3.tgz",
+ "integrity": "sha512-uxc/zpqFg6x7C8vOE7lh6Lbda8eEL9zmVm/PLeTPBRhh1xCgdWaQ+J1CUieGpIfm2HdtsUpRv+HshiasBMcc6A==",
+ "dev": true,
+ "license": "MIT",
+ "engines": {
+ "node": ">=6"
+ },
+ "funding": {
+ "type": "opencollective",
+ "url": "https://opencollective.com/webpack"
+ }
+ },
+ "node_modules/throttle-debounce": {
+ "version": "5.0.2",
+ "resolved": "https://registry.npmmirror.com/throttle-debounce/-/throttle-debounce-5.0.2.tgz",
+ "integrity": "sha512-B71/4oyj61iNH0KeCamLuE2rmKuTO5byTOSVwECM5FA7TiAiAW+UqTKZ9ERueC4qvgSttUhdmq1mXC3kJqGX7A==",
+ "license": "MIT",
+ "engines": {
+ "node": ">=12.22"
+ }
+ },
+ "node_modules/tinycolor2": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmmirror.com/tinycolor2/-/tinycolor2-1.6.0.tgz",
+ "integrity": "sha512-XPaBkWQJdsf3pLKJV9p4qN/S+fm2Oj8AIPo1BTUhg5oxkvm9+SVEGFdhyOz7tTdUTfvxMiAs4sp6/eZO2Ew+pw==",
+ "license": "MIT"
+ },
"node_modules/tinyglobby": {
"version": "0.2.16",
"resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz",
@@ -2475,6 +4826,12 @@
"url": "https://github.com/sponsors/SuperchupuDev"
}
},
+ "node_modules/toggle-selection": {
+ "version": "1.0.6",
+ "resolved": "https://registry.npmmirror.com/toggle-selection/-/toggle-selection-1.0.6.tgz",
+ "integrity": "sha512-BiZS+C1OS8g/q2RRbJmy59xpyghNBqrr6k5L/uKBGRsTfxmu3ffiRnd8mlGPUVayg8pvfi5urfnu8TU7DVOkLQ==",
+ "license": "MIT"
+ },
"node_modules/ts-api-utils": {
"version": "2.5.0",
"resolved": "https://registry.npmjs.org/ts-api-utils/-/ts-api-utils-2.5.0.tgz",
@@ -2492,9 +4849,7 @@
"version": "2.8.1",
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
"integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==",
- "dev": true,
- "license": "0BSD",
- "optional": true
+ "license": "0BSD"
},
"node_modules/type-check": {
"version": "0.4.0",
@@ -2596,6 +4951,15 @@
"punycode": "^2.1.0"
}
},
+ "node_modules/use-sync-external-store": {
+ "version": "1.6.0",
+ "resolved": "https://registry.npmmirror.com/use-sync-external-store/-/use-sync-external-store-1.6.0.tgz",
+ "integrity": "sha512-Pp6GSwGP/NrPIrxVFAIkOQeyw8lFenOHijQWkUTrDvrF4ALqylP2C/KCkeS9dpUM3KvYRQhna5vt7IL95+ZQ9w==",
+ "license": "MIT",
+ "peerDependencies": {
+ "react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
+ }
+ },
"node_modules/vite": {
"version": "8.0.13",
"resolved": "https://registry.npmjs.org/vite/-/vite-8.0.13.tgz",
@@ -2675,6 +5039,15 @@
}
}
},
+ "node_modules/warning": {
+ "version": "4.0.3",
+ "resolved": "https://registry.npmmirror.com/warning/-/warning-4.0.3.tgz",
+ "integrity": "sha512-rpJyN222KWIvHJ/F53XSZv0Zl/accqHR8et1kpaMTD/fLCRxtV8iX8czMzY7sVZupTI3zcUTg8eycS2kNF9l6w==",
+ "license": "MIT",
+ "dependencies": {
+ "loose-envify": "^1.0.0"
+ }
+ },
"node_modules/which": {
"version": "2.0.2",
"resolved": "https://registry.npmjs.org/which/-/which-2.0.2.tgz",
@@ -2744,6 +5117,21 @@
"peerDependencies": {
"zod": "^3.25.0 || ^4.0.0"
}
+ },
+ "node_modules/zrender": {
+ "version": "6.1.0",
+ "resolved": "https://registry.npmmirror.com/zrender/-/zrender-6.1.0.tgz",
+ "integrity": "sha512-oEGMDB6pOP2S6OwRR4PdVv610zrjnA3Bh+JnSG12fYJlBKjtNAoEb5fSUoCOOINlH96I2fU38/A2UpRKs67xYQ==",
+ "license": "BSD-3-Clause",
+ "dependencies": {
+ "tslib": "2.3.0"
+ }
+ },
+ "node_modules/zrender/node_modules/tslib": {
+ "version": "2.3.0",
+ "resolved": "https://registry.npmmirror.com/tslib/-/tslib-2.3.0.tgz",
+ "integrity": "sha512-N82ooyxVNm6h1riLCoyS9e3fuJ3AMG2zIZs2Gd1ATcSFjSA23Q0fzjjZeh0jbJvWVDZ0cJT8yaNNaaXHzueNjg==",
+ "license": "0BSD"
}
}
}
diff --git a/package.json b/package.json
index 1ebc2b3..b966a6b 100644
--- a/package.json
+++ b/package.json
@@ -10,11 +10,20 @@
"preview": "vite preview"
},
"dependencies": {
+ "@ant-design/icons": "^6.2.3",
+ "@ant-design/pro-components": "^2.8.10",
+ "@tanstack/react-query": "^5.100.11",
+ "antd": "^5.29.3",
+ "dayjs": "^1.11.20",
+ "echarts": "^6.1.0",
+ "echarts-for-react": "^3.0.6",
"react": "^19.2.6",
- "react-dom": "^19.2.6"
+ "react-dom": "^19.2.6",
+ "react-router-dom": "^7.15.1"
},
"devDependencies": {
"@eslint/js": "^10.0.1",
+ "@tailwindcss/vite": "^4.3.0",
"@types/node": "^24.12.3",
"@types/react": "^19.2.14",
"@types/react-dom": "^19.2.3",
@@ -23,6 +32,7 @@
"eslint-plugin-react-hooks": "^7.1.1",
"eslint-plugin-react-refresh": "^0.5.2",
"globals": "^17.6.0",
+ "tailwindcss": "^4.3.0",
"typescript": "~6.0.2",
"typescript-eslint": "^8.59.2",
"vite": "^8.0.12"
diff --git a/src/App.css b/src/App.css
index f1d8c73..b7bff72 100644
--- a/src/App.css
+++ b/src/App.css
@@ -1 +1 @@
-@import "tailwindcss";
+/* App-level styles */
diff --git a/src/App.tsx b/src/App.tsx
index 8e9385e..6d860ac 100644
--- a/src/App.tsx
+++ b/src/App.tsx
@@ -1,10 +1,21 @@
+import { Suspense, lazy } from 'react'
import { BrowserRouter, Routes, Route } from 'react-router-dom'
import { QueryClient, QueryClientProvider } from '@tanstack/react-query'
import { ConfigProvider } from 'antd'
import zhCN from 'antd/locale/zh_CN'
-import MainLayout from './layouts/MainLayout'
-import Dashboard from './pages/Dashboard'
-import './App.css'
+import { AuthProvider } from './contexts/AuthContext'
+import AuthGuard from './components/AuthGuard'
+import PermissionGuard from './components/PermissionGuard'
+import PageLoading from './components/PageLoading'
+import AdminLayout from './layouts/AdminLayout'
+
+const Login = lazy(() => import('./pages/Login'))
+const Dashboard = lazy(() => import('./pages/Dashboard'))
+const UserManagement = lazy(() => import('./pages/UserManagement'))
+const Placeholder = lazy(() => import('./pages/Placeholder'))
+const ForbiddenPage = lazy(() => import('./pages/403'))
+const NotFoundPage = lazy(() => import('./pages/404'))
+const ServerErrorPage = lazy(() => import('./pages/500'))
const queryClient = new QueryClient()
@@ -12,13 +23,74 @@ function App() {
return (
-
-
- }>
- } />
-
-
-
+
+
+ }>
+
+ } />
+ } />
+ } />
+ } />
+
+
+
+ }
+ >
+ } />
+
+
+
+ }
+ />
+
+
+
+ }
+ />
+ } />
+
+
+
+ }
+ />
+ } />
+ } />
+ } />
+ } />
+ } />
+
+
+
+ }
+ />
+
+
+
+ }
+ />
+ } />
+
+
+
+
+
)
diff --git a/src/assets/hero.png b/src/assets/hero.png
deleted file mode 100644
index 02251f4..0000000
Binary files a/src/assets/hero.png and /dev/null differ
diff --git a/src/assets/react.svg b/src/assets/react.svg
deleted file mode 100644
index 6c87de9..0000000
--- a/src/assets/react.svg
+++ /dev/null
@@ -1 +0,0 @@
-
\ No newline at end of file
diff --git a/src/assets/vite.svg b/src/assets/vite.svg
deleted file mode 100644
index 5101b67..0000000
--- a/src/assets/vite.svg
+++ /dev/null
@@ -1 +0,0 @@
-
diff --git a/src/components/AuditLogTable.tsx b/src/components/AuditLogTable.tsx
new file mode 100644
index 0000000..37e8289
--- /dev/null
+++ b/src/components/AuditLogTable.tsx
@@ -0,0 +1,180 @@
+import { useState } from 'react'
+import { ProTable } from '@ant-design/pro-components'
+import type { ProColumns } from '@ant-design/pro-components'
+import { Tag, Tooltip } from 'antd'
+import { EyeOutlined } from '@ant-design/icons'
+import dayjs from 'dayjs'
+import type { AuditLog } from '@/types/admin'
+import DetailDrawer from './DetailDrawer'
+
+interface AuditLogTableProps {
+ dataSource: AuditLog[]
+ loading?: boolean
+ pagination?: { current: number; pageSize: number; total: number }
+ onPageChange?: (page: number, pageSize: number) => void
+ headerTitle?: string
+ toolbarActions?: React.ReactNode[]
+}
+
+const actionLabels: Record = {
+ LOGIN: '登录',
+ LOGOUT: '退出',
+ LOGIN_FAILED: '登录失败',
+ CREATE_ADMIN: '创建管理员',
+ UPDATE_USER_STATUS: '更新用户状态',
+ DELETE_FILE: '删除文件',
+}
+
+const actionColors: Record = {
+ LOGIN: 'green',
+ LOGOUT: 'default',
+ LOGIN_FAILED: 'red',
+ CREATE_ADMIN: 'blue',
+ UPDATE_USER_STATUS: 'orange',
+ DELETE_FILE: 'red',
+}
+
+export default function AuditLogTable({
+ dataSource,
+ loading = false,
+ pagination,
+ onPageChange,
+ headerTitle,
+ toolbarActions,
+}: AuditLogTableProps) {
+ const [detailOpen, setDetailOpen] = useState(false)
+ const [selectedLog, setSelectedLog] = useState(null)
+
+ const columns: ProColumns[] = [
+ {
+ title: '时间',
+ dataIndex: 'createdAt',
+ width: 170,
+ render: (_, record) => (
+
+ {dayjs(record.createdAt).format('MM-DD HH:mm')}
+
+ ),
+ },
+ {
+ title: '操作者',
+ dataIndex: 'adminUserDisplayName',
+ width: 120,
+ ellipsis: true,
+ render: (_, record) => record.adminUserDisplayName || record.adminUserEmail || '-',
+ },
+ {
+ title: '操作',
+ dataIndex: 'action',
+ width: 110,
+ render: (_, record) => {
+ const label = actionLabels[record.action] || record.action
+ return (
+ {label}
+ )
+ },
+ },
+ {
+ title: '资源',
+ dataIndex: 'resourceType',
+ width: 140,
+ ellipsis: true,
+ render: (_, record) => {
+ if (!record.resourceType) return '-'
+ const label = resourceTypeLabels[record.resourceType] || record.resourceType
+ return (
+
+ {label}
+ {record.resourceId && (
+
+ ({record.resourceId.length > 10 ? record.resourceId.slice(0, 10) + '...' : record.resourceId})
+
+ )}
+
+ )
+ },
+ },
+ {
+ title: 'IP',
+ dataIndex: 'ip',
+ width: 140,
+ ellipsis: true,
+ search: false,
+ render: (_, record) => record.ip || '-',
+ },
+ {
+ title: '操作',
+ valueType: 'option',
+ width: 55,
+ render: (_, record) => [
+
+ { setSelectedLog(record); setDetailOpen(true) }}>
+
+
+ ,
+ ],
+ },
+ ]
+
+ return (
+ <>
+
+ columns={columns}
+ dataSource={dataSource}
+ loading={loading}
+ rowKey="id"
+ search={false}
+ options={false}
+ headerTitle={headerTitle}
+ toolbar={{ actions: toolbarActions }}
+ pagination={
+ pagination
+ ? {
+ current: pagination.current,
+ pageSize: pagination.pageSize,
+ total: pagination.total,
+ showSizeChanger: true,
+ showTotal: (total) => `共 ${total} 条`,
+ onChange: onPageChange,
+ }
+ : false
+ }
+ />
+ setDetailOpen(false)}
+ title="审计日志详情"
+ >
+ {selectedLog && (
+
+
+
+
+
+
+
+
+
+
+
+ )}
+
+ >
+ )
+}
+
+const resourceTypeLabels: Record = {
+ AdminUser: '管理员',
+ User: '用户',
+ UploadedFile: '文件',
+ KnowledgeBase: '知识库',
+}
+
+function DetailItem({ label, value }: { label: string; value: string }) {
+ return (
+
+ )
+}
diff --git a/src/components/AuthGuard.tsx b/src/components/AuthGuard.tsx
new file mode 100644
index 0000000..83929a9
--- /dev/null
+++ b/src/components/AuthGuard.tsx
@@ -0,0 +1,23 @@
+import { Navigate, useLocation } from 'react-router-dom'
+import { Spin } from 'antd'
+import { useAuth } from '../contexts/AuthContext'
+
+export default function AuthGuard({ children }: { children: React.ReactNode }) {
+ const { isAuthenticated, isLoading } = useAuth()
+ const location = useLocation()
+
+ if (isLoading) {
+ return (
+
+
+
+ )
+ }
+
+ if (!isAuthenticated) {
+ const redirect = location.pathname === '/login' ? '/' : location.pathname + location.search
+ return
+ }
+
+ return <>{children}>
+}
diff --git a/src/components/ConfirmDangerModal.tsx b/src/components/ConfirmDangerModal.tsx
new file mode 100644
index 0000000..dacd57e
--- /dev/null
+++ b/src/components/ConfirmDangerModal.tsx
@@ -0,0 +1,75 @@
+import { useState } from 'react'
+import { Modal, Input, Typography } from 'antd'
+
+interface ConfirmDangerModalProps {
+ open: boolean
+ onCancel: () => void
+ onConfirm: () => void
+ title: string
+ description?: string
+ targetName: string
+ loading?: boolean
+}
+
+export default function ConfirmDangerModal({
+ open,
+ onCancel,
+ onConfirm,
+ title,
+ description,
+ targetName,
+ loading = false,
+}: ConfirmDangerModalProps) {
+ const [inputValue, setInputValue] = useState('')
+
+ const handleOk = () => {
+ if (inputValue === targetName) {
+ onConfirm()
+ setInputValue('')
+ }
+ }
+
+ const handleCancel = () => {
+ setInputValue('')
+ onCancel()
+ }
+
+ return (
+
+
+ {description || `此操作不可撤销。请输入要删除的对象名称确认:`}
+
+
+ {targetName}
+
+ setInputValue(e.target.value)}
+ />
+
+ )
+}
diff --git a/src/components/DetailDrawer.tsx b/src/components/DetailDrawer.tsx
new file mode 100644
index 0000000..abe52c9
--- /dev/null
+++ b/src/components/DetailDrawer.tsx
@@ -0,0 +1,38 @@
+import { Drawer, Spin } from 'antd'
+import type { ReactNode } from 'react'
+
+interface DetailDrawerProps {
+ open: boolean
+ onClose: () => void
+ title: string
+ loading?: boolean
+ width?: number
+ children: ReactNode
+}
+
+export default function DetailDrawer({
+ open,
+ onClose,
+ title,
+ loading = false,
+ width = 560,
+ children,
+}: DetailDrawerProps) {
+ return (
+
+ {loading ? (
+
+
+
+ ) : (
+ children
+ )}
+
+ )
+}
diff --git a/src/components/EChartsChartContainer.tsx b/src/components/EChartsChartContainer.tsx
new file mode 100644
index 0000000..5631aec
--- /dev/null
+++ b/src/components/EChartsChartContainer.tsx
@@ -0,0 +1,38 @@
+import { Card, Spin, Empty } from 'antd'
+import type { ReactNode } from 'react'
+
+interface EChartsChartContainerProps {
+ title: string
+ loading?: boolean
+ isEmpty?: boolean
+ emptyDescription?: string
+ extra?: ReactNode
+ style?: React.CSSProperties
+ children: ReactNode
+}
+
+export default function EChartsChartContainer({
+ title,
+ loading = false,
+ isEmpty = false,
+ emptyDescription = '暂无数据',
+ extra,
+ style,
+ children,
+}: EChartsChartContainerProps) {
+ return (
+
+ {loading ? (
+
+
+
+ ) : isEmpty ? (
+
+
+
+ ) : (
+ children
+ )}
+
+ )
+}
diff --git a/src/components/EmptyState.tsx b/src/components/EmptyState.tsx
new file mode 100644
index 0000000..b13adff
--- /dev/null
+++ b/src/components/EmptyState.tsx
@@ -0,0 +1,29 @@
+import { Empty, Button } from 'antd'
+import type { ReactNode } from 'react'
+
+interface EmptyStateProps {
+ description?: string
+ action?: {
+ label: string
+ onClick: () => void
+ icon?: ReactNode
+ }
+ image?: ReactNode
+}
+
+export default function EmptyState({ description = '暂无数据', action, image }: EmptyStateProps) {
+ return (
+
+
+ {action && (
+
+ )}
+
+
+ )
+}
diff --git a/src/components/MetricCard.tsx b/src/components/MetricCard.tsx
new file mode 100644
index 0000000..aa46c7e
--- /dev/null
+++ b/src/components/MetricCard.tsx
@@ -0,0 +1,61 @@
+import { Card, Statistic, Spin, Typography } from 'antd'
+import type { ReactNode } from 'react'
+import { ArrowUpOutlined, ArrowDownOutlined } from '@ant-design/icons'
+
+interface MetricCardProps {
+ title: string
+ value?: number | string
+ prefix?: ReactNode
+ suffix?: ReactNode
+ precision?: number
+ loading?: boolean
+ trend?: 'up' | 'down'
+ trendValue?: string
+ trendLabel?: string
+ onClick?: () => void
+}
+
+export default function MetricCard({
+ title,
+ value,
+ prefix,
+ suffix,
+ precision,
+ loading = false,
+ trend,
+ trendValue,
+ trendLabel,
+ onClick,
+}: MetricCardProps) {
+ return (
+
+ {loading ? (
+
+
+
+ ) : (
+
+ )}
+ {!loading && trend && trendValue && (
+
+ {trend === 'up' ? : }
+ {' '}{trendValue}
+ {trendLabel && (
+
+ {trendLabel}
+
+ )}
+
+ )}
+
+ )
+}
diff --git a/src/components/PageLoading.tsx b/src/components/PageLoading.tsx
new file mode 100644
index 0000000..da16e76
--- /dev/null
+++ b/src/components/PageLoading.tsx
@@ -0,0 +1,9 @@
+import { Spin } from 'antd'
+
+export default function PageLoading() {
+ return (
+
+
+
+ )
+}
diff --git a/src/components/PermissionGuard.tsx b/src/components/PermissionGuard.tsx
new file mode 100644
index 0000000..157d81a
--- /dev/null
+++ b/src/components/PermissionGuard.tsx
@@ -0,0 +1,18 @@
+import { Navigate } from 'react-router-dom'
+import { useAuth } from '@/contexts/AuthContext'
+import type { AdminRole } from '@/types/admin'
+
+interface PermissionGuardProps {
+ requiredRole?: AdminRole
+ children: React.ReactNode
+}
+
+export default function PermissionGuard({ requiredRole, children }: PermissionGuardProps) {
+ const { hasPermission } = useAuth()
+
+ if (requiredRole && !hasPermission(requiredRole)) {
+ return
+ }
+
+ return <>{children}>
+}
diff --git a/src/components/StatusTag.tsx b/src/components/StatusTag.tsx
new file mode 100644
index 0000000..0d095d2
--- /dev/null
+++ b/src/components/StatusTag.tsx
@@ -0,0 +1,20 @@
+import { Tag } from 'antd'
+import type { ReactNode } from 'react'
+
+export interface StatusTagProps {
+ status: string
+ statusMap: Record
+ icon?: ReactNode
+}
+
+export default function StatusTag({ status, statusMap, icon }: StatusTagProps) {
+ const config = statusMap[status]
+ if (!config) {
+ return {status}
+ }
+ return (
+
+ {config.label}
+
+ )
+}
diff --git a/src/config/menu.tsx b/src/config/menu.tsx
new file mode 100644
index 0000000..c95db5c
--- /dev/null
+++ b/src/config/menu.tsx
@@ -0,0 +1,65 @@
+import type React from 'react'
+import {
+ DashboardOutlined,
+ UserOutlined,
+ BookOutlined,
+ ImportOutlined,
+ DollarOutlined,
+ SettingOutlined,
+ FileOutlined,
+ CloudOutlined,
+ SafetyOutlined,
+} from '@ant-design/icons'
+import type { AdminRole } from '@/types/admin'
+import { hasRole } from '@/constants/roles'
+
+export interface AdminMenuItem {
+ path: string
+ name: string
+ icon?: React.ReactNode
+ requiredRole?: AdminRole
+ children?: AdminMenuItem[]
+}
+
+export const adminMenuItems: AdminMenuItem[] = [
+ { path: '/', name: '总览', icon: },
+ {
+ path: '/users',
+ name: '用户管理',
+ icon: ,
+ requiredRole: 'ADMIN',
+ children: [
+ { path: '/users/admins', name: '管理员', requiredRole: 'SUPER_ADMIN' },
+ { path: '/users/members', name: '普通用户' },
+ ],
+ },
+ { path: '/membership', name: '会员与额度', icon: , requiredRole: 'ADMIN' },
+ {
+ path: '/knowledge',
+ name: '知识库管理',
+ icon: ,
+ children: [
+ { path: '/knowledge/bases', name: '知识库列表' },
+ { path: '/knowledge/sources', name: '知识源列表' },
+ ],
+ },
+ { path: '/imports', name: '文档导入', icon: },
+ { path: '/ai-costs', name: 'AI 调用与成本', icon: },
+ { path: '/files', name: '文件与 COS', icon: },
+ { path: '/settings', name: '系统配置', icon: , requiredRole: 'ADMIN' },
+ { path: '/audit', name: '审计日志', icon: , requiredRole: 'ADMIN' },
+]
+
+export function filterMenuByRole(items: AdminMenuItem[], role?: AdminRole): AdminMenuItem[] {
+ if (!role) return []
+ return items
+ .filter((item) => !item.requiredRole || hasRole(role, item.requiredRole))
+ .map((item) => ({
+ ...item,
+ children: item.children ? filterMenuByRole(item.children, role) : undefined,
+ }))
+ .filter((item) => {
+ if (item.children && item.children.length === 0) return false
+ return true
+ })
+}
diff --git a/src/constants/roles.ts b/src/constants/roles.ts
new file mode 100644
index 0000000..70f012d
--- /dev/null
+++ b/src/constants/roles.ts
@@ -0,0 +1,30 @@
+import type { AdminRole } from '@/types/admin'
+
+export const ADMIN_ROLE_LABELS: Record = {
+ SUPER_ADMIN: '超级管理员',
+ ADMIN: '管理员',
+ OPERATIONS: '运营人员',
+ DEVELOPER: '开发者',
+ READONLY: '只读用户',
+}
+
+export const ADMIN_ROLE_COLORS: Record = {
+ SUPER_ADMIN: 'red',
+ ADMIN: 'volcano',
+ OPERATIONS: 'orange',
+ DEVELOPER: 'blue',
+ READONLY: 'default',
+}
+
+export const ADMIN_ROLE_HIERARCHY: Record = {
+ SUPER_ADMIN: ['SUPER_ADMIN', 'ADMIN', 'OPERATIONS', 'DEVELOPER', 'READONLY'],
+ ADMIN: ['ADMIN', 'OPERATIONS', 'DEVELOPER', 'READONLY'],
+ OPERATIONS: ['OPERATIONS', 'READONLY'],
+ DEVELOPER: ['DEVELOPER', 'READONLY'],
+ READONLY: ['READONLY'],
+}
+
+export function hasRole(currentRole: AdminRole | undefined, required: AdminRole): boolean {
+ if (!currentRole) return false
+ return ADMIN_ROLE_HIERARCHY[currentRole]?.includes(required) ?? false
+}
diff --git a/src/contexts/AuthContext.tsx b/src/contexts/AuthContext.tsx
new file mode 100644
index 0000000..e2abf8b
--- /dev/null
+++ b/src/contexts/AuthContext.tsx
@@ -0,0 +1,55 @@
+import { createContext, useContext, type ReactNode } from 'react'
+import { useAdminUserQuery, useLoginMutation, useLogoutMutation } from '@/hooks/use-auth-query'
+import { hasRole } from '@/constants/roles'
+import type { AdminUser, AdminRole } from '@/types/admin'
+
+interface AuthState {
+ adminUser: AdminUser | null
+ isAuthenticated: boolean
+ isLoading: boolean
+ login: (email: string, password: string) => Promise
+ logout: () => Promise
+ hasPermission: (requiredRole: AdminRole) => boolean
+}
+
+const AuthContext = createContext(null)
+
+export function AuthProvider({ children }: { children: ReactNode }) {
+ const { data: adminUser, isLoading } = useAdminUserQuery()
+ const loginMutation = useLoginMutation()
+ const logoutMutation = useLogoutMutation()
+
+ const login = async (email: string, password: string) => {
+ await loginMutation.mutateAsync({ email, password })
+ }
+
+ const logout = async () => {
+ await logoutMutation.mutateAsync()
+ }
+
+ const hasPermission = (requiredRole: AdminRole): boolean => {
+ if (!adminUser) return false
+ return hasRole(adminUser.role as AdminRole, requiredRole)
+ }
+
+ return (
+
+ {children}
+
+ )
+}
+
+export function useAuth() {
+ const ctx = useContext(AuthContext)
+ if (!ctx) throw new Error('useAuth must be used within AuthProvider')
+ return ctx
+}
diff --git a/src/hooks/use-auth-query.ts b/src/hooks/use-auth-query.ts
new file mode 100644
index 0000000..f64ef39
--- /dev/null
+++ b/src/hooks/use-auth-query.ts
@@ -0,0 +1,54 @@
+import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'
+import {
+ loginAdmin,
+ logoutAdmin,
+ getCurrentAdmin,
+} from '@/services/admin-api'
+import { getAccessToken, setTokens, clearTokens, setStoredAdminUser } from '@/services/token-store'
+import type { AdminUser } from '@/types/admin'
+
+export function useAdminUserQuery() {
+ return useQuery({
+ queryKey: ['admin', 'me'],
+ queryFn: async () => {
+ const token = getAccessToken()
+ if (!token) return null
+ const user = await getCurrentAdmin()
+ setStoredAdminUser(user)
+ return user
+ },
+ staleTime: 5 * 60 * 1000,
+ retry: false,
+ })
+}
+
+export function useLoginMutation() {
+ const queryClient = useQueryClient()
+ return useMutation({
+ mutationFn: ({ email, password }: { email: string; password: string }) =>
+ loginAdmin(email, password),
+ onSuccess: (data) => {
+ setTokens(data.accessToken, data.refreshToken)
+ setStoredAdminUser(data.adminUser)
+ queryClient.setQueryData(['admin', 'me'], data.adminUser)
+ },
+ })
+}
+
+export function useLogoutMutation() {
+ const queryClient = useQueryClient()
+ return useMutation({
+ mutationFn: () => {
+ const token = localStorage.getItem('admin_refresh_token')
+ if (token) {
+ return logoutAdmin(token).catch(() => {})
+ }
+ return Promise.resolve()
+ },
+ onSettled: () => {
+ clearTokens()
+ queryClient.setQueryData(['admin', 'me'], null)
+ queryClient.clear()
+ },
+ })
+}
diff --git a/src/index.css b/src/index.css
index 5fb3313..340e51c 100644
--- a/src/index.css
+++ b/src/index.css
@@ -1,111 +1,9 @@
-:root {
- --text: #6b6375;
- --text-h: #08060d;
- --bg: #fff;
- --border: #e5e4e7;
- --code-bg: #f4f3ec;
- --accent: #aa3bff;
- --accent-bg: rgba(170, 59, 255, 0.1);
- --accent-border: rgba(170, 59, 255, 0.5);
- --social-bg: rgba(244, 243, 236, 0.5);
- --shadow:
- rgba(0, 0, 0, 0.1) 0 10px 15px -3px, rgba(0, 0, 0, 0.05) 0 4px 6px -2px;
-
- --sans: system-ui, 'Segoe UI', Roboto, sans-serif;
- --heading: system-ui, 'Segoe UI', Roboto, sans-serif;
- --mono: ui-monospace, Consolas, monospace;
-
- font: 18px/145% var(--sans);
- letter-spacing: 0.18px;
- color-scheme: light dark;
- color: var(--text);
- background: var(--bg);
- font-synthesis: none;
- text-rendering: optimizeLegibility;
+body {
+ margin: 0;
-webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale;
-
- @media (max-width: 1024px) {
- font-size: 16px;
- }
-}
-
-@media (prefers-color-scheme: dark) {
- :root {
- --text: #9ca3af;
- --text-h: #f3f4f6;
- --bg: #16171d;
- --border: #2e303a;
- --code-bg: #1f2028;
- --accent: #c084fc;
- --accent-bg: rgba(192, 132, 252, 0.15);
- --accent-border: rgba(192, 132, 252, 0.5);
- --social-bg: rgba(47, 48, 58, 0.5);
- --shadow:
- rgba(0, 0, 0, 0.4) 0 10px 15px -3px, rgba(0, 0, 0, 0.25) 0 4px 6px -2px;
- }
-
- #social .button-icon {
- filter: invert(1) brightness(2);
- }
}
#root {
- width: 1126px;
- max-width: 100%;
- margin: 0 auto;
- text-align: center;
- border-inline: 1px solid var(--border);
- min-height: 100svh;
- display: flex;
- flex-direction: column;
- box-sizing: border-box;
-}
-
-body {
- margin: 0;
-}
-
-h1,
-h2 {
- font-family: var(--heading);
- font-weight: 500;
- color: var(--text-h);
-}
-
-h1 {
- font-size: 56px;
- letter-spacing: -1.68px;
- margin: 32px 0;
- @media (max-width: 1024px) {
- font-size: 36px;
- margin: 20px 0;
- }
-}
-h2 {
- font-size: 24px;
- line-height: 118%;
- letter-spacing: -0.24px;
- margin: 0 0 8px;
- @media (max-width: 1024px) {
- font-size: 20px;
- }
-}
-p {
- margin: 0;
-}
-
-code,
-.counter {
- font-family: var(--mono);
- display: inline-flex;
- border-radius: 4px;
- color: var(--text-h);
-}
-
-code {
- font-size: 15px;
- line-height: 135%;
- padding: 4px 8px;
- background: var(--code-bg);
+ min-height: 100vh;
}
diff --git a/src/layouts/AdminLayout.tsx b/src/layouts/AdminLayout.tsx
new file mode 100644
index 0000000..25d37b7
--- /dev/null
+++ b/src/layouts/AdminLayout.tsx
@@ -0,0 +1,111 @@
+import { Outlet, useNavigate, useLocation } from 'react-router-dom'
+import { ProLayout } from '@ant-design/pro-components'
+import { Dropdown, Avatar, Tag, Space, message } from 'antd'
+import { LogoutOutlined, UserOutlined } from '@ant-design/icons'
+import { useAuth } from '@/contexts/AuthContext'
+import { filterMenuByRole, adminMenuItems } from '@/config/menu'
+import { ADMIN_ROLE_LABELS, ADMIN_ROLE_COLORS } from '@/constants/roles'
+import type { AdminRole } from '@/types/admin'
+
+const breadcrumbMap: Record = {
+ '/': '总览',
+ '/users': '用户管理',
+ '/users/admins': '管理员',
+ '/users/members': '普通用户',
+ '/membership': '会员与额度',
+ '/knowledge': '知识库管理',
+ '/knowledge/bases': '知识库列表',
+ '/knowledge/sources': '知识源列表',
+ '/imports': '文档导入',
+ '/ai-costs': 'AI 调用与成本',
+ '/files': '文件与 COS',
+ '/settings': '系统配置',
+ '/audit': '审计日志',
+}
+
+export default function AdminLayout() {
+ const navigate = useNavigate()
+ const location = useLocation()
+ const { adminUser, logout, hasPermission } = useAuth()
+
+ const currentRole = adminUser?.role as AdminRole | undefined
+
+ const handleLogout = async () => {
+ await logout()
+ message.success('已退出登录')
+ navigate('/login', { replace: true })
+ }
+
+ const userMenuItems = [
+ {
+ key: 'info',
+ label: (
+
+
{adminUser?.displayName}
+
{adminUser?.email}
+
+
+ {ADMIN_ROLE_LABELS[currentRole ?? 'READONLY']}
+
+
+
+ ),
+ disabled: true,
+ },
+ { type: 'divider' as const },
+ {
+ key: 'logout',
+ icon: ,
+ label: '退出登录',
+ onClick: handleLogout,
+ },
+ ]
+
+ const isDev = import.meta.env.DEV || import.meta.env.MODE !== 'production'
+
+ return (
+ {
+ const items = filterMenuByRole(adminMenuItems, currentRole)
+ return items
+ }}
+ menuItemRender={(item, dom) => (
+ item.path && navigate(item.path)}>{dom}
+ )}
+ breadcrumbRender={(routers) => {
+ return (routers ?? []).map((r) => {
+ const path = (r as any).path ?? ''
+ const label = breadcrumbMap[path] || (r as any).breadcrumbName || path
+ return { ...r, breadcrumbName: label, title: label } as any
+ })
+ }}
+ rightContentRender={() =>
+ hasPermission('READONLY') ? (
+
+ {isDev && (
+
+ DEV
+
+ )}
+
+
+
} />
+
{adminUser?.displayName || '管理员'}
+
+
+
+ ) : null
+ }
+ siderWidth={220}
+ token={{
+ header: { heightLayoutHeader: 48 },
+ pageContainer: { paddingBlockPageContainerContent: 24, paddingInlinePageContainerContent: 24 },
+ }}
+ >
+
+
+ )
+}
diff --git a/src/layouts/MainLayout.tsx b/src/layouts/MainLayout.tsx
deleted file mode 100644
index b3139af..0000000
--- a/src/layouts/MainLayout.tsx
+++ /dev/null
@@ -1,60 +0,0 @@
-import { Outlet, useNavigate, useLocation } from 'react-router-dom'
-import { ProLayout } from '@ant-design/pro-components'
-import {
- DashboardOutlined,
- UserOutlined,
- BookOutlined,
- ImportOutlined,
- DollarOutlined,
- SettingOutlined,
- FileOutlined,
- CloudOutlined,
- SafetyOutlined,
-} from '@ant-design/icons'
-
-const menuData = [
- { path: '/', name: '总览', icon: },
- { path: '/users', name: '用户管理', icon: },
- { path: '/membership', name: '会员与额度', icon: },
- {
- path: '/knowledge',
- name: '知识库管理',
- icon: ,
- children: [
- { path: '/knowledge/bases', name: '知识库列表' },
- { path: '/knowledge/sources', name: '知识源列表' },
- ],
- },
- {
- path: '/imports',
- name: '文档导入',
- icon: ,
- },
- {
- path: '/ai-costs',
- name: 'AI 调用与成本',
- icon: ,
- },
- { path: '/files', name: '文件与 COS', icon: },
- { path: '/settings', name: '系统配置', icon: },
- { path: '/audit', name: '审计日志', icon: },
-]
-
-export default function MainLayout() {
- const navigate = useNavigate()
- const location = useLocation()
-
- return (
- menuData}
- menuItemRender={(item, dom) => (
- item.path && navigate(item.path)}>{dom}
- )}
- >
-
-
- )
-}
diff --git a/src/pages/403.tsx b/src/pages/403.tsx
new file mode 100644
index 0000000..2615057
--- /dev/null
+++ b/src/pages/403.tsx
@@ -0,0 +1,20 @@
+import { Result, Button } from 'antd'
+import { useNavigate } from 'react-router-dom'
+
+export default function ForbiddenPage() {
+ const navigate = useNavigate()
+ return (
+
+ navigate('/', { replace: true })}>
+ 返回首页
+
+ }
+ />
+
+ )
+}
diff --git a/src/pages/404.tsx b/src/pages/404.tsx
new file mode 100644
index 0000000..2d54cd7
--- /dev/null
+++ b/src/pages/404.tsx
@@ -0,0 +1,20 @@
+import { Result, Button } from 'antd'
+import { useNavigate } from 'react-router-dom'
+
+export default function NotFoundPage() {
+ const navigate = useNavigate()
+ return (
+
+ navigate('/', { replace: true })}>
+ 返回首页
+
+ }
+ />
+
+ )
+}
diff --git a/src/pages/500.tsx b/src/pages/500.tsx
new file mode 100644
index 0000000..db69e5f
--- /dev/null
+++ b/src/pages/500.tsx
@@ -0,0 +1,18 @@
+import { Result, Button } from 'antd'
+
+export default function ServerErrorPage() {
+ return (
+
+ window.location.reload()}>
+ 刷新页面
+
+ }
+ />
+
+ )
+}
diff --git a/src/pages/Dashboard.tsx b/src/pages/Dashboard.tsx
index ca581d6..85d3246 100644
--- a/src/pages/Dashboard.tsx
+++ b/src/pages/Dashboard.tsx
@@ -1,31 +1,176 @@
-import { Card, Row, Col, Statistic } from 'antd'
-import { UserOutlined, BookOutlined, CloudOutlined } from '@ant-design/icons'
+import { useMemo } from 'react'
+import { Row, Col, Typography } from 'antd'
+import { useQuery } from '@tanstack/react-query'
+import ReactEChartsCore from 'echarts-for-react/esm/core'
+import * as echarts from 'echarts/core'
+import { LineChart, BarChart } from 'echarts/charts'
+import { GridComponent, TooltipComponent, TitleComponent, LegendComponent } from 'echarts/components'
+import { CanvasRenderer } from 'echarts/renderers'
+import {
+ UserOutlined,
+ BookOutlined,
+ CloudOutlined,
+ FileOutlined,
+} from '@ant-design/icons'
+import dayjs from 'dayjs'
+import MetricCard from '@/components/MetricCard'
+import EChartsChartContainer from '@/components/EChartsChartContainer'
+import AuditLogTable from '@/components/AuditLogTable'
+import { getDashboardStats, getAuditLogs } from '@/services/admin-api'
+
+echarts.use([LineChart, BarChart, GridComponent, TooltipComponent, TitleComponent, LegendComponent, CanvasRenderer])
+
+function formatStorage(bytes: number): string {
+ if (bytes >= 1073741824) return (bytes / 1073741824).toFixed(1) + ' GB'
+ if (bytes >= 1048576) return (bytes / 1048576).toFixed(1) + ' MB'
+ return (bytes / 1024).toFixed(1) + ' KB'
+}
export default function Dashboard() {
+ const { data: stats, isLoading: statsLoading } = useQuery({
+ queryKey: ['dashboard', 'stats'],
+ queryFn: getDashboardStats,
+ staleTime: 60_000,
+ })
+
+ const { data: auditData, isLoading: auditLoading } = useQuery({
+ queryKey: ['dashboard', 'audit-logs'],
+ queryFn: () => getAuditLogs({ page: 1, limit: 10 }),
+ staleTime: 30_000,
+ })
+
+ const userTrendOption = useMemo(() => ({
+ grid: { top: 20, right: 20, bottom: 20, left: 40 },
+ tooltip: { trigger: 'axis' as const },
+ xAxis: {
+ type: 'category' as const,
+ data: stats?.userTrend.map((p) => dayjs(p.date).format('MM-DD')) || [],
+ axisLabel: { fontSize: 11 },
+ },
+ yAxis: { type: 'value' as const, axisLabel: { fontSize: 11 } },
+ series: [{
+ name: '日活用户',
+ type: 'line',
+ data: stats?.userTrend.map((p) => p.value) || [],
+ smooth: true,
+ symbol: 'none',
+ lineStyle: { color: '#1677ff', width: 2 },
+ areaStyle: { color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
+ { offset: 0, color: 'rgba(22,119,255,0.15)' },
+ { offset: 1, color: 'rgba(22,119,255,0)' },
+ ])},
+ }],
+ }), [stats])
+
+ const aiCallTrendOption = useMemo(() => ({
+ grid: { top: 20, right: 20, bottom: 20, left: 40 },
+ tooltip: { trigger: 'axis' as const },
+ xAxis: {
+ type: 'category' as const,
+ data: stats?.aiCallTrend.map((p) => dayjs(p.date).format('MM-DD')) || [],
+ axisLabel: { fontSize: 11 },
+ },
+ yAxis: { type: 'value' as const, axisLabel: { fontSize: 11 } },
+ series: [{
+ name: 'AI 调用',
+ type: 'bar',
+ data: stats?.aiCallTrend.map((p) => p.value) || [],
+ itemStyle: {
+ color: new echarts.graphic.LinearGradient(0, 0, 0, 1, [
+ { offset: 0, color: '#52c41a' },
+ { offset: 1, color: '#b7eb8f' },
+ ]),
+ borderRadius: [4, 4, 0, 0],
+ },
+ }],
+ }), [stats])
+
return (
-
+
+
数据概览
+
-
-
- } />
-
+
+ }
+ trend="up"
+ trendValue={`+${stats?.newUsersToday ?? 0}`}
+ trendLabel="今日新增"
+ />
-
-
- } />
-
+
+ }
+ trend="up"
+ trendValue={`+${stats?.newKbsToday ?? 0}`}
+ trendLabel="今日新增"
+ />
-
-
- } />
-
+
+ }
+ />
-
-
- } />
-
+
+ }
+ suffix={`${stats?.totalFiles ?? 0} 个文件`}
+ />
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
)
}
diff --git a/src/pages/Login.tsx b/src/pages/Login.tsx
new file mode 100644
index 0000000..7c89559
--- /dev/null
+++ b/src/pages/Login.tsx
@@ -0,0 +1,207 @@
+import { useState } from 'react'
+import { useNavigate, useSearchParams } from 'react-router-dom'
+import { Form, Input, Button, Typography, Alert, message, theme } from 'antd'
+import { MailOutlined, LockOutlined, SafetyCertificateOutlined } from '@ant-design/icons'
+import { useAuth } from '@/contexts/AuthContext'
+
+const { Title, Text, Paragraph } = Typography
+
+export default function Login() {
+ const [loading, setLoading] = useState(false)
+ const [error, setError] = useState
(null)
+ const { login } = useAuth()
+ const navigate = useNavigate()
+ const [searchParams] = useSearchParams()
+ const { token } = theme.useToken()
+
+ const onFinish = async (values: { email: string; password: string }) => {
+ setLoading(true)
+ setError(null)
+ try {
+ await login(values.email, values.password)
+ message.success('登录成功')
+ const redirect = searchParams.get('redirect') || '/'
+ navigate(redirect, { replace: true })
+ } catch (err) {
+ setError(err instanceof Error ? err.message : '登录失败,请重试')
+ } finally {
+ setLoading(false)
+ }
+ }
+
+ return (
+
+ {/* Left brand panel */}
+
+ {/* Decorative circles */}
+
+
+
+
+
+
+
+
+ 知习 Admin
+
+
+ 企业知识管理中台
+
+
+
+
+
+ 内部管理系统 · 授权访问
+
+
+
+
+
+ {/* Right form panel */}
+
+
+
+
+ 登录后台
+
+
+ 使用管理员邮箱和密码登录管理控制台
+
+
+
+ {error && (
+
setError(null)}
+ style={{ marginBottom: 24 }}
+ />
+ )}
+
+
+ }
+ placeholder="管理员邮箱"
+ style={{ height: 48, borderRadius: 8 }}
+ />
+
+
+
+ }
+ placeholder="密码"
+ style={{ height: 48, borderRadius: 8 }}
+ />
+
+
+
+
+
+
+
+
+
+ 知习知识平台 · 内部管理系统
+
+
+
+
+
+ )
+}
diff --git a/src/pages/Placeholder.tsx b/src/pages/Placeholder.tsx
new file mode 100644
index 0000000..7978fb8
--- /dev/null
+++ b/src/pages/Placeholder.tsx
@@ -0,0 +1,5 @@
+import { Result } from 'antd'
+
+export default function Placeholder({ title }: { title: string }) {
+ return
+}
diff --git a/src/pages/UserManagement.tsx b/src/pages/UserManagement.tsx
new file mode 100644
index 0000000..a6c53e2
--- /dev/null
+++ b/src/pages/UserManagement.tsx
@@ -0,0 +1,241 @@
+import { useRef, useState } from 'react'
+import { ProTable } from '@ant-design/pro-components'
+import type { ProColumns, ActionType } from '@ant-design/pro-components'
+import { Button, Tag, message, Tooltip } from 'antd'
+import { PlusOutlined, ReloadOutlined } from '@ant-design/icons'
+import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'
+import type { AdminUser } from '@/types/admin'
+import { ADMIN_ROLE_LABELS, ADMIN_ROLE_COLORS } from '@/constants/roles'
+import { getAdminUsers, deleteAdminUser } from '@/services/admin-api'
+import DetailDrawer from '@/components/DetailDrawer'
+import ConfirmDangerModal from '@/components/ConfirmDangerModal'
+
+const statusLabelMap: Record = {
+ ACTIVE: '正常',
+ DISABLED: '已禁用',
+}
+
+const statusColorMap: Record = {
+ ACTIVE: 'green',
+ DISABLED: 'red',
+}
+
+export default function UserManagement() {
+ const actionRef = useRef(undefined)
+ const queryClient = useQueryClient()
+ const [detailOpen, setDetailOpen] = useState(false)
+ const [selectedUser, setSelectedUser] = useState(null)
+ const [deleteTarget, setDeleteTarget] = useState(null)
+
+ const { data, isLoading } = useQuery({
+ queryKey: ['admin-users'],
+ queryFn: () => getAdminUsers({ page: 1, limit: 20 }),
+ })
+
+ const deleteMutation = useMutation({
+ mutationFn: (id: string) => deleteAdminUser(id),
+ onSuccess: () => {
+ message.success('已删除')
+ queryClient.invalidateQueries({ queryKey: ['admin-users'] })
+ },
+ onError: () => message.error('删除失败'),
+ })
+
+ const columns: ProColumns[] = [
+ {
+ title: '姓名',
+ dataIndex: 'displayName',
+ width: 130,
+ ellipsis: true,
+ },
+ {
+ title: '邮箱',
+ dataIndex: 'email',
+ width: 200,
+ ellipsis: true,
+ },
+ {
+ title: '角色',
+ dataIndex: 'role',
+ width: 120,
+ valueType: 'select',
+ valueEnum: {
+ SUPER_ADMIN: { text: '超级管理员' },
+ ADMIN: { text: '管理员' },
+ OPERATIONS: { text: '运营人员' },
+ DEVELOPER: { text: '开发者' },
+ READONLY: { text: '只读用户' },
+ },
+ render: (_, record) => (
+ {ADMIN_ROLE_LABELS[record.role]}
+ ),
+ },
+ {
+ title: '状态',
+ dataIndex: 'status',
+ width: 90,
+ valueType: 'select',
+ valueEnum: {
+ ACTIVE: { text: '正常' },
+ DISABLED: { text: '已禁用' },
+ },
+ render: (_, record) => (
+ {statusLabelMap[record.status]}
+ ),
+ },
+ {
+ title: '最后登录',
+ dataIndex: 'lastLoginAt',
+ width: 160,
+ search: false,
+ render: (_, record) =>
+ record.lastLoginAt
+ ? new Date(record.lastLoginAt).toLocaleString('zh-CN', {
+ month: '2-digit',
+ day: '2-digit',
+ hour: '2-digit',
+ minute: '2-digit',
+ })
+ : '-',
+ },
+ {
+ title: '创建时间',
+ dataIndex: 'createdAt',
+ width: 160,
+ search: false,
+ render: (_, record) =>
+ new Date(record.createdAt).toLocaleString('zh-CN', {
+ month: '2-digit',
+ day: '2-digit',
+ hour: '2-digit',
+ minute: '2-digit',
+ }),
+ },
+ {
+ title: '操作',
+ valueType: 'option',
+ width: 100,
+ render: (_, record) => [
+ {
+ setSelectedUser(record)
+ setDetailOpen(true)
+ }}
+ >
+ 详情
+ ,
+
+ {
+ if (record.role !== 'SUPER_ADMIN') {
+ setDeleteTarget(record)
+ }
+ }}
+ >
+ 删除
+
+ ,
+ ],
+ },
+ ]
+
+ return (
+ <>
+
+ actionRef={actionRef}
+ columns={columns}
+ dataSource={data?.items || []}
+ loading={isLoading}
+ rowKey="id"
+ search={false}
+ options={false}
+ pagination={
+ data
+ ? {
+ current: data.page,
+ pageSize: data.limit,
+ total: data.total,
+ showSizeChanger: true,
+ showTotal: (total) => `共 ${total} 条`,
+ }
+ : false
+ }
+ headerTitle="管理员列表"
+ toolbar={{
+ actions: [
+ }>
+ 新建管理员
+ ,
+ }
+ onClick={() => queryClient.invalidateQueries({ queryKey: ['admin-users'] })}
+ >
+ 刷新
+ ,
+ ],
+ }}
+ />
+
+ setDetailOpen(false)}
+ title="管理员详情"
+ >
+ {selectedUser && (
+
+
+
姓名
+
{selectedUser.displayName}
+
+
+
邮箱
+
{selectedUser.email}
+
+
+
角色
+
{ADMIN_ROLE_LABELS[selectedUser.role]}
+
+
+
状态
+
{statusLabelMap[selectedUser.status]}
+
+
+
两步验证
+
{selectedUser.twoFactorEnabled ? '已启用' : '未启用'}
+
+
+
最后登录时间
+
{selectedUser.lastLoginAt || '-'}
+
+
+
最后登录 IP
+
{selectedUser.lastLoginIp || '-'}
+
+
+
创建时间
+
{selectedUser.createdAt}
+
+
+ )}
+
+
+ setDeleteTarget(null)}
+ onConfirm={() => {
+ if (deleteTarget) {
+ deleteMutation.mutate(deleteTarget.id)
+ setDeleteTarget(null)
+ }
+ }}
+ title="删除管理员"
+ description="此操作不可撤销。请输入管理员邮箱确认删除:"
+ targetName={deleteTarget?.email || ''}
+ loading={deleteMutation.isPending}
+ />
+ >
+ )
+}
diff --git a/src/routes/index.tsx b/src/routes/index.tsx
new file mode 100644
index 0000000..d1f9f2d
--- /dev/null
+++ b/src/routes/index.tsx
@@ -0,0 +1,27 @@
+import { lazy } from 'react'
+import type { AdminRole } from '@/types/admin'
+
+const Dashboard = lazy(() => import('@/pages/Dashboard'))
+const UserManagement = lazy(() => import('@/pages/UserManagement'))
+
+export interface RouteConfig {
+ path: string
+ title: string
+ element: React.LazyExoticComponent>
+ requiredRole?: AdminRole
+}
+
+export const routeConfig: RouteConfig[] = [
+ { path: '/', title: '总览', element: Dashboard },
+ { path: '/users', title: '用户管理', element: UserManagement, requiredRole: 'ADMIN' },
+ { path: '/users/admins', title: '管理员', element: UserManagement, requiredRole: 'SUPER_ADMIN' },
+ { path: '/users/members', title: '普通用户', element: UserManagement },
+ { path: '/membership', title: '会员与额度', element: UserManagement, requiredRole: 'ADMIN' },
+ { path: '/knowledge/bases', title: '知识库列表', element: UserManagement },
+ { path: '/knowledge/sources', title: '知识源列表', element: UserManagement },
+ { path: '/imports', title: '文档导入', element: UserManagement },
+ { path: '/ai-costs', title: 'AI 调用与成本', element: UserManagement },
+ { path: '/files', title: '文件与 COS', element: UserManagement },
+ { path: '/settings', title: '系统配置', element: UserManagement, requiredRole: 'ADMIN' },
+ { path: '/audit', title: '审计日志', element: UserManagement, requiredRole: 'ADMIN' },
+]
diff --git a/src/services/admin-api.ts b/src/services/admin-api.ts
new file mode 100644
index 0000000..6d6008a
--- /dev/null
+++ b/src/services/admin-api.ts
@@ -0,0 +1,136 @@
+import type {
+ AdminUser,
+ DashboardStats,
+ AuditLog,
+} from '@/types/admin'
+import type { PaginatedResult, PaginationParams } from '@/types/api'
+import { api } from './http-client'
+import { MOCK_DASHBOARD_STATS, MOCK_AUDIT_LOGS } from './mock-data'
+
+// ── Auth ──────────────────────────────────────────────
+
+interface LoginResponse {
+ accessToken: string
+ refreshToken: string
+ adminUser: AdminUser
+}
+
+export function loginAdmin(email: string, password: string): Promise {
+ return api.post('/admin-api/auth/login', { email, password })
+}
+
+interface TokenPair {
+ accessToken: string
+ refreshToken: string
+ adminUser: AdminUser
+}
+
+export function refreshAdminToken(refreshToken: string): Promise {
+ return api.post('/admin-api/auth/refresh', { refreshToken })
+}
+
+export function logoutAdmin(refreshToken: string): Promise {
+ return api.post('/admin-api/auth/logout', { refreshToken })
+}
+
+export function getCurrentAdmin(): Promise {
+ return api.get('/admin-api/auth/me')
+}
+
+// ── Dashboard ─────────────────────────────────────────
+
+export async function getDashboardStats(): Promise {
+ try {
+ return await api.get('/admin-api/dashboard/stats')
+ } catch {
+ if (import.meta.env.DEV) return MOCK_DASHBOARD_STATS
+ throw new Error('获取仪表盘数据失败')
+ }
+}
+
+// ── Admin Users ───────────────────────────────────────
+
+export interface AdminUsersQuery extends PaginationParams {
+ search?: string
+ role?: string
+ status?: string
+}
+
+export async function getAdminUsers(
+ params: AdminUsersQuery,
+): Promise> {
+ try {
+ return await api.get>(
+ `/admin-api/admin-users?${new URLSearchParams(params as Record).toString()}`,
+ )
+ } catch {
+ if (import.meta.env.DEV) {
+ return {
+ items: [],
+ total: 0,
+ page: params.page ?? 1,
+ limit: params.limit ?? 20,
+ totalPages: 0,
+ }
+ }
+ throw new Error('获取管理员列表失败')
+ }
+}
+
+export function getAdminUserById(id: string): Promise {
+ return api.get(`/admin-api/admin-users/${id}`)
+}
+
+export function createAdminUser(data: {
+ email: string
+ password: string
+ displayName: string
+ role: string
+}): Promise {
+ return api.post('/admin-api/admin-users', data)
+}
+
+export function updateAdminUser(
+ id: string,
+ data: { role?: string; status?: string; displayName?: string },
+): Promise {
+ return api.put(`/admin-api/admin-users/${id}`, data)
+}
+
+export function deleteAdminUser(id: string): Promise {
+ return api.delete(`/admin-api/admin-users/${id}`)
+}
+
+// ── Audit Logs ────────────────────────────────────────
+
+export interface AuditLogsQuery extends PaginationParams {
+ adminUserId?: string
+ action?: string
+ startDate?: string
+ endDate?: string
+}
+
+export async function getAuditLogs(
+ params: AuditLogsQuery,
+): Promise> {
+ try {
+ return await api.get>(
+ `/admin-api/audit-logs?${new URLSearchParams(params as Record).toString()}`,
+ )
+ } catch {
+ if (import.meta.env.DEV) {
+ return {
+ items: MOCK_AUDIT_LOGS,
+ total: MOCK_AUDIT_LOGS.length,
+ page: params.page ?? 1,
+ limit: params.limit ?? 20,
+ totalPages: 1,
+ }
+ }
+ throw new Error('获取审计日志失败')
+ }
+}
+
+export function getAuditLogById(id: string): Promise {
+ return api.get(`/admin-api/audit-logs/${id}`)
+}
diff --git a/src/services/api.ts b/src/services/api.ts
new file mode 100644
index 0000000..4a8cc25
--- /dev/null
+++ b/src/services/api.ts
@@ -0,0 +1,93 @@
+const BASE_URL = ''
+
+interface ApiResponse {
+ success: boolean
+ data: T
+ message?: string
+ statusCode?: number
+}
+
+function getToken(): string | null {
+ return localStorage.getItem('admin_access_token')
+}
+
+async function request(
+ path: string,
+ options: RequestInit = {},
+): Promise {
+ const token = getToken()
+ const headers: Record = {
+ 'Content-Type': 'application/json',
+ ...(options.headers as Record),
+ }
+ if (token) {
+ headers['Authorization'] = `Bearer ${token}`
+ }
+
+ const res = await fetch(`${BASE_URL}${path}`, { ...options, headers })
+
+ if (res.status === 401) {
+ const refreshed = await tryRefresh()
+ if (refreshed) {
+ headers['Authorization'] = `Bearer ${getToken()}`
+ const retryRes = await fetch(`${BASE_URL}${path}`, { ...options, headers })
+ return retryRes.json()
+ }
+ localStorage.removeItem('admin_access_token')
+ localStorage.removeItem('admin_refresh_token')
+ localStorage.removeItem('admin_user')
+ window.location.href = '/login'
+ throw new Error('登录已过期')
+ }
+
+ const json: ApiResponse = await res.json()
+ if (!json.success) {
+ throw new Error(json.message || '请求失败')
+ }
+ return json.data
+}
+
+async function tryRefresh(): Promise {
+ const refreshToken = localStorage.getItem('admin_refresh_token')
+ if (!refreshToken) return false
+
+ try {
+ const res = await fetch(`${BASE_URL}/admin-api/auth/refresh`, {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ refreshToken }),
+ })
+ if (!res.ok) return false
+ const json: ApiResponse<{ accessToken: string; refreshToken: string }> = await res.json()
+ if (json.success) {
+ localStorage.setItem('admin_access_token', json.data.accessToken)
+ localStorage.setItem('admin_refresh_token', json.data.refreshToken)
+ return true
+ }
+ } catch {}
+ return false
+}
+
+export const api = {
+ get(path: string) {
+ return request(path)
+ },
+ post(path: string, body?: unknown) {
+ return request(path, {
+ method: 'POST',
+ body: body ? JSON.stringify(body) : undefined,
+ })
+ },
+ put(path: string, body?: unknown) {
+ return request(path, {
+ method: 'PUT',
+ body: body ? JSON.stringify(body) : undefined,
+ })
+ },
+ delete(path: string, body?: unknown) {
+ return request(path, {
+ method: 'DELETE',
+ body: body ? JSON.stringify(body) : undefined,
+ })
+ },
+}
diff --git a/src/services/http-client.ts b/src/services/http-client.ts
new file mode 100644
index 0000000..263ee09
--- /dev/null
+++ b/src/services/http-client.ts
@@ -0,0 +1,113 @@
+import {
+ getAccessToken,
+ getRefreshToken,
+ setTokens,
+ clearTokens,
+} from './token-store'
+
+interface ApiResponse {
+ success: boolean
+ data: T
+ message?: string
+ statusCode?: number
+}
+
+export class ApiError extends Error {
+ httpStatus: number
+ code: number
+ constructor(message: string, httpStatus: number, code?: number) {
+ super(message)
+ this.name = 'ApiError'
+ this.httpStatus = httpStatus
+ this.code = code ?? httpStatus
+ }
+}
+
+let refreshPromise: Promise | null = null
+
+async function tryRefresh(): Promise {
+ const token = getRefreshToken()
+ if (!token) return false
+
+ if (!refreshPromise) {
+ refreshPromise = (async () => {
+ try {
+ const res = await fetch('/admin-api/auth/refresh', {
+ method: 'POST',
+ headers: { 'Content-Type': 'application/json' },
+ body: JSON.stringify({ refreshToken: token }),
+ })
+ if (!res.ok) return false
+ const json: ApiResponse<{ accessToken: string; refreshToken: string }> = await res.json()
+ if (json.success) {
+ setTokens(json.data.accessToken, json.data.refreshToken)
+ return true
+ }
+ return false
+ } catch {
+ return false
+ } finally {
+ refreshPromise = null
+ }
+ })()
+ }
+
+ return refreshPromise
+}
+
+async function request(
+ path: string,
+ options: RequestInit = {},
+ retried = false,
+): Promise {
+ const token = getAccessToken()
+ const headers: Record = {
+ 'Content-Type': 'application/json',
+ ...(options.headers as Record),
+ }
+ if (token) {
+ headers['Authorization'] = `Bearer ${token}`
+ }
+
+ const res = await fetch(path, { ...options, headers })
+
+ if (res.status === 401 && !retried) {
+ const refreshed = await tryRefresh()
+ if (refreshed) {
+ return request(path, options, true)
+ }
+ clearTokens()
+ window.location.href = '/login'
+ throw new ApiError('登录已过期', 401)
+ }
+
+ const json: ApiResponse = await res.json()
+ if (!json.success) {
+ throw new ApiError(json.message || '请求失败', res.status, json.statusCode)
+ }
+ return json.data
+}
+
+export const api = {
+ get(path: string) {
+ return request(path)
+ },
+ post(path: string, body?: unknown) {
+ return request(path, {
+ method: 'POST',
+ body: body != null ? JSON.stringify(body) : undefined,
+ })
+ },
+ put(path: string, body?: unknown) {
+ return request(path, {
+ method: 'PUT',
+ body: body != null ? JSON.stringify(body) : undefined,
+ })
+ },
+ delete(path: string, body?: unknown) {
+ return request(path, {
+ method: 'DELETE',
+ body: body != null ? JSON.stringify(body) : undefined,
+ })
+ },
+}
diff --git a/src/services/mock-data.ts b/src/services/mock-data.ts
new file mode 100644
index 0000000..cd2a60c
--- /dev/null
+++ b/src/services/mock-data.ts
@@ -0,0 +1,98 @@
+import type { DashboardStats, AuditLog, TrendPoint } from '@/types/admin'
+
+function makeTrend(days: number, base: number, variance: number): TrendPoint[] {
+ return Array.from({ length: days }, (_, i) => {
+ const d = new Date()
+ d.setDate(d.getDate() - (days - 1 - i))
+ return {
+ date: d.toISOString().split('T')[0],
+ value: Math.max(0, base + Math.round((Math.random() - 0.5) * variance)),
+ }
+ })
+}
+
+export const MOCK_DASHBOARD_STATS: DashboardStats = {
+ totalUsers: 1286,
+ newUsersToday: 23,
+ activeUsersToday: 347,
+ totalKnowledgeBases: 892,
+ newKbsToday: 15,
+ totalAiCallsToday: 4521,
+ totalFiles: 3412,
+ totalStorageBytes: 15_728_640_000,
+ userTrend: makeTrend(30, 40, 30),
+ aiCallTrend: makeTrend(30, 150, 100),
+}
+
+export const MOCK_AUDIT_LOGS: AuditLog[] = [
+ {
+ id: 'log-001',
+ adminUserId: 'admin-001',
+ adminUserEmail: 'admin@longde.cloud',
+ adminUserDisplayName: '超级管理员',
+ action: 'LOGIN',
+ resourceType: null,
+ resourceId: null,
+ beforeJson: null,
+ afterJson: null,
+ ip: '120.53.227.155',
+ userAgent: 'Chrome/130.0',
+ createdAt: new Date(Date.now() - 1000 * 60 * 5).toISOString(),
+ },
+ {
+ id: 'log-002',
+ adminUserId: 'admin-001',
+ adminUserEmail: 'admin@longde.cloud',
+ adminUserDisplayName: '超级管理员',
+ action: 'CREATE_ADMIN',
+ resourceType: 'AdminUser',
+ resourceId: 'admin-002',
+ beforeJson: null,
+ afterJson: { email: 'ops@longde.cloud', role: 'OPERATIONS' },
+ ip: '120.53.227.155',
+ userAgent: 'Chrome/130.0',
+ createdAt: new Date(Date.now() - 1000 * 60 * 30).toISOString(),
+ },
+ {
+ id: 'log-003',
+ adminUserId: 'admin-002',
+ adminUserEmail: 'ops@longde.cloud',
+ adminUserDisplayName: '运营小王',
+ action: 'UPDATE_USER_STATUS',
+ resourceType: 'User',
+ resourceId: 'user-123',
+ beforeJson: { status: 'active' },
+ afterJson: { status: 'disabled' },
+ ip: '120.53.227.156',
+ userAgent: 'Firefox/132.0',
+ createdAt: new Date(Date.now() - 1000 * 60 * 60).toISOString(),
+ },
+ {
+ id: 'log-004',
+ adminUserId: 'admin-001',
+ adminUserEmail: 'admin@longde.cloud',
+ adminUserDisplayName: '超级管理员',
+ action: 'DELETE_FILE',
+ resourceType: 'UploadedFile',
+ resourceId: 'file-456',
+ beforeJson: { filename: '违规内容.pdf' },
+ afterJson: null,
+ ip: '120.53.227.155',
+ userAgent: 'Chrome/130.0',
+ createdAt: new Date(Date.now() - 1000 * 60 * 60 * 2).toISOString(),
+ },
+ {
+ id: 'log-005',
+ adminUserId: 'admin-001',
+ adminUserEmail: 'admin@longde.cloud',
+ adminUserDisplayName: '超级管理员',
+ action: 'LOGIN_FAILED',
+ resourceType: null,
+ resourceId: null,
+ beforeJson: null,
+ afterJson: null,
+ ip: '10.0.0.1',
+ userAgent: null,
+ createdAt: new Date(Date.now() - 1000 * 60 * 60 * 3).toISOString(),
+ },
+]
diff --git a/src/services/token-store.ts b/src/services/token-store.ts
new file mode 100644
index 0000000..4786d15
--- /dev/null
+++ b/src/services/token-store.ts
@@ -0,0 +1,35 @@
+const ACCESS_TOKEN_KEY = 'admin_access_token'
+const REFRESH_TOKEN_KEY = 'admin_refresh_token'
+const ADMIN_USER_KEY = 'admin_user'
+
+export function getAccessToken(): string | null {
+ return localStorage.getItem(ACCESS_TOKEN_KEY)
+}
+
+export function getRefreshToken(): string | null {
+ return localStorage.getItem(REFRESH_TOKEN_KEY)
+}
+
+export function setTokens(access: string, refresh: string): void {
+ localStorage.setItem(ACCESS_TOKEN_KEY, access)
+ localStorage.setItem(REFRESH_TOKEN_KEY, refresh)
+}
+
+export function clearTokens(): void {
+ localStorage.removeItem(ACCESS_TOKEN_KEY)
+ localStorage.removeItem(REFRESH_TOKEN_KEY)
+ localStorage.removeItem(ADMIN_USER_KEY)
+}
+
+export function getStoredAdminUser(): T | null {
+ try {
+ const raw = localStorage.getItem(ADMIN_USER_KEY)
+ return raw ? (JSON.parse(raw) as T) : null
+ } catch {
+ return null
+ }
+}
+
+export function setStoredAdminUser(user: T): void {
+ localStorage.setItem(ADMIN_USER_KEY, JSON.stringify(user))
+}
diff --git a/src/types/admin.ts b/src/types/admin.ts
new file mode 100644
index 0000000..d9de3b4
--- /dev/null
+++ b/src/types/admin.ts
@@ -0,0 +1,46 @@
+export type AdminRole = 'SUPER_ADMIN' | 'ADMIN' | 'OPERATIONS' | 'DEVELOPER' | 'READONLY'
+
+export interface AdminUser {
+ id: string
+ email: string
+ displayName: string
+ role: AdminRole
+ status: 'ACTIVE' | 'DISABLED'
+ twoFactorEnabled: boolean
+ lastLoginAt: string | null
+ lastLoginIp: string | null
+ createdAt: string
+}
+
+export interface TrendPoint {
+ date: string
+ value: number
+}
+
+export interface DashboardStats {
+ totalUsers: number
+ newUsersToday: number
+ activeUsersToday: number
+ totalKnowledgeBases: number
+ newKbsToday: number
+ totalAiCallsToday: number
+ totalFiles: number
+ totalStorageBytes: number
+ userTrend: TrendPoint[]
+ aiCallTrend: TrendPoint[]
+}
+
+export interface AuditLog {
+ id: string
+ adminUserId: string
+ adminUserEmail?: string
+ adminUserDisplayName?: string
+ action: string
+ resourceType: string | null
+ resourceId: string | null
+ beforeJson: unknown
+ afterJson: unknown
+ ip: string | null
+ userAgent: string | null
+ createdAt: string
+}
diff --git a/src/types/api.ts b/src/types/api.ts
new file mode 100644
index 0000000..9460d71
--- /dev/null
+++ b/src/types/api.ts
@@ -0,0 +1,14 @@
+export interface PaginatedResult {
+ items: T[]
+ total: number
+ page: number
+ limit: number
+ totalPages: number
+}
+
+export interface PaginationParams {
+ page?: number
+ limit?: number
+ sortBy?: string
+ sortOrder?: 'asc' | 'desc'
+}
diff --git a/tsconfig.app.json b/tsconfig.app.json
index 7f42e5f..a358e37 100644
--- a/tsconfig.app.json
+++ b/tsconfig.app.json
@@ -16,6 +16,9 @@
"jsx": "react-jsx",
/* Linting */
+ "baseUrl": ".",
+ "paths": { "@/*": ["src/*"] },
+ "ignoreDeprecations": "6.0",
"noUnusedLocals": true,
"noUnusedParameters": true,
"erasableSyntaxOnly": true,
diff --git a/vite.config.ts b/vite.config.ts
index 7255652..23668c9 100644
--- a/vite.config.ts
+++ b/vite.config.ts
@@ -1,9 +1,15 @@
import { defineConfig } from 'vite'
import react from '@vitejs/plugin-react'
import tailwindcss from '@tailwindcss/vite'
+import path from 'node:path'
export default defineConfig({
plugins: [react(), tailwindcss()],
+ resolve: {
+ alias: {
+ '@': path.resolve(import.meta.dirname, 'src'),
+ },
+ },
server: {
port: 5174,
proxy: {