import React, { useState, useEffect, useCallback } from 'react'; import { type MetaFunction, type LoaderFunctionArgs } from "@remix-run/node"; import { useLoaderData, useSearchParams, Link, useNavigate, useFetcher, useRouteLoaderData } from "@remix-run/react"; import { Button } from '~/components/ui/Button'; import { Card } from '~/components/ui/Card'; import { Tag } from '~/components/ui/Tag'; import { StatusDot } from '~/components/ui/StatusDot'; import rulesStyles from "~/styles/pages/rules_index.css?url"; import type { Rule, RuleType, RulePriority } from '~/models/rule'; import { RULE_TYPE_COLORS, RULE_PRIORITY_LABELS, RULE_PRIORITY_COLORS } from '~/models/rule'; import type { TagColor } from '~/components/ui/Tag'; import { Table } from '~/components/ui/Table'; import { FilterPanel, FilterSelect, SearchFilter } from '~/components/ui/FilterPanel'; import { Pagination } from '~/components/ui/Pagination'; import { messageService } from '~/components/ui/MessageModal'; import { toastService } from '~/components/ui/Toast'; import { getRulesList, deleteRule, getRuleTypes, getRuleGroupsByType, type RuleType as ApiRuleType, type RuleGroup } from '~/api/evaluation_points/rules'; import type { UserRole } from '~/root'; export const links = () => [ { rel: "stylesheet", href: rulesStyles } ]; export const meta: MetaFunction = () => { return [ { title: "中国烟草AI合同及卷宗审核系统 - 评查点列表" }, { name: "rules", content: "管理评查点规则,支持根据类型、规则组和状态进行筛选" }, { name: "keywords", content: "评查点,合同审核,规则管理,中国烟草" } ]; }; // 声明loader返回的数据类型 export type LoaderData = { rules: Rule[]; totalCount: number; currentPage: number; pageSize: number; totalPages: number; ruleTypes: ApiRuleType[]; // 添加评查点类型 initialLoad?: boolean; // 添加初始加载标志 }; // API返回的数据映射到前端模型 interface ApiRule { id: string; code: string; name: string; ruleType: string; groupId: string; groupName: string; priority: string; description: string; isActive: boolean; createdAt: string; updatedAt: string; } interface ActionResponse { result: boolean; message: string; } function mapApiRuleToModel(apiRule: ApiRule): Rule { return { id: apiRule.id, code: apiRule.code, name: apiRule.name, ruleType: apiRule.ruleType as RuleType, // 类型转换 ruleGroupId: apiRule.groupId, groupName: apiRule.groupName, priority: apiRule.priority as RulePriority, // 类型转换 description: apiRule.description, checkMethod: 'automatic', // 默认值 prompt: apiRule.description, // 使用描述作为默认prompt isActive: apiRule.isActive, createdAt: apiRule.createdAt, updatedAt: apiRule.updatedAt }; } export async function loader({ request }: LoaderFunctionArgs) { const url = new URL(request.url); // 从 URL 参数中提取查询条件 const params = { page: parseInt(url.searchParams.get("page") || "1", 10), pageSize: parseInt(url.searchParams.get("pageSize") || "10", 10) }; try { // 获取评查点类型列表,供前端筛选使用 const typeResponse = await getRuleTypes(); if (typeResponse.error) { console.error('获取评查点类型失败:', typeResponse.error); } const ruleTypes = typeResponse.error ? [] : typeResponse.data; // 返回初始空数据,客户端将根据 sessionStorage 中的 reviewType 加载实际数据 return Response.json({ rules: [], totalCount: 0, currentPage: params.page, pageSize: params.pageSize, ruleTypes, initialLoad: true }, { headers: { "Cache-Control": "max-age=60, s-maxage=180" } }); } catch (error) { console.error('加载评查点列表失败:', error); return Response.json({ error: error || '加载评查点列表失败', status: 500 }, { status: 500 }); } } export async function action({ request }: LoaderFunctionArgs) { const formData = await request.formData(); const _action = formData.get('_action'); const ruleId = formData.get('ruleId'); if (!ruleId) { return Response.json({ result: false, message: "缺少评查点ID" }, { status: 400 }); } try { if (_action === 'delete') { // 调用API删除评查点 // console.log(`删除评查点 ${ruleId}`); const deleteResponse = await deleteRule(ruleId as string); if (deleteResponse.error) { return Response.json({ result: false, message: deleteResponse.error }, { status: deleteResponse.status || 500 }); } return Response.json({ result: true, message: "评查点删除成功" }, { status: 200 }); } } catch (error) { console.error('操作评查点失败:', error); return Response.json({ result: false, message: error instanceof Error ? error.message : "操作失败" }, { status: 500 }); } } // 规则优先级的描述标签映射 const priorityLabels = { 'high': '高', 'medium': '中', 'low': '低' }; export default function RulesIndex() { const loaderData = useLoaderData(); const rootData = useRouteLoaderData("root") as { userRole: UserRole }; const { rules: initialRules, totalCount: initialTotalCount, currentPage, pageSize, ruleTypes: initialRuleTypes, initialLoad } = loaderData; const [searchParams, setSearchParams] = useSearchParams(); const navigate = useNavigate(); const fetcher = useFetcher(); // 状态管理 const [ruleGroups, setRuleGroups] = useState([]); const [loadingGroups, setLoadingGroups] = useState(false); const [reviewType, setReviewType] = useState(null); const [loading, setLoading] = useState(false); const [filteredRules, setFilteredRules] = useState(initialRules); const [filteredTotalCount, setFilteredTotalCount] = useState(initialTotalCount); const [ruleTypes, setRuleTypes] = useState(initialRuleTypes); // 获取当前的ruleType值 const ruleTypeParam = searchParams.get('ruleType'); // 判断是否禁用规则组选择 const isRuleGroupSelectDisabled = loadingGroups || !ruleTypeParam || ruleGroups.length === 0; // 检查用户是否为开发者角色 const userRole = rootData?.userRole || 'common'; const isDeveloper = userRole === 'developer'; // 在组件渲染时初始化状态 useEffect(() => { setFilteredRules(initialRules); setFilteredTotalCount(initialTotalCount); setRuleTypes(initialRuleTypes); }, [initialRules, initialTotalCount, initialRuleTypes]); // 在组件挂载时从 sessionStorage 获取 reviewType useEffect(() => { try { if (typeof window !== 'undefined') { const storedReviewType = sessionStorage.getItem('reviewType'); if (storedReviewType !== reviewType) { setReviewType(storedReviewType); if (initialLoad) { // 如果是初始加载,立即加载数据 // 不需要更新 URL 参数,因为下一步的 loadRulesData 会处理 } else { // 如果不是初始加载,更新 URL 参数 const newParams = new URLSearchParams(searchParams); newParams.set('page', '1'); // 回到第一页 setSearchParams(newParams); } } } } catch (error) { console.error('获取 sessionStorage 中的 reviewType 失败:', error); } }, [reviewType, searchParams, setSearchParams, initialLoad]); // 使用useEffect监听loaderData.error变化并显示Toast useEffect(() => { if(loaderData.error) { toastService.error(loaderData.error); }else if(loaderData.ruleTypes.length === 0){ toastService.error("评查点类型数据为空"); } }, [loaderData.error,loaderData.ruleTypes]); // 当评查点类型变化时,加载对应的规则组 useEffect(() => { // 如果选择了"全部"或未选择,则清空规则组 if (!ruleTypeParam || ruleTypeParam === 'all') { setRuleGroups([]); return; } // 加载当前类型的规则组 const loadRuleGroups = async () => { setLoadingGroups(true); try { const response = await getRuleGroupsByType(ruleTypeParam); if (response.data) { setRuleGroups(response.data); } else if (response.error) { console.error('加载规则组失败:', response.error); setRuleGroups([]); } } catch (error) { console.error('加载规则组出错:', error); setRuleGroups([]); } finally { setLoadingGroups(false); } }; loadRuleGroups(); }, [ruleTypeParam]); // 使用useEffect监听fetcher状态变化并显示Toast useEffect(() => { if (fetcher.data && fetcher.state === 'idle') { if (fetcher.data.result) { toastService.success(fetcher.data.message); } else if (!fetcher.data.result) { if(fetcher.data.message.includes("evaluation_results_evaluation_point_id_fkey")) { toastService.error('对表evaluation_points进行更新或删除违反了表evaluation results上的外键约束evaluations results_evaluation _point_id_fkey'); } else { toastService.error(fetcher.data.message); } } } }, [fetcher.data,fetcher.state]); // 添加客户端数据加载函数 const loadRulesData = useCallback(async () => { if (!reviewType) return; setLoading(true); try { // 构建查询参数 const queryParams = { ruleType: ruleTypeParam || undefined, groupId: searchParams.get('groupId') || undefined, isActive: searchParams.get('isActive') ? searchParams.get('isActive') === 'true' : undefined, keyword: searchParams.get('keyword') || undefined, page: currentPage, pageSize, reviewType }; // 调用 API 获取数据 const response = await getRulesList(queryParams); if (response.data) { const apiRules = response.data.rules || []; const total = response.data.totalCount || 0; const mappedRules = apiRules.map((apiRule: ApiRule) => mapApiRuleToModel(apiRule)); setFilteredRules(mappedRules); setFilteredTotalCount(total); } } catch (error) { console.error('客户端加载评查点列表失败:', error); toastService.error('加载评查点列表失败'); } finally { setLoading(false); } }, [reviewType, ruleTypeParam, searchParams, currentPage, pageSize]); // 添加加载评查点类型的函数 const loadRuleTypes = useCallback(async () => { if (!reviewType) return; try { // 调用 API 获取评查点类型,传递 reviewType 参数 const response = await getRuleTypes(reviewType); if (response.data) { const typesData = response.data; // 这里可以添加类型过滤的逻辑,但实际上 API 中已经根据 reviewType 进行了过滤 // 更新状态 const updatedTypes = typesData; // 替换掉 loaderData 中的 ruleTypes setRuleTypes(updatedTypes); } } catch (error) { console.error('加载评查点类型失败:', error); toastService.error('加载评查点类型失败'); } }, [reviewType]); // 在 reviewType 变化时加载评查点类型 useEffect(() => { if (reviewType) { loadRuleTypes(); } }, [reviewType, loadRuleTypes]); // 在 reviewType 变化或其他参数变化时加载数据 useEffect(() => { if (reviewType) { loadRulesData(); } }, [reviewType, loadRulesData]); // 在初始加载完成后,如果有 reviewType,立即加载数据 useEffect(() => { if (initialLoad && reviewType) { loadRuleTypes(); loadRulesData(); } }, [initialLoad, reviewType, loadRuleTypes, loadRulesData]); // 筛选评查点 const handleFilterChange = (e: React.ChangeEvent) => { const { name, value } = e.target; const newParams = new URLSearchParams(searchParams); // 如果是规则组选择,但是当前应该被禁用,则不处理 if (name === 'groupId' && isRuleGroupSelectDisabled) { return; } if (value) { newParams.set(name, value); // 如果是评查点类型变更,清空规则组选择 if (name === 'ruleType') { newParams.delete('groupId'); // 如果选择了"全部"或空值,也清空规则组选择 if (value === '' || value === 'all') { setRuleGroups([]); } } } else { newParams.delete(name); // 如果清除评查点类型,也清除规则组 if (name === 'ruleType') { newParams.delete('groupId'); setRuleGroups([]); } } // 切换筛选条件时,重置到第一页 newParams.set('page', '1'); setSearchParams(newParams); }; // 搜索评查点 const handleSearch = (keyword: string) => { const newParams = new URLSearchParams(searchParams); if (keyword) { newParams.set('keyword', keyword); } else { newParams.delete('keyword'); } // 搜索时,重置到第一页 newParams.set('page', '1'); setSearchParams(newParams); }; // 删除评查点 const handleDeleteClick = (rule: Rule) => { messageService.show({ title: "确认删除", message: `确认删除评查点【${rule.name}】吗?`, type: "warning", confirmText: "删除", cancelText: "取消", onConfirm: () => { const form = new FormData(); form.append("_action", "delete"); form.append("ruleId", rule.id); fetcher.submit(form, { method: "post" }); } }); }; // 复制评查点 const handleCopy = (rule: Rule) => { navigate(`/rules-new?id=${rule.id}&mode=copy`); }; const handlePageChange = (page: number) => { const newParams = new URLSearchParams(searchParams); newParams.set('page', page.toString()); setSearchParams(newParams); }; // 处理每页条数变化 const handlePageSizeChange = (size: number) => { const newParams = new URLSearchParams(searchParams); newParams.set('pageSize', size.toString()); newParams.set('page', '1'); // 更改每页条数时,重置到第一页 setSearchParams(newParams); }; // 处理重置筛选 const handleReset = () => { const input = document.querySelector('input[placeholder="输入评查点名称或编码"]'); if (input) { (input as HTMLInputElement).value = ''; } // 保留reviewType的过滤条件,只重置其他条件 const newParams = new URLSearchParams(); setSearchParams(newParams); }; // 定义表格列配置 const columns = [ { title: "评查点编码", dataIndex: "code" as keyof Rule, key: "code", align: "left" as const, width: "20%", className: "whitespace-normal break-all", render: (value: string) => (
{value}
) }, { title: "评查点名称", dataIndex: "name" as keyof Rule, key: "name", align: "left" as const, width: "20%" }, { title: "评查点类型", key: "ruleType", align: "left" as const, width: "12%", render: (_: unknown, record: Rule) => { const typeColor = RULE_TYPE_COLORS[record.ruleType] as TagColor; return ( record.ruleType ? {record.ruleType} : null ); } }, { title: "所属规则组", dataIndex: "groupName" as keyof Rule, key: "groupName", align: "left" as const, width: "10%" }, { title: "优先级", key: "priority", align: "left" as const, width: "8%", render: (_: unknown, record: Rule) => { const priorityColor = RULE_PRIORITY_COLORS[record.priority] as TagColor; return ( {priorityLabels[record.priority as keyof typeof priorityLabels] || RULE_PRIORITY_LABELS[record.priority]} ); } }, { title: "状态", key: "isActive", align: "left" as const, width: "8%", render: (_: unknown, record: Rule) => ( ) }, { title: "创建时间", dataIndex: "createdAt" as keyof Rule, key: "createdAt", align: "left" as const, width: "10%" }, { title: "操作", key: "operation", align: "left" as const, width: "10%", render: (_: unknown, record: Rule) => (
{isDeveloper ? ( // 开发者可以看到编辑、复制、删除 <> 编辑 ) : ( // 普通用户只能查看 查看 )}
) } ]; return (
{/* 页面头部 */}

评查点管理

{isDeveloper && ( )}
{/* 筛选区域 */} } > ({ value: type.id, label: type.name })) ]} onChange={handleFilterChange} className="mr-3 w-[15%]" /> ({ value: group.id, label: group.name })) ]} onChange={handleFilterChange} className={`mr-3 w-[20%] ${isRuleGroupSelectDisabled ? 'opacity-50' : ''}`} /> {/* 评查点列表 - 使用Table组件 */}
0 ? filteredRules : initialRules} rowKey="id" emptyText="暂无评查点数据" className="rules-table" /> {/* 分页 */} {filteredTotalCount > 0 && ( 0 ? filteredTotalCount : initialTotalCount} pageSize={pageSize} onChange={handlePageChange} onPageSizeChange={handlePageSizeChange} showTotal={true} showPageSizeChanger={true} pageSizeOptions={[10, 20, 30, 50]} /> )} ); } // 错误边界 export function ErrorBoundary() { return (

出错了

加载评查点列表时发生错误。请稍后再试,或联系管理员。

); }