完整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:
yourname
2025-05-13 08:06:34 +00:00
parent bd72f60db8
commit 08ae3b85df
21 changed files with 818 additions and 907 deletions

View File

@@ -3,5 +3,6 @@
迁移管理页面在正式环境中需要验证env中配置的密码参数才能打开
2025.05.13 0.1.0
首页添加了迁移管理入口按钮, 无需登录即可访问
打开迁移管理页面时,将迁移历史读取出来
将admin api.ts 拆开
打开迁移管理页面时,将迁移历史读取出来
首页添加了迁移管理入口按钮, 无需登录即可访问

View File

@@ -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
View 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;
}
}
};

View 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
View 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;
}
}
};

View 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';

View 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
View 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;
}
}
};

View 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
View 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
View 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
View 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;
}
}
};

View File

@@ -16,7 +16,7 @@ import type { MinioUploadPolicy, OSSUploadPolicy } from '@d8d-appcontainer/types
import 'dayjs/locale/zh-cn';
import { OssType } from '../share/types.ts';
import { FileAPI } from './api.ts';
import { FileAPI } from './api/index.ts';
// MinIO文件上传组件
export const Uploader = ({

View File

@@ -24,7 +24,7 @@ import {
import {
AuthAPI,
ThemeAPI
} from './api.ts';
} from './api/index.ts';
// 配置 dayjs 插件

View File

@@ -1,11 +1,6 @@
import React from 'react';
import {
Layout, Menu, Button, Table, Space,
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
Card, Spin, Row, Col, Statistic,
} from 'antd';
import {
@@ -15,13 +10,10 @@ import { Line , Pie, Column} from "@ant-design/plots";
import 'dayjs/locale/zh-cn';
import { ChartAPI } from './api.ts';
import { ChartAPI } from './api/index.ts';
import { useTheme } from './hooks_sys.tsx';
interface ChartTooltipInfo {
items: Array<Record<string, any>>;
title: string;
}
// 用户活跃度图表组件
const UserActivityChart: React.FC = () => {
@@ -49,7 +41,7 @@ const UserActivityChart: React.FC = () => {
};
return (
<Card title="用户活跃度趋势" bordered={false}>
<Card title="用户活跃度趋势" variant="borderless">
<Line {...config} />
</Card>
);
@@ -92,7 +84,7 @@ const FileUploadsChart: React.FC = () => {
};
return (
<Card title="文件上传统计" bordered={false}>
<Card title="文件上传统计" variant="borderless">
<Column {...config} />
</Card>
);
@@ -130,7 +122,7 @@ const FileTypesChart: React.FC = () => {
};
return (
<Card title="文件类型分布" bordered={false}>
<Card title="文件类型分布" variant="borderless">
<Pie {...config} />
</Card>
);
@@ -151,7 +143,7 @@ const DashboardOverview: React.FC = () => {
return (
<Row gutter={[16, 16]}>
<Col xs={12} sm={12} md={6}>
<Card bordered={false}>
<Card variant="borderless">
<Statistic
title="用户总数"
value={overviewData?.userCount || 0}
@@ -160,7 +152,7 @@ const DashboardOverview: React.FC = () => {
</Card>
</Col>
<Col xs={12} sm={12} md={6}>
<Card bordered={false}>
<Card variant="borderless">
<Statistic
title="文件总数"
value={overviewData?.fileCount || 0}
@@ -169,7 +161,7 @@ const DashboardOverview: React.FC = () => {
</Card>
</Col>
<Col xs={12} sm={12} md={6}>
<Card bordered={false}>
<Card variant="borderless">
<Statistic
title="文章总数"
value={overviewData?.articleCount || 0}
@@ -178,7 +170,7 @@ const DashboardOverview: React.FC = () => {
</Card>
</Col>
<Col xs={12} sm={12} md={6}>
<Card bordered={false}>
<Card variant="borderless">
<Statistic
title="今日登录"
value={overviewData?.todayLoginCount || 0}

View File

@@ -24,22 +24,19 @@ import weekday from 'dayjs/plugin/weekday';
import localeData from 'dayjs/plugin/localeData';
import 'dayjs/locale/zh-cn';
import type {
FileLibrary, FileCategory, KnowInfo
KnowInfo
} from '../share/types.ts';
import {
AuditStatus,AuditStatusNameMap,
OssType,
} from '../share/types.ts';
import { getEnumOptions } from './utils.ts';
import {
FileAPI,
UserAPI,
KnowInfoAPI,
type KnowInfoListResponse
} from './api.ts';
} from './api/index.ts';
// 配置 dayjs 插件
@@ -49,7 +46,6 @@ dayjs.extend(localeData);
// 设置 dayjs 语言
dayjs.locale('zh-cn');
const { Title } = Typography;
// 知识库管理页面组件

View File

@@ -1,19 +1,11 @@
import React, { useState, useEffect } from 'react';
import React, { useState } from 'react';
import {
Layout, Menu, Button, Table, Space,
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,
Tree
Button, Space, Drawer,
Select, message,
Card, Spin, Typography,Descriptions,DatePicker,
} from 'antd';
import {
MenuFoldOutlined,
MenuUnfoldOutlined,
AppstoreOutlined,
EnvironmentOutlined,
SearchOutlined,
ClockCircleOutlined,
UserOutlined,
GlobalOutlined
@@ -28,7 +20,8 @@ import type {
MarkerData, LoginLocation, LoginLocationDetail, User
} 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';
const { RangePicker } = DatePicker;

View File

@@ -1,19 +1,17 @@
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 type { TableProps } from 'antd';
import dayjs from 'dayjs';
import 'dayjs/locale/zh-cn';
import { MessageAPI } from './api.ts';
import { UserAPI } from './api.ts';
import { MessageAPI , UserAPI } from './api/index.ts';
import type { UserMessage } from '../share/types.ts';
import { MessageStatusNameMap , MessageStatus} from '../share/types.ts';
export const MessagesPage = () => {
const queryClient = useQueryClient();
const [form] = Form.useForm();
const [selectedRowKeys, setSelectedRowKeys] = useState<React.Key[]>([]);
const [isModalVisible, setIsModalVisible] = useState(false);
const [searchParams, setSearchParams] = useState({
page: 1,

View File

@@ -1,20 +1,14 @@
import React, { useState, useEffect } from 'react';
import React, { useEffect } from 'react';
import {
Layout, Menu, Button, Table, Space,
Button,Space,
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, InputNumber,ColorPicker,
Popover
Card, Spin, Typography,
Switch, Tabs, Alert, InputNumber
} from 'antd';
import {
UploadOutlined,
ReloadOutlined,
SaveOutlined,
BgColorsOutlined
} from '@ant-design/icons';
import { debounce } from 'lodash';
import {
useQuery,
useMutation,
@@ -25,25 +19,20 @@ import weekday from 'dayjs/plugin/weekday';
import localeData from 'dayjs/plugin/localeData';
import 'dayjs/locale/zh-cn';
import type {
FileLibrary, FileCategory, KnowInfo, SystemSetting, SystemSettingValue,
ColorScheme
SystemSetting, SystemSettingValue
} from '../share/types.ts';
import { ThemeMode } from '../share/types.ts';
import {
SystemSettingGroup,
SystemSettingKey,
FontSize,
CompactMode,
AllowedFileType
} from '../share/types.ts';
import { getEnumOptions } from './utils.ts';
import {
SystemAPI,
} from './api.ts';
SystemAPI
} from './api/index.ts';
import { useTheme } from './hooks_sys.tsx';
@@ -241,44 +230,44 @@ export const SettingsPage = () => {
items={Object.values(SystemSettingGroup).map(group => ({
key: group,
label: String(GROUP_TITLES[group]),
children: (
<div>
<Alert
children: (
<div>
<Alert
message={GROUP_DESCRIPTIONS[group]}
type="info"
showIcon
style={{ marginBottom: 24 }}
/>
<Form
form={form}
layout="vertical"
onFinish={handleSubmit}
>
type="info"
showIcon
style={{ marginBottom: 24 }}
/>
<Form
form={form}
layout="vertical"
onFinish={handleSubmit}
>
{settingsData
?.find(g => g.name === group)
?.settings.map(setting => (
<Form.Item
<Form.Item
key={setting.key}
label={setting.description || setting.key}
name={setting.key}
rules={[{ required: true, message: `请输入${setting.description || setting.key}` }]}
>
{renderSettingInput(setting)}
</Form.Item>
</Form.Item>
))}
<Form.Item>
<Button
type="primary"
htmlType="submit"
icon={<SaveOutlined />}
loading={updateSettingsMutation.isPending}
>
</Button>
</Form.Item>
</Form>
</div>
)
<Form.Item>
<Button
type="primary"
htmlType="submit"
icon={<SaveOutlined />}
loading={updateSettingsMutation.isPending}
>
</Button>
</Form.Item>
</Form>
</div>
)
}))}
/>
</Spin>

View File

@@ -25,20 +25,15 @@ import { uploadMinIOWithPolicy,uploadOSSWithPolicy } from '@d8d-appcontainer/api
import type { MinioUploadPolicy, OSSUploadPolicy } from '@d8d-appcontainer/types';
import 'dayjs/locale/zh-cn';
import type {
FileLibrary, FileCategory, KnowInfo
FileLibrary, FileCategory
} from '../share/types.ts';
import {
AuditStatus,AuditStatusNameMap,
OssType,
} from '../share/types.ts';
import { getEnumOptions } from './utils.ts';
import {
FileAPI,
UserAPI,
} from './api.ts';
import { FileAPI , UserAPI } from './api/index.ts';
// 配置 dayjs 插件

View File

@@ -1,53 +1,30 @@
import React, { useState, useEffect } from 'react';
import {
Layout, Menu, Button, Table, Space,
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, InputNumber,ColorPicker,
Popover
Button, Space,
Form, message,
Card, Spin, Typography,
Switch,
Popconfirm, Radio, InputNumber,ColorPicker,
} 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 weekday from 'dayjs/plugin/weekday';
import localeData from 'dayjs/plugin/localeData';
import 'dayjs/locale/zh-cn';
import type {
FileLibrary, FileCategory, KnowInfo, SystemSetting, SystemSettingValue,
ColorScheme
} from '../share/types.ts';
import { ThemeMode } from '../share/types.ts';
import {
SystemSettingGroup,
SystemSettingKey,
FontSize,
CompactMode,
AllowedFileType
} from '../share/types.ts';
import { getEnumOptions } from './utils.ts';
import {
SystemAPI,
} from './api.ts';
import { useTheme } from './hooks_sys.tsx';
import { Uploader } from './components_uploader.tsx';
// 配置 dayjs 插件
dayjs.extend(weekday);