完整API模块结构:
client/admin/api/ ├── auth.ts (认证API) ├── users.ts (用户API) ├── files.ts (文件API) ├── theme.ts (主题API) ├── charts.ts (图表API) ├── messages.ts (消息API) ├── sys.ts (系统API) ├── know_info.ts (知识库API) ├── maps.ts (地图API) └── index.ts (统一入口)
This commit is contained in:
@@ -3,5 +3,6 @@
|
|||||||
迁移管理页面,在正式环境中,需要验证env中配置的密码参数才能打开
|
迁移管理页面,在正式环境中,需要验证env中配置的密码参数才能打开
|
||||||
|
|
||||||
2025.05.13 0.1.0
|
2025.05.13 0.1.0
|
||||||
首页添加了迁移管理入口按钮, 无需登录即可访问
|
将admin api.ts 拆开
|
||||||
打开迁移管理页面时,将迁移历史读取出来
|
打开迁移管理页面时,将迁移历史读取出来
|
||||||
|
首页添加了迁移管理入口按钮, 无需登录即可访问
|
||||||
@@ -1,782 +0,0 @@
|
|||||||
import axios from 'axios';
|
|
||||||
import { getGlobalConfig } from './utils.ts';
|
|
||||||
import type { MinioUploadPolicy, OSSUploadPolicy } from '@d8d-appcontainer/types';
|
|
||||||
import 'dayjs/locale/zh-cn';
|
|
||||||
import type {
|
|
||||||
User, FileLibrary, FileCategory, ThemeSettings,
|
|
||||||
SystemSetting, SystemSettingGroupData,
|
|
||||||
LoginLocation, LoginLocationDetail,
|
|
||||||
Message, UserMessage, KnowInfo
|
|
||||||
} from '../share/types.ts';
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// 定义API基础URL
|
|
||||||
const API_BASE_URL = '/api';
|
|
||||||
|
|
||||||
// 获取OSS完整URL
|
|
||||||
export const getOssUrl = (path: string): string => {
|
|
||||||
// 获取全局配置中的OSS_HOST,如果不存在使用默认值
|
|
||||||
const ossHost = getGlobalConfig('OSS_BASE_URL') || '';
|
|
||||||
// 确保path不以/开头
|
|
||||||
const ossPath = path.startsWith('/') ? path.substring(1) : path;
|
|
||||||
return `${ossHost}/${ossPath}`;
|
|
||||||
};
|
|
||||||
|
|
||||||
// ===================
|
|
||||||
// Auth API 定义部分
|
|
||||||
// ===================
|
|
||||||
|
|
||||||
// 定义API返回数据类型
|
|
||||||
interface AuthLoginResponse {
|
|
||||||
message: string;
|
|
||||||
token: string;
|
|
||||||
refreshToken?: string;
|
|
||||||
user: User;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface AuthResponse {
|
|
||||||
message: string;
|
|
||||||
[key: string]: any;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 定义Auth API接口类型
|
|
||||||
interface AuthAPIType {
|
|
||||||
login: (username: string, password: string, latitude?: number, longitude?: number) => Promise<AuthLoginResponse>;
|
|
||||||
register: (username: string, email: string, password: string) => Promise<AuthResponse>;
|
|
||||||
logout: () => Promise<AuthResponse>;
|
|
||||||
getCurrentUser: () => Promise<User>;
|
|
||||||
updateUser: (userId: number, userData: Partial<User>) => Promise<User>;
|
|
||||||
changePassword: (oldPassword: string, newPassword: string) => Promise<AuthResponse>;
|
|
||||||
requestPasswordReset: (email: string) => Promise<AuthResponse>;
|
|
||||||
resetPassword: (token: string, newPassword: string) => Promise<AuthResponse>;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// Auth相关API
|
|
||||||
export const AuthAPI: AuthAPIType = {
|
|
||||||
// 登录API
|
|
||||||
login: async (username: string, password: string, latitude?: number, longitude?: number) => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/auth/login`, {
|
|
||||||
username,
|
|
||||||
password,
|
|
||||||
latitude,
|
|
||||||
longitude
|
|
||||||
});
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 注册API
|
|
||||||
register: async (username: string, email: string, password: string) => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/auth/register`, { username, email, password });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 登出API
|
|
||||||
logout: async () => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/auth/logout`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取当前用户信息
|
|
||||||
getCurrentUser: async () => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/auth/me`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 更新用户信息
|
|
||||||
updateUser: async (userId: number, userData: Partial<User>) => {
|
|
||||||
try {
|
|
||||||
const response = await axios.put(`${API_BASE_URL}/auth/users/${userId}`, userData);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 修改密码
|
|
||||||
changePassword: async (oldPassword: string, newPassword: string) => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/auth/change-password`, { oldPassword, newPassword });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 请求重置密码
|
|
||||||
requestPasswordReset: async (email: string) => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/auth/request-password-reset`, { email });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 重置密码
|
|
||||||
resetPassword: async (token: string, newPassword: string) => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/auth/reset-password`, { token, newPassword });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// 为UserAPI添加的接口响应类型
|
|
||||||
interface UsersResponse {
|
|
||||||
data: User[];
|
|
||||||
pagination: {
|
|
||||||
total: number;
|
|
||||||
current: number;
|
|
||||||
pageSize: number;
|
|
||||||
totalPages: number;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
interface UserResponse {
|
|
||||||
data: User;
|
|
||||||
message?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface UserCreateResponse {
|
|
||||||
message: string;
|
|
||||||
data: User;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface UserUpdateResponse {
|
|
||||||
message: string;
|
|
||||||
data: User;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface UserDeleteResponse {
|
|
||||||
message: string;
|
|
||||||
id: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 用户管理API
|
|
||||||
export const UserAPI = {
|
|
||||||
// 获取用户列表
|
|
||||||
getUsers: async (params?: { page?: number, limit?: number, search?: string }): Promise<UsersResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/users`, { params });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取单个用户详情
|
|
||||||
getUser: async (userId: number): Promise<UserResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/users/${userId}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 创建用户
|
|
||||||
createUser: async (userData: Partial<User>): Promise<UserCreateResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/users`, userData);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 更新用户信息
|
|
||||||
updateUser: async (userId: number, userData: Partial<User>): Promise<UserUpdateResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.put(`${API_BASE_URL}/users/${userId}`, userData);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 删除用户
|
|
||||||
deleteUser: async (userId: number): Promise<UserDeleteResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.delete(`${API_BASE_URL}/users/${userId}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
// 定义文件相关接口类型
|
|
||||||
interface FileUploadPolicyResponse {
|
|
||||||
message: string;
|
|
||||||
data: MinioUploadPolicy | OSSUploadPolicy;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileListResponse {
|
|
||||||
message: string;
|
|
||||||
data: {
|
|
||||||
list: FileLibrary[];
|
|
||||||
pagination: {
|
|
||||||
current: number;
|
|
||||||
pageSize: number;
|
|
||||||
total: number;
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileSaveResponse {
|
|
||||||
message: string;
|
|
||||||
data: FileLibrary;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileInfoResponse {
|
|
||||||
message: string;
|
|
||||||
data: FileLibrary;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileDeleteResponse {
|
|
||||||
message: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
interface FileCategoryListResponse {
|
|
||||||
data: FileCategory[];
|
|
||||||
total: number;
|
|
||||||
page: number;
|
|
||||||
pageSize: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileCategoryCreateResponse {
|
|
||||||
message: string;
|
|
||||||
data: FileCategory;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileCategoryUpdateResponse {
|
|
||||||
message: string;
|
|
||||||
data: FileCategory;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileCategoryDeleteResponse {
|
|
||||||
message: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 文件API接口定义
|
|
||||||
export const FileAPI = {
|
|
||||||
// 获取文件上传策略
|
|
||||||
getUploadPolicy: async (filename: string, prefix: string = 'uploads/', maxSize: number = 10 * 1024 * 1024): Promise<FileUploadPolicyResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/upload/policy`, {
|
|
||||||
params: { filename, prefix, maxSize }
|
|
||||||
});
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 保存文件信息
|
|
||||||
saveFileInfo: async (fileData: Partial<FileLibrary>): Promise<FileSaveResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/upload/save`, fileData);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取文件列表
|
|
||||||
getFileList: async (params?: {
|
|
||||||
page?: number,
|
|
||||||
pageSize?: number,
|
|
||||||
category_id?: number,
|
|
||||||
fileType?: string,
|
|
||||||
keyword?: string
|
|
||||||
}): Promise<FileListResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/upload/list`, { params });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取单个文件信息
|
|
||||||
getFileInfo: async (id: number): Promise<FileInfoResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/upload/${id}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 更新文件下载计数
|
|
||||||
updateDownloadCount: async (id: number): Promise<FileDeleteResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/upload/${id}/download`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 删除文件
|
|
||||||
deleteFile: async (id: number): Promise<FileDeleteResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.delete(`${API_BASE_URL}/upload/${id}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取文件分类列表
|
|
||||||
getCategories: async (params?: {
|
|
||||||
page?: number,
|
|
||||||
pageSize?: number,
|
|
||||||
search?: string
|
|
||||||
}): Promise<FileCategoryListResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/file-categories`, { params });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 创建文件分类
|
|
||||||
createCategory: async (data: Partial<FileCategory>): Promise<FileCategoryCreateResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/file-categories`, data);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 更新文件分类
|
|
||||||
updateCategory: async (id: number, data: Partial<FileCategory>): Promise<FileCategoryUpdateResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.put(`${API_BASE_URL}/file-categories/${id}`, data);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 删除文件分类
|
|
||||||
deleteCategory: async (id: number): Promise<FileCategoryDeleteResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.delete(`${API_BASE_URL}/file-categories/${id}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Theme API 响应类型
|
|
||||||
export interface ThemeSettingsResponse {
|
|
||||||
message: string;
|
|
||||||
data: ThemeSettings;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Theme API 定义
|
|
||||||
export const ThemeAPI = {
|
|
||||||
// 获取主题设置
|
|
||||||
getThemeSettings: async (): Promise<ThemeSettings> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/theme`);
|
|
||||||
return response.data.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 更新主题设置
|
|
||||||
updateThemeSettings: async (themeData: Partial<ThemeSettings>): Promise<ThemeSettings> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.put(`${API_BASE_URL}/theme`, themeData);
|
|
||||||
return response.data.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 重置主题设置
|
|
||||||
resetThemeSettings: async (): Promise<ThemeSettings> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/theme/reset`);
|
|
||||||
return response.data.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// 图表数据API接口类型
|
|
||||||
interface ChartDataResponse<T> {
|
|
||||||
message: string;
|
|
||||||
data: T;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface UserActivityData {
|
|
||||||
date: string;
|
|
||||||
count: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileUploadsData {
|
|
||||||
month: string;
|
|
||||||
count: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface FileTypesData {
|
|
||||||
type: string;
|
|
||||||
value: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface DashboardOverviewData {
|
|
||||||
userCount: number;
|
|
||||||
fileCount: number;
|
|
||||||
articleCount: number;
|
|
||||||
todayLoginCount: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 图表数据API
|
|
||||||
export const ChartAPI = {
|
|
||||||
// 获取用户活跃度数据
|
|
||||||
getUserActivity: async (): Promise<ChartDataResponse<UserActivityData[]>> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/charts/user-activity`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取文件上传统计数据
|
|
||||||
getFileUploads: async (): Promise<ChartDataResponse<FileUploadsData[]>> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/charts/file-uploads`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取文件类型分布数据
|
|
||||||
getFileTypes: async (): Promise<ChartDataResponse<FileTypesData[]>> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/charts/file-types`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取仪表盘概览数据
|
|
||||||
getDashboardOverview: async (): Promise<ChartDataResponse<DashboardOverviewData>> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/charts/dashboard-overview`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// 消息API接口类型
|
|
||||||
interface MessagesResponse {
|
|
||||||
data: UserMessage[];
|
|
||||||
pagination: {
|
|
||||||
total: number;
|
|
||||||
current: number;
|
|
||||||
pageSize: number;
|
|
||||||
totalPages: number;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
interface MessageResponse {
|
|
||||||
data: Message;
|
|
||||||
message?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface MessageCountResponse {
|
|
||||||
count: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 消息API
|
|
||||||
export const MessageAPI = {
|
|
||||||
// 获取消息列表
|
|
||||||
getMessages: async (params?: {
|
|
||||||
page?: number,
|
|
||||||
pageSize?: number,
|
|
||||||
type?: string,
|
|
||||||
status?: string,
|
|
||||||
search?: string
|
|
||||||
}): Promise<MessagesResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/messages`, { params });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 发送消息
|
|
||||||
sendMessage: async (data: {
|
|
||||||
title: string,
|
|
||||||
content: string,
|
|
||||||
type: string,
|
|
||||||
receiver_ids: number[]
|
|
||||||
}): Promise<MessageResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/messages`, data);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取未读消息数
|
|
||||||
getUnreadCount: async (): Promise<MessageCountResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/messages/count/unread`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 标记消息为已读
|
|
||||||
markAsRead: async (id: number): Promise<MessageResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/messages/${id}/read`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 删除消息
|
|
||||||
deleteMessage: async (id: number): Promise<MessageResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.delete(`${API_BASE_URL}/messages/${id}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// 地图相关API的接口类型定义
|
|
||||||
export interface LoginLocationResponse {
|
|
||||||
message: string;
|
|
||||||
data: LoginLocation[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface LoginLocationDetailResponse {
|
|
||||||
message: string;
|
|
||||||
data: LoginLocationDetail;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface LoginLocationUpdateResponse {
|
|
||||||
message: string;
|
|
||||||
data: LoginLocationDetail;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 知识库相关接口类型定义
|
|
||||||
export interface KnowInfoListResponse {
|
|
||||||
data: KnowInfo[];
|
|
||||||
pagination: {
|
|
||||||
total: number;
|
|
||||||
current: number;
|
|
||||||
pageSize: number;
|
|
||||||
totalPages: number;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
interface KnowInfoResponse {
|
|
||||||
data: KnowInfo;
|
|
||||||
message?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface KnowInfoCreateResponse {
|
|
||||||
message: string;
|
|
||||||
data: KnowInfo;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface KnowInfoUpdateResponse {
|
|
||||||
message: string;
|
|
||||||
data: KnowInfo;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface KnowInfoDeleteResponse {
|
|
||||||
message: string;
|
|
||||||
id: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// 地图相关API
|
|
||||||
export const MapAPI = {
|
|
||||||
// 获取地图标记点数据
|
|
||||||
getMarkers: async (params?: {
|
|
||||||
startTime?: string;
|
|
||||||
endTime?: string;
|
|
||||||
userId?: number
|
|
||||||
}): Promise<LoginLocationResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/map/markers`, { params });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取登录位置详情
|
|
||||||
getLocationDetail: async (locationId: number): Promise<LoginLocationDetailResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/map/location/${locationId}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 更新登录位置信息
|
|
||||||
updateLocation: async (locationId: number, data: {
|
|
||||||
longitude: number;
|
|
||||||
latitude: number;
|
|
||||||
location_name?: string;
|
|
||||||
}): Promise<LoginLocationUpdateResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.put(`${API_BASE_URL}/map/location/${locationId}`, data);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// 系统设置API
|
|
||||||
// 知识库API
|
|
||||||
export const KnowInfoAPI = {
|
|
||||||
// 获取知识库列表
|
|
||||||
getKnowInfos: async (params?: {
|
|
||||||
page?: number;
|
|
||||||
pageSize?: number;
|
|
||||||
title?: string;
|
|
||||||
category?: string;
|
|
||||||
tags?: string;
|
|
||||||
}): Promise<KnowInfoListResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/know-infos`, { params });
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取单个知识详情
|
|
||||||
getKnowInfo: async (id: number): Promise<KnowInfoResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/know-infos/${id}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 创建知识
|
|
||||||
createKnowInfo: async (data: Partial<KnowInfo>): Promise<KnowInfoCreateResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/know-infos`, data);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 更新知识
|
|
||||||
updateKnowInfo: async (id: number, data: Partial<KnowInfo>): Promise<KnowInfoUpdateResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.put(`${API_BASE_URL}/know-infos/${id}`, data);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 删除知识
|
|
||||||
deleteKnowInfo: async (id: number): Promise<KnowInfoDeleteResponse> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.delete(`${API_BASE_URL}/know-infos/${id}`);
|
|
||||||
return response.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
export const SystemAPI = {
|
|
||||||
// 获取所有系统设置
|
|
||||||
getSettings: async (): Promise<SystemSettingGroupData[]> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/settings`);
|
|
||||||
return response.data.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 获取指定分组的系统设置
|
|
||||||
getSettingsByGroup: async (group: string): Promise<SystemSetting[]> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.get(`${API_BASE_URL}/settings/group/${group}`);
|
|
||||||
return response.data.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 更新系统设置
|
|
||||||
updateSettings: async (settings: Partial<SystemSetting>[]): Promise<SystemSetting[]> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.put(`${API_BASE_URL}/settings`, settings);
|
|
||||||
return response.data.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
// 重置系统设置
|
|
||||||
resetSettings: async (): Promise<SystemSetting[]> => {
|
|
||||||
try {
|
|
||||||
const response = await axios.post(`${API_BASE_URL}/settings/reset`);
|
|
||||||
return response.data.data;
|
|
||||||
} catch (error) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
106
client/admin/api/auth.ts
Normal file
106
client/admin/api/auth.ts
Normal file
@@ -0,0 +1,106 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import type { User } from '../../share/types.ts';
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
interface AuthLoginResponse {
|
||||||
|
message: string;
|
||||||
|
token: string;
|
||||||
|
refreshToken?: string;
|
||||||
|
user: User;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface AuthResponse {
|
||||||
|
message: string;
|
||||||
|
[key: string]: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface AuthAPIType {
|
||||||
|
login: (username: string, password: string, latitude?: number, longitude?: number) => Promise<AuthLoginResponse>;
|
||||||
|
register: (username: string, email: string, password: string) => Promise<AuthResponse>;
|
||||||
|
logout: () => Promise<AuthResponse>;
|
||||||
|
getCurrentUser: () => Promise<User>;
|
||||||
|
updateUser: (userId: number, userData: Partial<User>) => Promise<User>;
|
||||||
|
changePassword: (oldPassword: string, newPassword: string) => Promise<AuthResponse>;
|
||||||
|
requestPasswordReset: (email: string) => Promise<AuthResponse>;
|
||||||
|
resetPassword: (token: string, newPassword: string) => Promise<AuthResponse>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const AuthAPI: AuthAPIType = {
|
||||||
|
login: async (username: string, password: string, latitude?: number, longitude?: number) => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/auth/login`, {
|
||||||
|
username,
|
||||||
|
password,
|
||||||
|
latitude,
|
||||||
|
longitude
|
||||||
|
});
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
register: async (username: string, email: string, password: string) => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/auth/register`, { username, email, password });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
logout: async () => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/auth/logout`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getCurrentUser: async () => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/auth/me`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
updateUser: async (userId: number, userData: Partial<User>) => {
|
||||||
|
try {
|
||||||
|
const response = await axios.put(`${API_BASE_URL}/auth/users/${userId}`, userData);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
changePassword: async (oldPassword: string, newPassword: string) => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/auth/change-password`, { oldPassword, newPassword });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
requestPasswordReset: async (email: string) => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/auth/request-password-reset`, { email });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
resetPassword: async (token: string, newPassword: string) => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/auth/reset-password`, { token, newPassword });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
68
client/admin/api/charts.ts
Normal file
68
client/admin/api/charts.ts
Normal file
@@ -0,0 +1,68 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
interface ChartDataResponse<T> {
|
||||||
|
message: string;
|
||||||
|
data: T;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UserActivityData {
|
||||||
|
date: string;
|
||||||
|
count: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileUploadsData {
|
||||||
|
month: string;
|
||||||
|
count: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileTypesData {
|
||||||
|
type: string;
|
||||||
|
value: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface DashboardOverviewData {
|
||||||
|
userCount: number;
|
||||||
|
fileCount: number;
|
||||||
|
articleCount: number;
|
||||||
|
todayLoginCount: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ChartAPI = {
|
||||||
|
getUserActivity: async (): Promise<ChartDataResponse<UserActivityData[]>> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/charts/user-activity`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getFileUploads: async (): Promise<ChartDataResponse<FileUploadsData[]>> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/charts/file-uploads`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getFileTypes: async (): Promise<ChartDataResponse<FileTypesData[]>> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/charts/file-types`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getDashboardOverview: async (): Promise<ChartDataResponse<DashboardOverviewData>> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/charts/dashboard-overview`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
161
client/admin/api/files.ts
Normal file
161
client/admin/api/files.ts
Normal file
@@ -0,0 +1,161 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import type { FileLibrary, FileCategory } from '../../share/types.ts';
|
||||||
|
import type { MinioUploadPolicy, OSSUploadPolicy } from '@d8d-appcontainer/types';
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
interface FileUploadPolicyResponse {
|
||||||
|
message: string;
|
||||||
|
data: MinioUploadPolicy | OSSUploadPolicy;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileListResponse {
|
||||||
|
message: string;
|
||||||
|
data: {
|
||||||
|
list: FileLibrary[];
|
||||||
|
pagination: {
|
||||||
|
current: number;
|
||||||
|
pageSize: number;
|
||||||
|
total: number;
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileSaveResponse {
|
||||||
|
message: string;
|
||||||
|
data: FileLibrary;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileInfoResponse {
|
||||||
|
message: string;
|
||||||
|
data: FileLibrary;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileDeleteResponse {
|
||||||
|
message: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileCategoryListResponse {
|
||||||
|
data: FileCategory[];
|
||||||
|
total: number;
|
||||||
|
page: number;
|
||||||
|
pageSize: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileCategoryCreateResponse {
|
||||||
|
message: string;
|
||||||
|
data: FileCategory;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileCategoryUpdateResponse {
|
||||||
|
message: string;
|
||||||
|
data: FileCategory;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface FileCategoryDeleteResponse {
|
||||||
|
message: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const FileAPI = {
|
||||||
|
getUploadPolicy: async (filename: string, prefix: string = 'uploads/', maxSize: number = 10 * 1024 * 1024): Promise<FileUploadPolicyResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/upload/policy`, {
|
||||||
|
params: { filename, prefix, maxSize }
|
||||||
|
});
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
saveFileInfo: async (fileData: Partial<FileLibrary>): Promise<FileSaveResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/upload/save`, fileData);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getFileList: async (params?: {
|
||||||
|
page?: number,
|
||||||
|
pageSize?: number,
|
||||||
|
category_id?: number,
|
||||||
|
fileType?: string,
|
||||||
|
keyword?: string
|
||||||
|
}): Promise<FileListResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/upload/list`, { params });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getFileInfo: async (id: number): Promise<FileInfoResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/upload/${id}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
updateDownloadCount: async (id: number): Promise<FileDeleteResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/upload/${id}/download`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
deleteFile: async (id: number): Promise<FileDeleteResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.delete(`${API_BASE_URL}/upload/${id}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getCategories: async (params?: {
|
||||||
|
page?: number,
|
||||||
|
pageSize?: number,
|
||||||
|
search?: string
|
||||||
|
}): Promise<FileCategoryListResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/file-categories`, { params });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
createCategory: async (data: Partial<FileCategory>): Promise<FileCategoryCreateResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/file-categories`, data);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
updateCategory: async (id: number, data: Partial<FileCategory>): Promise<FileCategoryUpdateResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.put(`${API_BASE_URL}/file-categories/${id}`, data);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
deleteCategory: async (id: number): Promise<FileCategoryDeleteResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.delete(`${API_BASE_URL}/file-categories/${id}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
9
client/admin/api/index.ts
Normal file
9
client/admin/api/index.ts
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
export * from './auth.ts';
|
||||||
|
export * from './users.ts';
|
||||||
|
export * from './files.ts';
|
||||||
|
export * from './theme.ts';
|
||||||
|
export * from './charts.ts';
|
||||||
|
export * from './messages.ts';
|
||||||
|
export * from './sys.ts';
|
||||||
|
export * from './know_info.ts';
|
||||||
|
export * from './maps.ts';
|
||||||
95
client/admin/api/know_info.ts
Normal file
95
client/admin/api/know_info.ts
Normal file
@@ -0,0 +1,95 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import type { KnowInfo } from '../../share/types.ts';
|
||||||
|
|
||||||
|
export interface KnowInfoListResponse {
|
||||||
|
data: KnowInfo[];
|
||||||
|
pagination: {
|
||||||
|
current: number;
|
||||||
|
pageSize: number;
|
||||||
|
total: number;
|
||||||
|
totalPages: number;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
interface KnowInfoResponse {
|
||||||
|
data: KnowInfo;
|
||||||
|
message?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface KnowInfoCreateResponse {
|
||||||
|
message: string;
|
||||||
|
data: KnowInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface KnowInfoUpdateResponse {
|
||||||
|
message: string;
|
||||||
|
data: KnowInfo;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface KnowInfoDeleteResponse {
|
||||||
|
message: string;
|
||||||
|
id: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
|
||||||
|
// 知识库API
|
||||||
|
export const KnowInfoAPI = {
|
||||||
|
// 获取知识库列表
|
||||||
|
getKnowInfos: async (params?: {
|
||||||
|
page?: number;
|
||||||
|
pageSize?: number;
|
||||||
|
title?: string;
|
||||||
|
category?: string;
|
||||||
|
tags?: string;
|
||||||
|
}): Promise<KnowInfoListResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/know-infos`, { params });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 获取单个知识详情
|
||||||
|
getKnowInfo: async (id: number): Promise<KnowInfoResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/know-infos/${id}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 创建知识
|
||||||
|
createKnowInfo: async (data: Partial<KnowInfo>): Promise<KnowInfoCreateResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/know-infos`, data);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 更新知识
|
||||||
|
updateKnowInfo: async (id: number, data: Partial<KnowInfo>): Promise<KnowInfoUpdateResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.put(`${API_BASE_URL}/know-infos/${id}`, data);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 删除知识
|
||||||
|
deleteKnowInfo: async (id: number): Promise<KnowInfoDeleteResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.delete(`${API_BASE_URL}/know-infos/${id}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
64
client/admin/api/maps.ts
Normal file
64
client/admin/api/maps.ts
Normal file
@@ -0,0 +1,64 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import type {
|
||||||
|
LoginLocation, LoginLocationDetail,
|
||||||
|
} from '../../share/types.ts';
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
|
||||||
|
// 地图相关API的接口类型定义
|
||||||
|
export interface LoginLocationResponse {
|
||||||
|
message: string;
|
||||||
|
data: LoginLocation[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LoginLocationDetailResponse {
|
||||||
|
message: string;
|
||||||
|
data: LoginLocationDetail;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface LoginLocationUpdateResponse {
|
||||||
|
message: string;
|
||||||
|
data: LoginLocationDetail;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 地图相关API
|
||||||
|
export const MapAPI = {
|
||||||
|
// 获取地图标记点数据
|
||||||
|
getMarkers: async (params?: {
|
||||||
|
startTime?: string;
|
||||||
|
endTime?: string;
|
||||||
|
userId?: number
|
||||||
|
}): Promise<LoginLocationResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/map/markers`, { params });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 获取登录位置详情
|
||||||
|
getLocationDetail: async (locationId: number): Promise<LoginLocationDetailResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/map/location/${locationId}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 更新登录位置信息
|
||||||
|
updateLocation: async (locationId: number, data: {
|
||||||
|
longitude: number;
|
||||||
|
latitude: number;
|
||||||
|
location_name?: string;
|
||||||
|
}): Promise<LoginLocationUpdateResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.put(`${API_BASE_URL}/map/location/${locationId}`, data);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
81
client/admin/api/messages.ts
Normal file
81
client/admin/api/messages.ts
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import type { UserMessage, Message } from '../../share/types.ts';
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
interface MessagesResponse {
|
||||||
|
data: UserMessage[];
|
||||||
|
pagination: {
|
||||||
|
total: number;
|
||||||
|
current: number;
|
||||||
|
pageSize: number;
|
||||||
|
totalPages: number;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
interface MessageResponse {
|
||||||
|
data: Message;
|
||||||
|
message?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface MessageCountResponse {
|
||||||
|
count: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const MessageAPI = {
|
||||||
|
getMessages: async (params?: {
|
||||||
|
page?: number,
|
||||||
|
pageSize?: number,
|
||||||
|
type?: string,
|
||||||
|
status?: string,
|
||||||
|
search?: string
|
||||||
|
}): Promise<MessagesResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/messages`, { params });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
sendMessage: async (data: {
|
||||||
|
title: string,
|
||||||
|
content: string,
|
||||||
|
type: string,
|
||||||
|
receiver_ids: number[]
|
||||||
|
}): Promise<MessageResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/messages`, data);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getUnreadCount: async (): Promise<MessageCountResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/messages/count/unread`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
markAsRead: async (id: number): Promise<MessageResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/messages/${id}/read`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
deleteMessage: async (id: number): Promise<MessageResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.delete(`${API_BASE_URL}/messages/${id}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
49
client/admin/api/sys.ts
Normal file
49
client/admin/api/sys.ts
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
import type {
|
||||||
|
SystemSetting, SystemSettingGroupData,
|
||||||
|
} from '../../share/types.ts';
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
export const SystemAPI = {
|
||||||
|
// 获取所有系统设置
|
||||||
|
getSettings: async (): Promise<SystemSettingGroupData[]> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/settings`);
|
||||||
|
return response.data.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 获取指定分组的系统设置
|
||||||
|
getSettingsByGroup: async (group: string): Promise<SystemSetting[]> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/settings/group/${group}`);
|
||||||
|
return response.data.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 更新系统设置
|
||||||
|
updateSettings: async (settings: Partial<SystemSetting>[]): Promise<SystemSetting[]> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.put(`${API_BASE_URL}/settings`, settings);
|
||||||
|
return response.data.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// 重置系统设置
|
||||||
|
resetSettings: async (): Promise<SystemSetting[]> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/settings/reset`);
|
||||||
|
return response.data.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
38
client/admin/api/theme.ts
Normal file
38
client/admin/api/theme.ts
Normal file
@@ -0,0 +1,38 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import type { ThemeSettings } from '../../share/types.ts';
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
export interface ThemeSettingsResponse {
|
||||||
|
message: string;
|
||||||
|
data: ThemeSettings;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const ThemeAPI = {
|
||||||
|
getThemeSettings: async (): Promise<ThemeSettings> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/theme`);
|
||||||
|
return response.data.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
updateThemeSettings: async (themeData: Partial<ThemeSettings>): Promise<ThemeSettings> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.put(`${API_BASE_URL}/theme`, themeData);
|
||||||
|
return response.data.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
resetThemeSettings: async (): Promise<ThemeSettings> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/theme/reset`);
|
||||||
|
return response.data.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
81
client/admin/api/users.ts
Normal file
81
client/admin/api/users.ts
Normal file
@@ -0,0 +1,81 @@
|
|||||||
|
import axios from 'axios';
|
||||||
|
import type { User } from '../../share/types.ts';
|
||||||
|
|
||||||
|
const API_BASE_URL = '/api';
|
||||||
|
|
||||||
|
interface UsersResponse {
|
||||||
|
data: User[];
|
||||||
|
pagination: {
|
||||||
|
total: number;
|
||||||
|
current: number;
|
||||||
|
pageSize: number;
|
||||||
|
totalPages: number;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UserResponse {
|
||||||
|
data: User;
|
||||||
|
message?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UserCreateResponse {
|
||||||
|
message: string;
|
||||||
|
data: User;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UserUpdateResponse {
|
||||||
|
message: string;
|
||||||
|
data: User;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface UserDeleteResponse {
|
||||||
|
message: string;
|
||||||
|
id: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const UserAPI = {
|
||||||
|
getUsers: async (params?: { page?: number, limit?: number, search?: string }): Promise<UsersResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/users`, { params });
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
getUser: async (userId: number): Promise<UserResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.get(`${API_BASE_URL}/users/${userId}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
createUser: async (userData: Partial<User>): Promise<UserCreateResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.post(`${API_BASE_URL}/users`, userData);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
updateUser: async (userId: number, userData: Partial<User>): Promise<UserUpdateResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.put(`${API_BASE_URL}/users/${userId}`, userData);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
deleteUser: async (userId: number): Promise<UserDeleteResponse> => {
|
||||||
|
try {
|
||||||
|
const response = await axios.delete(`${API_BASE_URL}/users/${userId}`);
|
||||||
|
return response.data;
|
||||||
|
} catch (error) {
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
@@ -16,7 +16,7 @@ import type { MinioUploadPolicy, OSSUploadPolicy } from '@d8d-appcontainer/types
|
|||||||
import 'dayjs/locale/zh-cn';
|
import 'dayjs/locale/zh-cn';
|
||||||
import { OssType } from '../share/types.ts';
|
import { OssType } from '../share/types.ts';
|
||||||
|
|
||||||
import { FileAPI } from './api.ts';
|
import { FileAPI } from './api/index.ts';
|
||||||
|
|
||||||
// MinIO文件上传组件
|
// MinIO文件上传组件
|
||||||
export const Uploader = ({
|
export const Uploader = ({
|
||||||
|
|||||||
@@ -24,7 +24,7 @@ import {
|
|||||||
import {
|
import {
|
||||||
AuthAPI,
|
AuthAPI,
|
||||||
ThemeAPI
|
ThemeAPI
|
||||||
} from './api.ts';
|
} from './api/index.ts';
|
||||||
|
|
||||||
|
|
||||||
// 配置 dayjs 插件
|
// 配置 dayjs 插件
|
||||||
|
|||||||
@@ -1,11 +1,6 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import {
|
import {
|
||||||
Layout, Menu, Button, Table, Space,
|
Card, Spin, Row, Col, Statistic,
|
||||||
Form, Input, Select, message, Modal,
|
|
||||||
Card, Spin, Row, Col, Breadcrumb, Avatar,
|
|
||||||
Dropdown, ConfigProvider, theme, Typography,
|
|
||||||
Switch, Badge, Image, Upload, Divider, Descriptions,
|
|
||||||
Popconfirm, Tag, Statistic, DatePicker, Radio, Progress, Tabs, List, Alert, Collapse, Empty, Drawer
|
|
||||||
} from 'antd';
|
} from 'antd';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
@@ -15,13 +10,10 @@ import { Line , Pie, Column} from "@ant-design/plots";
|
|||||||
import 'dayjs/locale/zh-cn';
|
import 'dayjs/locale/zh-cn';
|
||||||
|
|
||||||
|
|
||||||
import { ChartAPI } from './api.ts';
|
import { ChartAPI } from './api/index.ts';
|
||||||
import { useTheme } from './hooks_sys.tsx';
|
import { useTheme } from './hooks_sys.tsx';
|
||||||
|
|
||||||
interface ChartTooltipInfo {
|
|
||||||
items: Array<Record<string, any>>;
|
|
||||||
title: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 用户活跃度图表组件
|
// 用户活跃度图表组件
|
||||||
const UserActivityChart: React.FC = () => {
|
const UserActivityChart: React.FC = () => {
|
||||||
@@ -49,7 +41,7 @@ const UserActivityChart: React.FC = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card title="用户活跃度趋势" bordered={false}>
|
<Card title="用户活跃度趋势" variant="borderless">
|
||||||
<Line {...config} />
|
<Line {...config} />
|
||||||
</Card>
|
</Card>
|
||||||
);
|
);
|
||||||
@@ -92,7 +84,7 @@ const FileUploadsChart: React.FC = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card title="文件上传统计" bordered={false}>
|
<Card title="文件上传统计" variant="borderless">
|
||||||
<Column {...config} />
|
<Column {...config} />
|
||||||
</Card>
|
</Card>
|
||||||
);
|
);
|
||||||
@@ -130,7 +122,7 @@ const FileTypesChart: React.FC = () => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Card title="文件类型分布" bordered={false}>
|
<Card title="文件类型分布" variant="borderless">
|
||||||
<Pie {...config} />
|
<Pie {...config} />
|
||||||
</Card>
|
</Card>
|
||||||
);
|
);
|
||||||
@@ -151,7 +143,7 @@ const DashboardOverview: React.FC = () => {
|
|||||||
return (
|
return (
|
||||||
<Row gutter={[16, 16]}>
|
<Row gutter={[16, 16]}>
|
||||||
<Col xs={12} sm={12} md={6}>
|
<Col xs={12} sm={12} md={6}>
|
||||||
<Card bordered={false}>
|
<Card variant="borderless">
|
||||||
<Statistic
|
<Statistic
|
||||||
title="用户总数"
|
title="用户总数"
|
||||||
value={overviewData?.userCount || 0}
|
value={overviewData?.userCount || 0}
|
||||||
@@ -160,7 +152,7 @@ const DashboardOverview: React.FC = () => {
|
|||||||
</Card>
|
</Card>
|
||||||
</Col>
|
</Col>
|
||||||
<Col xs={12} sm={12} md={6}>
|
<Col xs={12} sm={12} md={6}>
|
||||||
<Card bordered={false}>
|
<Card variant="borderless">
|
||||||
<Statistic
|
<Statistic
|
||||||
title="文件总数"
|
title="文件总数"
|
||||||
value={overviewData?.fileCount || 0}
|
value={overviewData?.fileCount || 0}
|
||||||
@@ -169,7 +161,7 @@ const DashboardOverview: React.FC = () => {
|
|||||||
</Card>
|
</Card>
|
||||||
</Col>
|
</Col>
|
||||||
<Col xs={12} sm={12} md={6}>
|
<Col xs={12} sm={12} md={6}>
|
||||||
<Card bordered={false}>
|
<Card variant="borderless">
|
||||||
<Statistic
|
<Statistic
|
||||||
title="文章总数"
|
title="文章总数"
|
||||||
value={overviewData?.articleCount || 0}
|
value={overviewData?.articleCount || 0}
|
||||||
@@ -178,7 +170,7 @@ const DashboardOverview: React.FC = () => {
|
|||||||
</Card>
|
</Card>
|
||||||
</Col>
|
</Col>
|
||||||
<Col xs={12} sm={12} md={6}>
|
<Col xs={12} sm={12} md={6}>
|
||||||
<Card bordered={false}>
|
<Card variant="borderless">
|
||||||
<Statistic
|
<Statistic
|
||||||
title="今日登录"
|
title="今日登录"
|
||||||
value={overviewData?.todayLoginCount || 0}
|
value={overviewData?.todayLoginCount || 0}
|
||||||
|
|||||||
@@ -24,22 +24,19 @@ import weekday from 'dayjs/plugin/weekday';
|
|||||||
import localeData from 'dayjs/plugin/localeData';
|
import localeData from 'dayjs/plugin/localeData';
|
||||||
import 'dayjs/locale/zh-cn';
|
import 'dayjs/locale/zh-cn';
|
||||||
import type {
|
import type {
|
||||||
FileLibrary, FileCategory, KnowInfo
|
KnowInfo
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
AuditStatus,AuditStatusNameMap,
|
AuditStatus,AuditStatusNameMap,
|
||||||
OssType,
|
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
|
|
||||||
import { getEnumOptions } from './utils.ts';
|
import { getEnumOptions } from './utils.ts';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
FileAPI,
|
|
||||||
UserAPI,
|
|
||||||
KnowInfoAPI,
|
KnowInfoAPI,
|
||||||
type KnowInfoListResponse
|
type KnowInfoListResponse
|
||||||
} from './api.ts';
|
} from './api/index.ts';
|
||||||
|
|
||||||
|
|
||||||
// 配置 dayjs 插件
|
// 配置 dayjs 插件
|
||||||
@@ -49,7 +46,6 @@ dayjs.extend(localeData);
|
|||||||
// 设置 dayjs 语言
|
// 设置 dayjs 语言
|
||||||
dayjs.locale('zh-cn');
|
dayjs.locale('zh-cn');
|
||||||
|
|
||||||
const { Title } = Typography;
|
|
||||||
|
|
||||||
|
|
||||||
// 知识库管理页面组件
|
// 知识库管理页面组件
|
||||||
|
|||||||
@@ -1,19 +1,11 @@
|
|||||||
import React, { useState, useEffect } from 'react';
|
import React, { useState } from 'react';
|
||||||
import {
|
import {
|
||||||
Layout, Menu, Button, Table, Space,
|
Button, Space, Drawer,
|
||||||
Form, Input, Select, message, Modal,
|
Select, message,
|
||||||
Card, Spin, Row, Col, Breadcrumb, Avatar,
|
Card, Spin, Typography,Descriptions,DatePicker,
|
||||||
Dropdown, ConfigProvider, theme, Typography,
|
|
||||||
Switch, Badge, Image, Upload, Divider, Descriptions,
|
|
||||||
Popconfirm, Tag, Statistic, DatePicker, Radio, Progress, Tabs, List, Alert, Collapse, Empty, Drawer,
|
|
||||||
Tree
|
|
||||||
} from 'antd';
|
} from 'antd';
|
||||||
import {
|
import {
|
||||||
MenuFoldOutlined,
|
|
||||||
MenuUnfoldOutlined,
|
|
||||||
AppstoreOutlined,
|
|
||||||
EnvironmentOutlined,
|
EnvironmentOutlined,
|
||||||
SearchOutlined,
|
|
||||||
ClockCircleOutlined,
|
ClockCircleOutlined,
|
||||||
UserOutlined,
|
UserOutlined,
|
||||||
GlobalOutlined
|
GlobalOutlined
|
||||||
@@ -28,7 +20,8 @@ import type {
|
|||||||
MarkerData, LoginLocation, LoginLocationDetail, User
|
MarkerData, LoginLocation, LoginLocationDetail, User
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
|
|
||||||
import { MapAPI,UserAPI } from './api.ts';
|
import { UserAPI } from './api/index.ts';
|
||||||
|
import { MapAPI } from './api/index.ts';
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
|
|
||||||
const { RangePicker } = DatePicker;
|
const { RangePicker } = DatePicker;
|
||||||
|
|||||||
@@ -1,19 +1,17 @@
|
|||||||
import React, { useState } from 'react';
|
import React, { useState } from 'react';
|
||||||
import { useQuery, useMutation, useQueryClient, UseMutationResult } from '@tanstack/react-query';
|
import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
|
||||||
import { Button, Table, Space, Modal, Form, Input, Select, message } from 'antd';
|
import { Button, Table, Space, Modal, Form, Input, Select, message } from 'antd';
|
||||||
import type { TableProps } from 'antd';
|
import type { TableProps } from 'antd';
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
import 'dayjs/locale/zh-cn';
|
import 'dayjs/locale/zh-cn';
|
||||||
|
|
||||||
import { MessageAPI } from './api.ts';
|
import { MessageAPI , UserAPI } from './api/index.ts';
|
||||||
import { UserAPI } from './api.ts';
|
|
||||||
import type { UserMessage } from '../share/types.ts';
|
import type { UserMessage } from '../share/types.ts';
|
||||||
import { MessageStatusNameMap , MessageStatus} from '../share/types.ts';
|
import { MessageStatusNameMap , MessageStatus} from '../share/types.ts';
|
||||||
|
|
||||||
export const MessagesPage = () => {
|
export const MessagesPage = () => {
|
||||||
const queryClient = useQueryClient();
|
const queryClient = useQueryClient();
|
||||||
const [form] = Form.useForm();
|
const [form] = Form.useForm();
|
||||||
const [selectedRowKeys, setSelectedRowKeys] = useState<React.Key[]>([]);
|
|
||||||
const [isModalVisible, setIsModalVisible] = useState(false);
|
const [isModalVisible, setIsModalVisible] = useState(false);
|
||||||
const [searchParams, setSearchParams] = useState({
|
const [searchParams, setSearchParams] = useState({
|
||||||
page: 1,
|
page: 1,
|
||||||
|
|||||||
@@ -1,20 +1,14 @@
|
|||||||
import React, { useState, useEffect } from 'react';
|
import React, { useEffect } from 'react';
|
||||||
import {
|
import {
|
||||||
Layout, Menu, Button, Table, Space,
|
Button,Space,
|
||||||
Form, Input, Select, message, Modal,
|
Form, Input, Select, message, Modal,
|
||||||
Card, Spin, Row, Col, Breadcrumb, Avatar,
|
Card, Spin, Typography,
|
||||||
Dropdown, ConfigProvider, theme, Typography,
|
Switch, Tabs, Alert, InputNumber
|
||||||
Switch, Badge, Image, Upload, Divider, Descriptions,
|
|
||||||
Popconfirm, Tag, Statistic, DatePicker, Radio, Progress, Tabs, List, Alert, Collapse, Empty, Drawer, InputNumber,ColorPicker,
|
|
||||||
Popover
|
|
||||||
} from 'antd';
|
} from 'antd';
|
||||||
import {
|
import {
|
||||||
UploadOutlined,
|
|
||||||
ReloadOutlined,
|
ReloadOutlined,
|
||||||
SaveOutlined,
|
SaveOutlined,
|
||||||
BgColorsOutlined
|
|
||||||
} from '@ant-design/icons';
|
} from '@ant-design/icons';
|
||||||
import { debounce } from 'lodash';
|
|
||||||
import {
|
import {
|
||||||
useQuery,
|
useQuery,
|
||||||
useMutation,
|
useMutation,
|
||||||
@@ -25,25 +19,20 @@ import weekday from 'dayjs/plugin/weekday';
|
|||||||
import localeData from 'dayjs/plugin/localeData';
|
import localeData from 'dayjs/plugin/localeData';
|
||||||
import 'dayjs/locale/zh-cn';
|
import 'dayjs/locale/zh-cn';
|
||||||
import type {
|
import type {
|
||||||
FileLibrary, FileCategory, KnowInfo, SystemSetting, SystemSettingValue,
|
SystemSetting, SystemSettingValue
|
||||||
ColorScheme
|
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
import { ThemeMode } from '../share/types.ts';
|
|
||||||
|
|
||||||
import {
|
import {
|
||||||
SystemSettingGroup,
|
SystemSettingGroup,
|
||||||
SystemSettingKey,
|
SystemSettingKey,
|
||||||
FontSize,
|
|
||||||
CompactMode,
|
|
||||||
AllowedFileType
|
AllowedFileType
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
|
|
||||||
|
|
||||||
import { getEnumOptions } from './utils.ts';
|
|
||||||
|
|
||||||
import {
|
import {
|
||||||
SystemAPI,
|
SystemAPI
|
||||||
} from './api.ts';
|
} from './api/index.ts';
|
||||||
|
|
||||||
import { useTheme } from './hooks_sys.tsx';
|
import { useTheme } from './hooks_sys.tsx';
|
||||||
|
|
||||||
@@ -241,44 +230,44 @@ export const SettingsPage = () => {
|
|||||||
items={Object.values(SystemSettingGroup).map(group => ({
|
items={Object.values(SystemSettingGroup).map(group => ({
|
||||||
key: group,
|
key: group,
|
||||||
label: String(GROUP_TITLES[group]),
|
label: String(GROUP_TITLES[group]),
|
||||||
children: (
|
children: (
|
||||||
<div>
|
<div>
|
||||||
<Alert
|
<Alert
|
||||||
message={GROUP_DESCRIPTIONS[group]}
|
message={GROUP_DESCRIPTIONS[group]}
|
||||||
type="info"
|
type="info"
|
||||||
showIcon
|
showIcon
|
||||||
style={{ marginBottom: 24 }}
|
style={{ marginBottom: 24 }}
|
||||||
/>
|
/>
|
||||||
<Form
|
<Form
|
||||||
form={form}
|
form={form}
|
||||||
layout="vertical"
|
layout="vertical"
|
||||||
onFinish={handleSubmit}
|
onFinish={handleSubmit}
|
||||||
>
|
>
|
||||||
{settingsData
|
{settingsData
|
||||||
?.find(g => g.name === group)
|
?.find(g => g.name === group)
|
||||||
?.settings.map(setting => (
|
?.settings.map(setting => (
|
||||||
<Form.Item
|
<Form.Item
|
||||||
key={setting.key}
|
key={setting.key}
|
||||||
label={setting.description || setting.key}
|
label={setting.description || setting.key}
|
||||||
name={setting.key}
|
name={setting.key}
|
||||||
rules={[{ required: true, message: `请输入${setting.description || setting.key}` }]}
|
rules={[{ required: true, message: `请输入${setting.description || setting.key}` }]}
|
||||||
>
|
>
|
||||||
{renderSettingInput(setting)}
|
{renderSettingInput(setting)}
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
))}
|
))}
|
||||||
<Form.Item>
|
<Form.Item>
|
||||||
<Button
|
<Button
|
||||||
type="primary"
|
type="primary"
|
||||||
htmlType="submit"
|
htmlType="submit"
|
||||||
icon={<SaveOutlined />}
|
icon={<SaveOutlined />}
|
||||||
loading={updateSettingsMutation.isPending}
|
loading={updateSettingsMutation.isPending}
|
||||||
>
|
>
|
||||||
保存设置
|
保存设置
|
||||||
</Button>
|
</Button>
|
||||||
</Form.Item>
|
</Form.Item>
|
||||||
</Form>
|
</Form>
|
||||||
</div>
|
</div>
|
||||||
)
|
)
|
||||||
}))}
|
}))}
|
||||||
/>
|
/>
|
||||||
</Spin>
|
</Spin>
|
||||||
|
|||||||
@@ -25,20 +25,15 @@ import { uploadMinIOWithPolicy,uploadOSSWithPolicy } from '@d8d-appcontainer/api
|
|||||||
import type { MinioUploadPolicy, OSSUploadPolicy } from '@d8d-appcontainer/types';
|
import type { MinioUploadPolicy, OSSUploadPolicy } from '@d8d-appcontainer/types';
|
||||||
import 'dayjs/locale/zh-cn';
|
import 'dayjs/locale/zh-cn';
|
||||||
import type {
|
import type {
|
||||||
FileLibrary, FileCategory, KnowInfo
|
FileLibrary, FileCategory
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
AuditStatus,AuditStatusNameMap,
|
|
||||||
OssType,
|
OssType,
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
|
|
||||||
import { getEnumOptions } from './utils.ts';
|
|
||||||
|
|
||||||
import {
|
import { FileAPI , UserAPI } from './api/index.ts';
|
||||||
FileAPI,
|
|
||||||
UserAPI,
|
|
||||||
} from './api.ts';
|
|
||||||
|
|
||||||
|
|
||||||
// 配置 dayjs 插件
|
// 配置 dayjs 插件
|
||||||
|
|||||||
@@ -1,53 +1,30 @@
|
|||||||
import React, { useState, useEffect } from 'react';
|
import React, { useState, useEffect } from 'react';
|
||||||
import {
|
import {
|
||||||
Layout, Menu, Button, Table, Space,
|
Button, Space,
|
||||||
Form, Input, Select, message, Modal,
|
Form, message,
|
||||||
Card, Spin, Row, Col, Breadcrumb, Avatar,
|
Card, Spin, Typography,
|
||||||
Dropdown, ConfigProvider, theme, Typography,
|
Switch,
|
||||||
Switch, Badge, Image, Upload, Divider, Descriptions,
|
Popconfirm, Radio, InputNumber,ColorPicker,
|
||||||
Popconfirm, Tag, Statistic, DatePicker, Radio, Progress, Tabs, List, Alert, Collapse, Empty, Drawer, InputNumber,ColorPicker,
|
|
||||||
Popover
|
|
||||||
} from 'antd';
|
} from 'antd';
|
||||||
import {
|
|
||||||
UploadOutlined,
|
|
||||||
ReloadOutlined,
|
|
||||||
SaveOutlined,
|
|
||||||
BgColorsOutlined
|
|
||||||
} from '@ant-design/icons';
|
|
||||||
import { debounce } from 'lodash';
|
|
||||||
import {
|
|
||||||
useQuery,
|
|
||||||
useMutation,
|
|
||||||
useQueryClient,
|
|
||||||
} from '@tanstack/react-query';
|
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
import weekday from 'dayjs/plugin/weekday';
|
import weekday from 'dayjs/plugin/weekday';
|
||||||
import localeData from 'dayjs/plugin/localeData';
|
import localeData from 'dayjs/plugin/localeData';
|
||||||
import 'dayjs/locale/zh-cn';
|
import 'dayjs/locale/zh-cn';
|
||||||
import type {
|
import type {
|
||||||
FileLibrary, FileCategory, KnowInfo, SystemSetting, SystemSettingValue,
|
|
||||||
ColorScheme
|
ColorScheme
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
import { ThemeMode } from '../share/types.ts';
|
import { ThemeMode } from '../share/types.ts';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
SystemSettingGroup,
|
|
||||||
SystemSettingKey,
|
|
||||||
FontSize,
|
FontSize,
|
||||||
CompactMode,
|
CompactMode,
|
||||||
AllowedFileType
|
|
||||||
} from '../share/types.ts';
|
} from '../share/types.ts';
|
||||||
|
|
||||||
|
|
||||||
import { getEnumOptions } from './utils.ts';
|
|
||||||
|
|
||||||
import {
|
|
||||||
SystemAPI,
|
|
||||||
} from './api.ts';
|
|
||||||
|
|
||||||
import { useTheme } from './hooks_sys.tsx';
|
import { useTheme } from './hooks_sys.tsx';
|
||||||
|
|
||||||
import { Uploader } from './components_uploader.tsx';
|
|
||||||
|
|
||||||
// 配置 dayjs 插件
|
// 配置 dayjs 插件
|
||||||
dayjs.extend(weekday);
|
dayjs.extend(weekday);
|
||||||
|
|||||||
Reference in New Issue
Block a user