feat: 添加管理端前端 (HMS 基座 React 管理面板)
Some checks failed
Main Merge / backend (push) Has been cancelled
Main Merge / frontend (push) Has been cancelled

- 从 HMS 基座复制 apps/web/ (React + Ant Design + Vite + TypeScript)
- 管理端自动代理 API 到 localhost:3000 (vite.config.ts)
- 更新 scripts/dev.sh 支持三端启动: backend/admin/app
- 登录验证通过, 用户管理/角色权限/审计日志等页面正常
- 添加 .gitignore 排除 node_modules/dist
This commit is contained in:
iven
2026-06-02 10:03:13 +08:00
parent 181bfb1f3e
commit 8111471e93
341 changed files with 72102 additions and 1059 deletions

View File

@@ -0,0 +1,147 @@
import { useEffect, useRef } from 'react';
import { Table, Select, Input, Tag } from 'antd';
import type { ColumnsType, TablePaginationConfig } from 'antd/es/table';
import { listAuditLogs, type AuditLogItem, type AuditLogQuery } from '../../api/auditLogs';
import { listUsers } from '../../api/users';
import { useThemeMode } from '../../hooks/useThemeMode';
import { usePaginatedData } from '../../hooks/usePaginatedData';
const RESOURCE_TYPE_OPTIONS = [
{ value: 'user', label: '用户' }, { value: 'role', label: '角色' },
{ value: 'position', label: '岗位' }, { value: 'organization', label: '组织' },
{ value: 'department', label: '部门' }, { value: 'process_instance', label: '流程实例' },
{ value: 'process_definition', label: '流程定义' }, { value: 'task', label: '流程任务' },
{ value: 'dictionary', label: '字典' }, { value: 'menu', label: '菜单' },
{ value: 'setting', label: '设置' }, { value: 'numbering_rule', label: '编号规则' },
{ value: 'patient', label: '患者' }, { value: 'patient_tag', label: '患者标签' },
{ value: 'patient_family_member', label: '家庭成员' }, { value: 'patient_doctor_relation', label: '医患关系' },
{ value: 'points_transaction', label: '积分流水' }, { value: 'points_product', label: '积分商品' },
{ value: 'points_order', label: '积分订单' }, { value: 'points_rule', label: '积分规则' },
{ value: 'offline_event', label: '线下活动' }, { value: 'offline_event_registration', label: '活动签到' },
];
const ACTION_STYLES: Record<string, { bg: string; color: string; text: string }> = {
create: { bg: '#ECFDF5', color: '#059669', text: '创建' },
update: { bg: '#eff6ff', color: '#2563eb', text: '更新' },
delete: { bg: '#FEF2F2', color: '#dc2626', text: '删除' },
};
function formatDateTime(value: string): string {
return new Date(value).toLocaleString('zh-CN', {
year: 'numeric', month: '2-digit', day: '2-digit',
hour: '2-digit', minute: '2-digit', second: '2-digit',
});
}
export default function AuditLogViewer() {
const isDark = useThemeMode();
const userNameCache = useRef<Record<string, string>>({});
const cacheLoaded = useRef(false);
const { data: logs, total, page, loading, filters, setFilters, refresh } = usePaginatedData<AuditLogItem, AuditLogQuery>(
async (p, pageSize, query) => {
const result = await listAuditLogs({ ...query, page: p, page_size: pageSize });
return { data: result.data, total: result.total };
},
{ pageSize: 20, defaultFilters: { page: 1, page_size: 20 } as unknown as AuditLogQuery, autoFetch: false },
);
// Load user name cache
useEffect(() => {
if (cacheLoaded.current) return;
let cancelled = false;
const loadAllUsers = async () => {
try {
let currentPage = 1;
const pageSize = 100;
let hasMore = true;
while (hasMore && !cancelled) {
const result = await listUsers(currentPage, pageSize);
for (const user of result.data) {
userNameCache.current[user.id] = user.display_name || user.username;
}
hasMore = result.data.length >= pageSize;
currentPage += 1;
}
if (!cancelled) cacheLoaded.current = true;
} catch { /* silent */ }
};
loadAllUsers();
return () => { cancelled = true; };
}, []);
useEffect(() => { refresh(1); }, []);
const handleFilterChange = (field: keyof AuditLogQuery, value: string | undefined) => {
setFilters((prev) => ({ ...prev, [field]: value || undefined, page: 1 }));
};
const columns: ColumnsType<AuditLogItem> = [
{
title: '操作', dataIndex: 'action', key: 'action', width: 100,
render: (action: string) => {
const info = ACTION_STYLES[action] || { bg: '#f8fafc', color: '#475569', text: action };
return <Tag style={{ background: info.bg, border: 'none', color: info.color, fontWeight: 500 }}>{info.text}</Tag>;
},
},
{
title: '资源类型', dataIndex: 'resource_type', key: 'resource_type', width: 120,
render: (v: string) => <Tag style={{ background: isDark ? '#0f172a' : '#f8fafc', border: 'none', color: isDark ? '#CBD5E1' : '#475569' }}>{v}</Tag>,
},
{
title: '资源 ID', dataIndex: 'resource_id', key: 'resource_id', width: 200, ellipsis: true,
render: (v: string) => <span style={{ fontFamily: 'monospace', fontSize: 12, color: isDark ? '#94a3b8' : '#475569' }}>{v}</span>,
},
{
title: '操作用户', dataIndex: 'user_id', key: 'user_id', width: 200, ellipsis: true,
render: (v: string) => {
const name = userNameCache.current[v];
return <span title={v} style={{ fontSize: 13, color: isDark ? '#CBD5E1' : '#334155' }}>{name || v}</span>;
},
},
{
title: '时间', dataIndex: 'created_at', key: 'created_at', width: 180,
render: (value: string) => <span style={{ color: isDark ? '#475569' : '#94a3b8', fontSize: 13 }}>{formatDateTime(value)}</span>,
},
];
return (
<div>
{/* 筛选工具栏 */}
<div style={{
display: 'flex', alignItems: 'center', gap: 12, marginBottom: 16, padding: 12,
background: isDark ? '#111827' : '#FFFFFF', borderRadius: 10,
border: `1px solid ${isDark ? '#0f172a' : '#f8fafc'}`,
}}>
<Select
allowClear placeholder="资源类型" style={{ width: 160 }}
options={RESOURCE_TYPE_OPTIONS}
value={filters.resource_type}
onChange={(value) => handleFilterChange('resource_type', value)}
/>
<Input
allowClear placeholder="操作用户 ID" style={{ width: 240 }}
value={filters.user_id ?? ''}
onChange={(e) => handleFilterChange('user_id', e.target.value)}
/>
<span style={{ fontSize: 13, color: isDark ? '#475569' : '#94a3b8', marginLeft: 'auto' }}> {total} </span>
</div>
{/* 表格 */}
<div style={{
background: isDark ? '#111827' : '#FFFFFF', borderRadius: 12,
border: `1px solid ${isDark ? '#0f172a' : '#f8fafc'}`, overflow: 'hidden',
}}>
<Table
rowKey="id" columns={columns} dataSource={logs} loading={loading}
onChange={(pagination: TablePaginationConfig) => refresh(pagination.current)}
pagination={{
current: page, pageSize: 20, total,
showSizeChanger: true, showTotal: (t) => `${t}`,
}}
scroll={{ x: 900 }}
/>
</div>
</div>
);
}