fix: restore reviews detail layout and leaudit data wiring

This commit is contained in:
wren
2026-05-06 17:31:48 +08:00
parent 63bf3f56ce
commit 796ce90e32
8 changed files with 1652 additions and 607 deletions
+1
View File
@@ -92,6 +92,7 @@ interface StatsData {
success: number;
warning: number;
error: number;
notApplicable?: number;
score: number;
}
+30 -3
View File
@@ -82,6 +82,16 @@ export interface CharPosition {
score: number; // OCR识别置信度
}
export interface PdfBboxHighlight {
fieldKey: string;
bbox: [number, number, number, number];
pageBox: [number, number, number, number];
pageNum?: number;
page?: number;
confidence?: number;
matchMethod?: string;
}
/**
* 评查点类型定义
* 用于展示单个评查结果
@@ -98,7 +108,7 @@ export interface ReviewPoint {
title: string;
groupName: string;
status: string;
content: Record<string, { page?: number | string, value?: object }>;
content: Record<string, unknown>;
suggestion: string;
needsHumanReview?: boolean;
humanReviewNote?: string;
@@ -124,6 +134,7 @@ export interface ReviewPoint {
failMessage?: string;
passMessage?: string;
evaluationConfig?: {
confidence?: number;
rules?: Array<{
type: string;
config?: {
@@ -140,7 +151,22 @@ export interface ReviewPoint {
res?: boolean;
config: Record<string, unknown>;
}>;
skip_reason?: string;
stages?: Array<Record<string, unknown>>;
[key: string]: unknown;
};
fieldPositions?: Record<string, {
bbox?: [number, number, number, number];
page_box?: [number, number, number, number];
page_num?: number;
confidence?: number;
match_method?: string;
}>;
confidence?: number;
ruleStatus?: string;
skipReason?: string;
remediation?: unknown;
riskLevel?: string;
}
// 统计数据类型
@@ -149,6 +175,7 @@ interface Statistics {
success: number;
warning: number;
error: number;
notApplicable?: number;
score: number;
}
@@ -194,8 +221,8 @@ interface EvaluationSummary {
interface ReviewPointsListProps {
reviewPoints: ReviewPoint[];
statistics: Statistics;
activeReviewPointResultId: string | null;
onReviewPointSelect: (id: string, page?: number, charPositions?: CharPosition[], value?: string) => void;
activeReviewPointResultId: string | number | null;
onReviewPointSelect: (id: string | number, page?: number, charPositions?: CharPosition[], value?: string) => void;
onStatusChange: (id: string, editAuditStatusId: string | number, status: string, message: string) => void;
fileFormat?: string; // 文档格式类型(PDF、DOCX等)
onAiSuggestionReplace?: (searchText: string, replaceText: string, pageNumber: number) => void; // AI建议替换回调
+2 -2
View File
@@ -6,7 +6,7 @@ export { FileInfo } from './FileInfo';
export { ReviewTabs } from './ReviewTabs';
export { FilePreview } from './FilePreview';
export { ReviewPointsList } from './ReviewPointsList';
export type { ReviewPoint } from './ReviewPointsList';
export type { ReviewPoint, PdfBboxHighlight } from './ReviewPointsList';
export { AIAnalysis } from './AIAnalysis';
export { FileDetails } from './FileDetails';
export { Comparison } from './Comparison';
@@ -15,4 +15,4 @@ export { Comparison } from './Comparison';
export { RulesDirectory } from './leftColumn/RulesDirectory';
export { DetailPanel } from './rightColumn/DetailPanel';
export { ReviewPointDetailCard } from './rightColumn/ReviewPointDetailCard';
export { FileInfoPanel } from './rightColumn/FileInfoPanel';
export { FileInfoPanel } from './rightColumn/FileInfoPanel';
@@ -1,8 +1,8 @@
/**
* 左栏 · 规则目录
* 示文件、分数进度、搜索评查点分组列表
* 左栏 / 规则目录
* 示文件信息、分数进度、搜索评查点列表
*/
import { useState, useMemo } from 'react';
import { useMemo, useState } from 'react';
import type { ReviewPoint } from '../ReviewPointsList';
interface Statistics {
@@ -24,14 +24,17 @@ interface RulesDirectoryProps {
}
type PointStatus = 'pass' | 'warn' | 'fail' | 'skipped';
type StatusFilter = 'all' | PointStatus;
function classifyPoint(p: ReviewPoint): PointStatus {
if (p.status === 'notApplicable' || p.status === 'not_applicable') return 'skipped';
if (p.result === true || (p.result === undefined && p.status === 'success')) return 'pass';
if (p.result === false) {
if (p.status === 'error') return 'fail';
if (p.status === 'warning' || p.status === 'info') return 'warn';
function classifyPoint(point: ReviewPoint): PointStatus {
if (point.status === 'notApplicable' || point.status === 'not_applicable') return 'skipped';
if (point.result === true || (point.result === undefined && point.status === 'success')) return 'pass';
if (point.result === false) {
if (point.status === 'error') return 'fail';
if (point.status === 'warning' || point.status === 'info') return 'warn';
}
if (point.status === 'error') return 'fail';
if (point.status === 'warning' || point.status === 'info') return 'warn';
return 'pass';
}
@@ -42,6 +45,53 @@ const STATUS_ICON: Record<PointStatus, { icon: string; color: string }> = {
skipped: { icon: 'ri-forbid-2-line', color: 'text-slate-400' },
};
const FILTER_CHIPS: Array<{
key: StatusFilter;
label: string;
getClassName: (active: boolean) => string;
}> = [
{
key: 'all',
label: '全部',
getClassName: (active) =>
active
? 'bg-slate-700 text-white border-slate-700'
: 'bg-white text-slate-600 border-slate-200 hover:bg-slate-50',
},
{
key: 'warn',
label: '提醒',
getClassName: (active) =>
active
? 'bg-amber-500 text-white border-amber-500'
: 'bg-white text-amber-700 border-amber-200 hover:bg-amber-50',
},
{
key: 'fail',
label: '问题',
getClassName: (active) =>
active
? 'bg-red-500 text-white border-red-500'
: 'bg-white text-red-700 border-red-200 hover:bg-red-50',
},
{
key: 'pass',
label: '通过',
getClassName: (active) =>
active
? 'bg-emerald-500 text-white border-emerald-500'
: 'bg-white text-emerald-700 border-emerald-200 hover:bg-emerald-50',
},
{
key: 'skipped',
label: '跳过',
getClassName: (active) =>
active
? 'bg-slate-500 text-white border-slate-500'
: 'bg-white text-slate-600 border-slate-200 hover:bg-slate-100',
},
];
function RuleListItem({
point,
isActive,
@@ -53,33 +103,28 @@ function RuleListItem({
showCategory?: boolean;
onClick: () => void;
}) {
const cls = classifyPoint(point);
const s = STATUS_ICON[cls];
const status = classifyPoint(point);
const statusMeta = STATUS_ICON[status];
const trailingLabel = showCategory ? point.groupName : point.pointCode || point.pointId || '';
return (
<button
type="button"
className={`rule-item relative w-full py-2 px-3 text-left transition ${
isActive ? 'bg-[rgba(0,104,74,0.08)]' : 'hover:bg-slate-50'
isActive ? 'bg-[#e8f3ef]' : 'hover:bg-slate-50'
}`}
onClick={onClick}
>
{isActive && (
<span className="absolute left-0 top-2 bottom-2 w-0.5 bg-[#00684a] rounded-r" />
)}
{isActive && <span className="absolute left-0 top-2 bottom-2 w-0.5 bg-[#00684a] rounded-r" />}
<div className="flex items-center gap-2 min-w-0">
<i className={`${s.icon} ${s.color} shrink-0 text-[14px]`} />
<i className={`${statusMeta.icon} ${statusMeta.color} shrink-0 text-[14px]`} />
<span
className={`text-[12.5px] text-slate-800 truncate flex-1 ${
isActive ? 'font-semibold' : ''
}`}
className={`text-[12.5px] text-slate-800 truncate flex-1 ${isActive ? 'font-semibold' : ''}`}
>
{point.pointName}
</span>
{showCategory && (
<span className="shrink-0 text-[10px] text-slate-400">
{point.groupName}
</span>
{trailingLabel && (
<span className="shrink-0 text-[10px] text-slate-400 font-mono">{trailingLabel}</span>
)}
</div>
</button>
@@ -95,60 +140,76 @@ export function RulesDirectory({
onBack,
}: RulesDirectoryProps) {
const [searchText, setSearchText] = useState('');
const [statusFilter, setStatusFilter] = useState<StatusFilter>('all');
const [passOpen, setPassOpen] = useState(true);
const [openCategories, setOpenCategories] = useState<Set<string>>(new Set());
const [openCategories, setOpenCategories] = useState<Set<string>>(
() => new Set(reviewPoints.map((point) => point.groupName).filter(Boolean))
);
const q = searchText.toLowerCase();
const matchSearch = (p: ReviewPoint) =>
const matchSearch = (point: ReviewPoint) =>
!q ||
(p.pointName?.toLowerCase().includes(q)) ||
(p.pointCode?.toLowerCase().includes(q)) ||
(p.groupName?.toLowerCase().includes(q));
point.pointName?.toLowerCase().includes(q) ||
String(point.pointId ?? '').toLowerCase().includes(q) ||
String(point.pointCode ?? '').toLowerCase().includes(q) ||
point.groupName?.toLowerCase().includes(q);
const { needAttention, passed, passedByGroup } = useMemo(() => {
const filtered = reviewPoints.filter(matchSearch);
const needAttention = filtered.filter(
(p) => classifyPoint(p) !== 'pass' && classifyPoint(p) !== 'skipped'
const matchStatus = (point: ReviewPoint) =>
statusFilter === 'all' || classifyPoint(point) === statusFilter;
const statusCounts = useMemo(() => {
return reviewPoints.reduce<Record<StatusFilter, number>>(
(acc, point) => {
const status = classifyPoint(point);
acc.all += 1;
acc[status] += 1;
return acc;
},
{ all: 0, pass: 0, warn: 0, fail: 0, skipped: 0 }
);
const passed = filtered.filter((p) => classifyPoint(p) === 'pass');
}, [reviewPoints]);
const { needAttention, passed, passedByGroup, hasSearchResult } = useMemo(() => {
const filtered = reviewPoints.filter((point) => matchSearch(point) && matchStatus(point));
const needAttention = filtered.filter((point) => classifyPoint(point) !== 'pass');
const passed = filtered.filter((point) => classifyPoint(point) === 'pass');
const passedByGroup: Record<string, ReviewPoint[]> = {};
passed.forEach((p) => {
const key = p.groupName || '未分组';
(passedByGroup[key] = passedByGroup[key] || []).push(p);
passed.forEach((point) => {
const key = point.groupName || '未分组';
(passedByGroup[key] = passedByGroup[key] || []).push(point);
});
return { needAttention, passed, passedByGroup };
}, [reviewPoints, searchText]);
return {
needAttention,
passed,
passedByGroup,
hasSearchResult: filtered.length > 0,
};
}, [reviewPoints, q, statusFilter]);
// 计算进度条百分比
const total = statistics.total || reviewPoints.length;
const passPct = total > 0 ? (statistics.success / total) * 100 : 0;
const warnPct = total > 0 ? (statistics.warning / total) * 100 : 0;
const errPct = total > 0 ? (statistics.error / total) * 100 : 0;
const naPct =
total > 0
? ((statistics.notApplicable || 0) / total) * 100
: 0;
const naPct = total > 0 ? ((statistics.notApplicable || 0) / total) * 100 : 0;
const attentionCount = needAttention.length;
const passedCount = passed.length;
const attentionCount = reviewPoints.filter((point) => classifyPoint(point) !== 'pass').length;
const passedCount = reviewPoints.filter((point) => classifyPoint(point) === 'pass').length;
const toggleCategory = (cat: string) => {
const toggleCategory = (category: string) => {
setOpenCategories((prev) => {
const next = new Set(prev);
if (next.has(cat)) next.delete(cat);
else next.add(cat);
if (next.has(category)) next.delete(category);
else next.add(category);
return next;
});
};
return (
<aside className="border-r border-slate-200 bg-white flex flex-col min-h-0">
{/* 顶部区域: 文件名 + 分数 + 进度条 */}
<div className="shrink-0 px-3 py-3 border-b border-slate-100 space-y-2">
{/* 文件名 */}
<div className="flex items-center gap-1.5 text-[11px] text-slate-500">
<button
type="button"
@@ -162,7 +223,6 @@ export function RulesDirectory({
<span className="flex-1 break-all">{fileName}</span>
</div>
{/* 分数 + 进度条 */}
<div className="flex items-center gap-2">
<span className="text-[22px] font-semibold text-slate-900 tabular-nums leading-none">
{Math.round(statistics.score)}
@@ -170,41 +230,26 @@ export function RulesDirectory({
<span className="text-[10.5px] text-slate-400">/100</span>
<div className="flex-1 h-1.5 rounded-full overflow-hidden bg-slate-100 ml-1">
<div className="flex h-full">
<div
className="bg-emerald-500"
style={{ width: `${passPct}%` }}
/>
<div
className="bg-amber-400"
style={{ width: `${warnPct}%` }}
/>
<div
className="bg-red-500"
style={{ width: `${errPct}%` }}
/>
<div
className="bg-slate-300"
style={{ width: `${naPct}%` }}
/>
<div className="bg-emerald-500" style={{ width: `${passPct}%` }} />
<div className="bg-amber-400" style={{ width: `${warnPct}%` }} />
<div className="bg-red-500" style={{ width: `${errPct}%` }} />
<div className="bg-slate-300" style={{ width: `${naPct}%` }} />
</div>
</div>
</div>
{/* 状态摘要 */}
<div className="flex items-center justify-between text-[11px]">
<span className="inline-flex items-center gap-1 text-orange-700 bg-orange-50 border border-orange-200 rounded px-1.5 py-0.5 font-medium">
<i className="ri-focus-3-line" />{' '}
<i className="ri-focus-3-line" />
<span className="font-mono">{attentionCount}</span>
</span>
<span className="text-slate-400">
<span className="font-mono text-slate-500">{passedCount}</span> /{' '}
{total}
<span className="font-mono text-slate-500">{passedCount}</span> / {total}
</span>
</div>
</div>
{/* 搜索框 */}
<div className="shrink-0 p-2.5 border-b border-slate-100">
<div className="shrink-0 p-2.5 border-b border-slate-100 space-y-2">
<div className="relative">
<i className="ri-search-line absolute left-2.5 top-1/2 -translate-y-1/2 text-slate-400 text-[14px]" />
<input
@@ -214,33 +259,46 @@ export function RulesDirectory({
onChange={(e) => setSearchText(e.target.value)}
/>
</div>
<div className="flex flex-wrap gap-1">
{FILTER_CHIPS.map((filter) => {
const isActive = statusFilter === filter.key;
const count = statusCounts[filter.key];
return (
<button
key={filter.key}
type="button"
className={`inline-flex items-center gap-1 px-2 h-6 rounded border text-[11px] font-medium transition ${filter.getClassName(
isActive
)}`}
onClick={() => setStatusFilter(filter.key)}
>
<span>{filter.label}</span>
<span className="font-mono text-[10px] opacity-80">{count}</span>
</button>
);
})}
</div>
</div>
{/* 评查点列表 */}
<div className="flex-1 min-h-0 overflow-y-auto scroll-slim py-1">
{/* 需关注 */}
{needAttention.length > 0 ? (
{needAttention.length > 0 && (
<div>
{needAttention.map((p) => (
<div className="px-3 pt-1 pb-1 text-[10px] font-medium uppercase tracking-[0.18em] text-slate-400">
</div>
{needAttention.map((point) => (
<RuleListItem
key={p.id}
point={p}
isActive={p.id === activeReviewPointResultId}
key={point.id}
point={point}
isActive={point.id === activeReviewPointResultId}
showCategory
onClick={() => onRuleSelect(p.id)}
onClick={() => onRuleSelect(point.id)}
/>
))}
</div>
) : (
<div className="text-center py-6 text-[12px] text-slate-400">
<i className="ri-check-double-line text-2xl text-emerald-400" />
<div className="mt-1">
{q ? '没有匹配' : '已全部处理'}
</div>
</div>
)}
{/* 已通过 */}
{passed.length > 0 && (
<div className="border-t border-slate-200 mt-2">
<button
@@ -255,40 +313,38 @@ export function RulesDirectory({
/>
<i className="ri-checkbox-circle-fill text-emerald-500 text-[14px]" />
<span className="text-[12px] text-slate-600"></span>
<span className="ml-auto font-mono text-[11px] text-slate-400">
{passed.length}
</span>
<span className="ml-auto font-mono text-[11px] text-slate-400">{passed.length}</span>
</button>
{passOpen &&
Object.entries(passedByGroup).map(([cat, points]) => {
const catOpen = openCategories.has(cat);
Object.entries(passedByGroup).map(([category, points]) => {
const isOpen = openCategories.has(category);
return (
<div key={cat}>
<div key={category}>
<button
type="button"
className="w-full flex items-center gap-1.5 px-3 py-1.5 bg-slate-50/40 hover:bg-slate-100 text-left"
onClick={() => toggleCategory(cat)}
onClick={() => toggleCategory(category)}
>
<i
className={`ri-arrow-right-s-line text-slate-400 text-[11px] transition-transform ${
catOpen ? 'rotate-90' : ''
isOpen ? 'rotate-90' : ''
}`}
/>
<span className="text-[10px] font-medium text-slate-500 uppercase tracking-wider">
{cat}
{category}
</span>
<span className="ml-auto font-mono text-[10px] text-slate-400">
{points.length}
</span>
</button>
{catOpen &&
points.map((p) => (
{isOpen &&
points.map((point) => (
<RuleListItem
key={p.id}
point={p}
isActive={p.id === activeReviewPointResultId}
onClick={() => onRuleSelect(p.id)}
key={point.id}
point={point}
isActive={point.id === activeReviewPointResultId}
onClick={() => onRuleSelect(point.id)}
/>
))}
</div>
@@ -296,6 +352,17 @@ export function RulesDirectory({
})}
</div>
)}
{!hasSearchResult && (
<div className="text-center py-8 text-[12px] text-slate-400">
<i
className={`text-2xl ${
q ? 'ri-search-eye-line text-slate-300' : 'ri-check-double-line text-emerald-400'
}`}
/>
<div className="mt-1">{q ? '没有匹配结果' : '当前筛选下暂无规则'}</div>
</div>
)}
</div>
</aside>
);
@@ -11,7 +11,7 @@
import { useState, useEffect, useMemo, useRef, useCallback } from 'react';
import { Document, Page, pdfjs } from 'react-pdf';
import { toastService } from '~/components/ui/Toast';
import type { ReviewPoint } from '../ReviewPointsList';
import type { ReviewPoint, PdfBboxHighlight } from '../ReviewPointsList';
import 'react-pdf/dist/esm/Page/TextLayer.css';
import 'react-pdf/dist/esm/Page/AnnotationLayer.css';
@@ -33,6 +33,7 @@ interface PdfPreviewProps {
filePath: string;
targetPage?: number;
charPositions?: Array<{ box: number[][]; char: string; score: number }>;
bboxHighlight?: PdfBboxHighlight;
isStructuredView?: boolean;
activeReviewPointResultId?: string | null;
pageOffset?: number;
@@ -43,11 +44,20 @@ interface PdfPreviewProps {
reviewPoints?: ReviewPoint[];
}
const THUMB_WIDTH = 112;
const THUMB_ESTIMATED_HEIGHT = 210;
const THUMB_OVERSCAN = 3;
const MAIN_PAGE_MAX_DEVICE_PIXEL_RATIO = 1.5;
// ============================================================
// ReviewPoint → 状态映射
// ============================================================
const STATUS_ORDER: Record<WorstStatus, number> = { fail: 0, warn: 1, pending: 2, pass: 3 };
function getContentItemPage(item: ReviewPoint['content'][string]): number | string | undefined {
return typeof item === 'string' ? undefined : item?.page;
}
const STATUS_BADGE: Record<WorstStatus, { cls: string; ic: string }> = {
fail: { cls: 'bg-red-500', ic: 'ri-close-circle-fill' },
warn: { cls: 'bg-amber-500', ic: 'ri-lightbulb-flash-fill' },
@@ -77,7 +87,7 @@ function getPointPages(p: ReviewPoint): number[] {
if (Number.isFinite(n) && n > 0) set.add(n);
};
if (p.contentPage) Object.values(p.contentPage).forEach(addMaybe);
if (p.content) Object.values(p.content).forEach(v => addMaybe(v?.page));
if (p.content) Object.values(p.content).forEach(v => addMaybe(getContentItemPage(v)));
return [...set].sort((a, b) => a - b);
}
@@ -95,12 +105,20 @@ function getFieldsOnPage(p: ReviewPoint, page: number): string[] {
}
if (!fields.length && p.content) {
Object.entries(p.content).forEach(([k, v]) => {
if (matches(v?.page)) fields.push(k);
if (matches(getContentItemPage(v))) fields.push(k);
});
}
return fields;
}
function clamp(value: number, min: number, max: number): number {
return Math.min(max, Math.max(min, value));
}
function isValidQuad(value: unknown): value is [number, number, number, number] {
return Array.isArray(value) && value.length === 4 && value.every(item => typeof item === 'number' && Number.isFinite(item));
}
// ============================================================
// 组件
// ============================================================
@@ -108,6 +126,7 @@ export function PdfPreviewTest({
filePath,
targetPage,
charPositions,
bboxHighlight,
isStructuredView = false,
activeReviewPointResultId,
pageOffset = 0,
@@ -115,63 +134,55 @@ export function PdfPreviewTest({
onZoomChange,
reviewPoints,
}: PdfPreviewProps) {
// ---------- 基础状态 ----------
const [numPages, setNumPages] = useState<number | null>(null);
const [currentPage, setCurrentPage] = useState<number>(1);
const [zoomLevel, setZoomLevel] = useState(100);
const [loadError, setLoadError] = useState<string | null>(null);
// 略缩图面板
const [showThumbs, setShowThumbs] = useState(true);
const [thumbMode, setThumbMode] = useState<'filtered' | 'all'>('filtered');
// 坐标校准(保留主视口高亮逻辑)
const [coordinateScale, setCoordinateScale] = useState(0.83);
const [isScaleAutoCalculated, setIsScaleAutoCalculated] = useState(false);
// 页码跳转输入
const [pageInputValue, setPageInputValue] = useState<string>('');
const [pageRenderTick, setPageRenderTick] = useState(0);
const [thumbsScrollTop, setThumbsScrollTop] = useState(0);
const [thumbsViewportHeight, setThumbsViewportHeight] = useState(0);
const viewportRef = useRef<HTMLDivElement>(null);
const thumbsPanelRef = useRef<HTMLDivElement>(null);
// ---------- 派生数据 ----------
const fileUrl = useMemo(
() => `/api/pdf-proxy?path=${encodeURIComponent(filePath)}`,
[filePath],
);
const fileUrl = useMemo(() => `/api/pdf-proxy?path=${encodeURIComponent(filePath)}`, [filePath]);
const activePoint = useMemo<ReviewPoint | undefined>(
() => reviewPoints?.find(p => p.id === activeReviewPointResultId),
[reviewPoints, activeReviewPointResultId],
);
// 当前规则涉及的页(字段级,带 pageOffset
const rulePages = useMemo<number[]>(() => {
if (!activePoint) return [];
return getPointPages(activePoint).map(p => p + pageOffset);
return getPointPages(activePoint).map(page => page + pageOffset);
}, [activePoint, pageOffset]);
// 每页状态聚合
const pageStatusMap = useMemo<Map<number, PageAgg>>(() => {
const m = new Map<number, PageAgg>();
if (!reviewPoints?.length) return m;
reviewPoints.forEach(p => {
const cls = classifyReviewPoint(p);
if (cls === 'skipped') return;
const pages = getPointPages(p).map(x => x + pageOffset);
pages.forEach(pg => {
const cur = m.get(pg) || { worst: 'pass', count: 0, issues: 0 };
if (STATUS_ORDER[cls] < STATUS_ORDER[cur.worst]) cur.worst = cls;
cur.count += 1;
if (cls !== 'pass') cur.issues += 1;
m.set(pg, cur);
const map = new Map<number, PageAgg>();
if (!reviewPoints?.length) return map;
reviewPoints.forEach(point => {
const status = classifyReviewPoint(point);
if (status === 'skipped') return;
getPointPages(point).map(page => page + pageOffset).forEach(page => {
const current = map.get(page) || { worst: 'pass', count: 0, issues: 0 };
if (STATUS_ORDER[status] < STATUS_ORDER[current.worst]) current.worst = status;
current.count += 1;
if (status !== 'pass') current.issues += 1;
map.set(page, current);
});
});
return m;
return map;
}, [reviewPoints, pageOffset]);
// 当前高亮标签(工具栏右侧)
const highlightLabel = useMemo(() => {
if (!activePoint) return null;
const code = activePoint.pointCode || activePoint.id;
@@ -179,7 +190,42 @@ export function PdfPreviewTest({
return `${code}${name ? ' · ' + name : ''}`;
}, [activePoint]);
// ---------- 通知上层 ----------
const effThumbMode: 'filtered' | 'all' =
thumbMode === 'filtered' && rulePages.length > 0 ? 'filtered' : 'all';
const thumbPages = useMemo<number[]>(() => {
if (!numPages) return [];
if (effThumbMode === 'filtered') return rulePages.filter(page => page >= 1 && page <= numPages);
return Array.from({ length: numPages }, (_, index) => index + 1);
}, [effThumbMode, rulePages, numPages]);
const rulePageSet = useMemo(() => new Set(rulePages), [rulePages]);
const totalThumbHeight = thumbPages.length * THUMB_ESTIMATED_HEIGHT;
const visibleThumbRange = useMemo(() => {
if (!thumbPages.length) {
return { start: 0, end: 0 };
}
const viewportHeight = thumbsViewportHeight || THUMB_ESTIMATED_HEIGHT * 3;
const start = Math.max(0, Math.floor(thumbsScrollTop / THUMB_ESTIMATED_HEIGHT) - THUMB_OVERSCAN);
const visibleCount = Math.ceil(viewportHeight / THUMB_ESTIMATED_HEIGHT) + THUMB_OVERSCAN * 2;
const end = Math.min(thumbPages.length, start + visibleCount);
return { start, end };
}, [thumbPages.length, thumbsScrollTop, thumbsViewportHeight]);
const visibleThumbItems = useMemo(
() => thumbPages.slice(visibleThumbRange.start, visibleThumbRange.end),
[thumbPages, visibleThumbRange.start, visibleThumbRange.end],
);
const mainPageDevicePixelRatio = useMemo(() => {
if (typeof window === 'undefined') return 1;
return Math.min(window.devicePixelRatio || 1, MAIN_PAGE_MAX_DEVICE_PIXEL_RATIO);
}, []);
useEffect(() => {
if (numPages && onNumPagesChange) onNumPagesChange(numPages);
}, [numPages, onNumPagesChange]);
@@ -188,7 +234,6 @@ export function PdfPreviewTest({
if (onZoomChange) onZoomChange(zoomLevel);
}, [zoomLevel, onZoomChange]);
// ---------- targetPage 跳转 ----------
useEffect(() => {
if (targetPage && numPages) {
const next = Math.max(1, Math.min(numPages, targetPage + pageOffset));
@@ -196,33 +241,77 @@ export function PdfPreviewTest({
}
}, [targetPage, numPages, pageOffset, activeReviewPointResultId]);
// ---------- 切换规则:重置略缩图模式为 filtered ----------
useEffect(() => {
if (activeReviewPointResultId) setThumbMode('filtered');
}, [activeReviewPointResultId]);
// ---------- 文件路径变化:重置坐标自动计算 ----------
useEffect(() => {
setIsScaleAutoCalculated(false);
}, [filePath]);
// ---------- 略缩图滚动到当前页 ----------
useEffect(() => {
const host = thumbsPanelRef.current;
if (!host) return;
const el = host.querySelector<HTMLElement>(`[data-thumb-page="${currentPage}"]`);
if (el) el.scrollIntoView({ block: 'nearest', behavior: 'smooth' });
}, [currentPage, thumbMode, showThumbs]);
// ---------- PDF 加载 ----------
const currentIndex = thumbPages.indexOf(currentPage);
if (currentIndex < 0) return;
const itemTop = currentIndex * THUMB_ESTIMATED_HEIGHT;
const itemBottom = itemTop + THUMB_ESTIMATED_HEIGHT;
const viewportTop = host.scrollTop;
const viewportBottom = viewportTop + host.clientHeight;
if (itemTop < viewportTop) {
host.scrollTo({ top: itemTop, behavior: 'smooth' });
return;
}
if (itemBottom > viewportBottom) {
host.scrollTo({
top: Math.max(0, itemBottom - host.clientHeight),
behavior: 'smooth',
});
}
}, [currentPage, thumbMode, showThumbs, thumbPages]);
useEffect(() => {
const host = thumbsPanelRef.current;
if (!host) return;
const updateViewport = () => {
setThumbsViewportHeight(host.clientHeight);
setThumbsScrollTop(host.scrollTop);
};
updateViewport();
const handleScroll = () => {
setThumbsScrollTop(host.scrollTop);
};
host.addEventListener('scroll', handleScroll, { passive: true });
let resizeObserver: ResizeObserver | null = null;
if (typeof ResizeObserver !== 'undefined') {
resizeObserver = new ResizeObserver(() => {
updateViewport();
});
resizeObserver.observe(host);
}
return () => {
host.removeEventListener('scroll', handleScroll);
resizeObserver?.disconnect();
};
}, [showThumbs, thumbMode, filePath]);
const onDocumentLoadSuccess = useCallback(
({ numPages: n }: { numPages: number }) => {
setNumPages(n);
// 初始页:优先 targetPage,否则第 1 页
({ numPages: loadedNumPages }: { numPages: number }) => {
setNumPages(loadedNumPages);
if (targetPage) {
setCurrentPage(Math.max(1, Math.min(n, targetPage + pageOffset)));
setCurrentPage(Math.max(1, Math.min(loadedNumPages, targetPage + pageOffset)));
} else {
setCurrentPage(p => Math.max(1, Math.min(n, p)));
setCurrentPage(page => Math.max(1, Math.min(loadedNumPages, page)));
}
},
[targetPage, pageOffset],
@@ -233,19 +322,18 @@ export function PdfPreviewTest({
setLoadError('PDF文档加载失败:' + (error.message || '未知错误'));
}, []);
// ---------- 主页面加载(自动校准坐标) ----------
const onMainPageLoadSuccess = useCallback(
(page: any) => {
setPageRenderTick(tick => tick + 1);
if (isScaleAutoCalculated) return;
setTimeout(() => {
const pdfOriginalWidthPt = page.view?.[2] || page.originalWidth || page.width;
const canvas = viewportRef.current?.querySelector(
'.pdf-main-canvas .react-pdf__Page__canvas',
) as HTMLCanvasElement | null;
const canvas = viewportRef.current?.querySelector('.pdf-main-canvas .react-pdf__Page__canvas') as HTMLCanvasElement | null;
if (canvas && pdfOriginalWidthPt) {
const canvasDisplayWidth = canvas.offsetWidth;
const currentScale = zoomLevel / 100;
const autoScale = (canvasDisplayWidth / currentScale) / pdfOriginalWidthPt;
const autoScale = canvasDisplayWidth / currentScale / pdfOriginalWidthPt;
setCoordinateScale(autoScale);
setIsScaleAutoCalculated(true);
}
@@ -254,71 +342,89 @@ export function PdfPreviewTest({
[isScaleAutoCalculated, zoomLevel],
);
// ---------- 翻页 / 缩放 ----------
const goPrev = () => setCurrentPage(p => Math.max(1, p - 1));
const goNext = () => setCurrentPage(p => Math.min(numPages || p, p + 1));
const zoomIn = () => setZoomLevel(z => Math.min(200, z + 10));
const zoomOut = () => setZoomLevel(z => Math.max(50, z - 10));
const goPrev = () => setCurrentPage(page => Math.max(1, page - 1));
const goNext = () => setCurrentPage(page => Math.min(numPages || page, page + 1));
const zoomIn = () => setZoomLevel(level => Math.min(200, level + 10));
const zoomOut = () => setZoomLevel(level => Math.max(50, level - 10));
const jumpToHighlight = () => {
if (!activePoint || rulePages.length === 0) {
toastService.info('当前规则无关联页');
return;
}
const first = rulePages[0];
if (numPages && first >= 1 && first <= numPages) setCurrentPage(first);
const firstPage = rulePages[0];
if (numPages && firstPage >= 1 && firstPage <= numPages) setCurrentPage(firstPage);
};
const handlePageInputChange = (e: React.ChangeEvent<HTMLInputElement>) => {
setPageInputValue(e.target.value.replace(/\D/g, ''));
};
const handlePageJump = () => {
if (!pageInputValue || !numPages) return;
const n = parseInt(pageInputValue, 10);
if (n > 0 && n <= numPages) {
setCurrentPage(n);
setPageInputValue('');
} else {
toastService.warning(`请输入有效页码 (1-${numPages})`);
const nextPage = parseInt(pageInputValue, 10);
if (nextPage > 0 && nextPage <= numPages) {
setCurrentPage(nextPage);
setPageInputValue('');
return;
}
toastService.warning(`请输入有效页码 (1-${numPages})`);
setPageInputValue('');
};
// ---------- 高亮矩形(对齐原 PdfPreview 的字符位置) ----------
const mainHighlight = useMemo(() => {
if (!charPositions?.length || !targetPage) {
console.log('[PdfPreviewTest] highlight skipped: no charPositions/targetPage', {
hasCharPositions: !!charPositions?.length,
targetPage,
});
return null;
}
if (currentPage !== targetPage + pageOffset) {
console.log('[PdfPreviewTest] highlight skipped: page mismatch', {
currentPage,
targetPage,
pageOffset,
expected: targetPage + pageOffset,
});
return null;
}
const bboxRectHighlight = useMemo(() => {
if (!bboxHighlight || !isValidQuad(bboxHighlight.bbox) || !isValidQuad(bboxHighlight.pageBox)) return null;
const expectedPage = bboxHighlight.page ?? (typeof bboxHighlight.pageNum === 'number' ? bboxHighlight.pageNum + 1 : targetPage);
if (!expectedPage || currentPage !== expectedPage + pageOffset) return null;
const canvas = viewportRef.current?.querySelector('.pdf-main-canvas .react-pdf__Page__canvas') as HTMLCanvasElement | null;
if (!canvas) return null;
const [pageX0, pageY0, pageX1, pageY1] = bboxHighlight.pageBox;
const [bboxLeft, bboxTop, bboxRight, bboxBottom] = bboxHighlight.bbox;
const pageWidth = pageX1 - pageX0;
const pageHeight = pageY1 - pageY0;
if (pageWidth <= 0 || pageHeight <= 0) return null;
const left = clamp(Math.min(bboxLeft, bboxRight), pageX0, pageX1);
const right = clamp(Math.max(bboxLeft, bboxRight), pageX0, pageX1);
const top = clamp(Math.min(bboxTop, bboxBottom), pageY0, pageY1);
const bottom = clamp(Math.max(bboxTop, bboxBottom), pageY0, pageY1);
if (right <= left || bottom <= top) return null;
return {
x: ((left - pageX0) / pageWidth) * canvas.offsetWidth,
y: ((top - pageY0) / pageHeight) * canvas.offsetHeight,
width: ((right - left) / pageWidth) * canvas.offsetWidth,
height: ((bottom - top) / pageHeight) * canvas.offsetHeight,
text: bboxHighlight.fieldKey,
};
}, [bboxHighlight, targetPage, currentPage, pageOffset, zoomLevel, pageRenderTick]);
const charRectHighlight = useMemo(() => {
if (bboxRectHighlight || !charPositions?.length || !targetPage) return null;
if (currentPage !== targetPage + pageOffset) return null;
const scale = zoomLevel / 100;
let minX = Infinity,
minY = Infinity,
maxX = -Infinity,
maxY = -Infinity;
let minX = Infinity;
let minY = Infinity;
let maxX = -Infinity;
let maxY = -Infinity;
const chars: string[] = [];
charPositions.forEach(cp => {
chars.push(cp.char);
cp.box.forEach(pt => {
const [x, y] = pt;
charPositions.forEach(position => {
chars.push(position.char);
position.box.forEach(([x, y]) => {
if (x < minX) minX = x;
if (x > maxX) maxX = x;
if (y < minY) minY = y;
if (y > maxY) maxY = y;
});
});
return {
x: minX * coordinateScale * scale,
y: minY * coordinateScale * scale,
@@ -326,21 +432,10 @@ export function PdfPreviewTest({
height: (maxY - minY) * coordinateScale * scale,
text: chars.join(''),
};
}, [charPositions, targetPage, currentPage, pageOffset, zoomLevel, coordinateScale]);
}, [bboxRectHighlight, charPositions, targetPage, currentPage, pageOffset, zoomLevel, coordinateScale, pageRenderTick]);
// ---------- 略缩图可见页列表 ----------
const effThumbMode: 'filtered' | 'all' =
thumbMode === 'filtered' && rulePages.length > 0 ? 'filtered' : 'all';
const mainHighlight = bboxRectHighlight || charRectHighlight;
const thumbPages = useMemo<number[]>(() => {
if (!numPages) return [];
if (effThumbMode === 'filtered') return rulePages.filter(p => p >= 1 && p <= numPages);
return Array.from({ length: numPages }, (_, i) => i + 1);
}, [effThumbMode, rulePages, numPages]);
// ============================================================
// 渲染
// ============================================================
if (loadError) {
return (
<div className="w-full h-full grid place-items-center text-red-500 p-4">{loadError}</div>
@@ -391,7 +486,7 @@ export function PdfPreviewTest({
onKeyDown={e => {
if (e.key === 'Enter') handlePageJump();
}}
className="w-5 h-6 text-center bg-slate-50 border border-slate-200 rounded text-[12px] font-mono outline-none focus:border-primary"
className="w-6 h-6 text-center bg-slate-50 border border-slate-200 rounded text-[12px] font-mono outline-none focus:border-primary"
/>
<span className="text-slate-400"> / {numPages ?? '-'}</span>
</span>
@@ -481,7 +576,7 @@ export function PdfPreviewTest({
{/* 略缩图列表 */}
<div
ref={thumbsPanelRef}
className="flex-1 overflow-y-auto py-2 px-2 space-y-2"
className="flex-1 overflow-y-auto py-2 px-2"
>
{numPages === null ? (
<div className="text-center text-[11px] text-slate-400 py-4"></div>
@@ -491,76 +586,81 @@ export function PdfPreviewTest({
<div className="mt-1"></div>
</div>
) : (
thumbPages.map(p => {
const info = pageStatusMap.get(p);
const isCur = p === currentPage;
const isRulePage = rulePages.includes(p);
<div className="relative w-full" style={{ height: totalThumbHeight }}>
{visibleThumbItems.map((p, visibleIndex) => {
const itemIndex = visibleThumbRange.start + visibleIndex;
const info = pageStatusMap.get(p);
const isCur = p === currentPage;
const isRulePage = rulePageSet.has(p);
let badge: React.ReactNode = null;
if (info) {
const b = STATUS_BADGE[info.worst];
const num = info.issues > 0 ? info.issues : info.worst === 'pass' ? '' : info.count;
badge = (
<span
className={`absolute top-1 right-1 inline-flex items-center justify-center min-w-[14px] h-[14px] px-0.5 rounded-full text-[9px] font-semibold text-white ${b.cls} shadow ring-1 ring-white`}
>
{num ? num : <i className={`${b.ic} text-[9px]`}></i>}
</span>
);
}
let badge: React.ReactNode = null;
if (info) {
const b = STATUS_BADGE[info.worst];
const num = info.issues > 0 ? info.issues : info.worst === 'pass' ? '' : info.count;
badge = (
<span
className={`absolute top-1 right-1 inline-flex items-center justify-center min-w-[14px] h-[14px] px-0.5 rounded-full text-[9px] font-semibold text-white ${b.cls} shadow ring-1 ring-white`}
>
{num ? num : <i className={`${b.ic} text-[9px]`}></i>}
</span>
);
}
const frameCls = isCur
? 'ring-2 ring-[#00684a] shadow-md'
: effThumbMode === 'all' && isRulePage
? 'ring-1 ring-[#00684a]/40 shadow-sm'
: 'border border-slate-200 group-hover:border-slate-400 shadow-sm';
const frameCls = isCur
? 'ring-2 ring-[#00684a] shadow-md'
: effThumbMode === 'all' && isRulePage
? 'ring-1 ring-[#00684a]/40 shadow-sm'
: 'border border-slate-200 group-hover:border-slate-400 shadow-sm';
let fieldsLabel: React.ReactNode = null;
if (effThumbMode === 'filtered' && activePoint) {
const fs = getFieldsOnPage(activePoint, p - pageOffset);
const txt = fs.length ? fs.join(' · ') : '规则锚定页';
fieldsLabel = (
<div
className="text-[10px] leading-tight text-slate-500 text-center mt-0.5 line-clamp-2"
title={txt}
>
{txt}
</div>
);
}
return (
<button
key={p}
data-thumb-page={p}
onClick={() => setCurrentPage(p)}
className="block w-full group"
title={`${p}`}
>
<div className={`relative rounded overflow-hidden bg-white transition ${frameCls}`}>
<div className="w-full bg-gradient-to-b from-white to-slate-50 overflow-hidden">
<Page
pageNumber={p}
width={112}
renderTextLayer={false}
renderAnnotationLayer={false}
loading={<div className="w-full h-[150px] bg-slate-50" />}
error={<div className="w-full h-[150px] bg-slate-50" />}
/>
let fieldsLabel: React.ReactNode = null;
if (effThumbMode === 'filtered' && activePoint) {
const fs = getFieldsOnPage(activePoint, p - pageOffset);
const txt = fs.length ? fs.join(' · ') : '规则锚定页';
fieldsLabel = (
<div
className="text-[10px] leading-tight text-slate-500 text-center mt-0.5 line-clamp-2"
title={txt}
>
{txt}
</div>
{badge}
</div>
<div
className={`text-center text-[10.5px] mt-1 ${
isCur ? 'text-[#00684a] font-semibold' : 'text-slate-500 group-hover:text-slate-700'
}`}
);
}
return (
<button
key={p}
data-thumb-page={p}
onClick={() => setCurrentPage(p)}
className="absolute left-0 block w-full group"
style={{ top: itemIndex * THUMB_ESTIMATED_HEIGHT }}
title={`${p}`}
>
{p}
</div>
{fieldsLabel}
</button>
);
})
<div className={`relative rounded overflow-hidden bg-white transition ${frameCls}`}>
<div className="w-full bg-gradient-to-b from-white to-slate-50 overflow-hidden">
<Page
pageNumber={p}
width={THUMB_WIDTH}
devicePixelRatio={1}
renderTextLayer={false}
renderAnnotationLayer={false}
loading={<div className="w-full h-[150px] bg-slate-50" />}
error={<div className="w-full h-[150px] bg-slate-50" />}
/>
</div>
{badge}
</div>
<div
className={`text-center text-[10.5px] mt-1 ${
isCur ? 'text-[#00684a] font-semibold' : 'text-slate-500 group-hover:text-slate-700'
}`}
>
{p}
</div>
{fieldsLabel}
</button>
);
})}
</div>
)}
</div>
</div>
@@ -586,7 +686,7 @@ export function PdfPreviewTest({
<Page
pageNumber={Math.min(currentPage, numPages)}
scale={zoomLevel / 100}
devicePixelRatio={typeof window !== 'undefined' ? window.devicePixelRatio || 1 : 1}
devicePixelRatio={mainPageDevicePixelRatio}
renderTextLayer={false}
renderAnnotationLayer={false}
onLoadSuccess={onMainPageLoadSuccess}
@@ -610,7 +710,8 @@ export function PdfPreviewTest({
y={mainHighlight.y}
width={mainHighlight.width}
height={mainHighlight.height}
fill="#00AA00"
// fill="#00AA00"
fill="#abf694"
fillOpacity="0.1"
stroke="#00684a"
strokeWidth="0.5"
@@ -1,8 +1,8 @@
/**
* 右栏 · 详情面板
* 包含 3 个选项卡(评查结果、抽取字段、文件信息)+ 底部操作栏
* 包含 3 个选项卡(评查结果、抽取字段、文件信息)底部操作栏
*/
import type { ReviewPoint, CharPosition } from '../ReviewPointsList';
import type { ReviewPoint, CharPosition, PdfBboxHighlight } from '../ReviewPointsList';
import { ReviewPointDetailCard } from './ReviewPointDetailCard';
import { FileInfoPanel } from './FileInfoPanel';
@@ -32,10 +32,11 @@ interface DetailPanelProps {
onTabChange: (tab: TabKey) => void;
activeReviewPoint: ReviewPoint | null;
reviewPoints: ReviewPoint[];
detailMode?: 'legacy' | 'leaudit';
fileInfo: FileInfoData;
reviewInfo: ReviewInfoData;
onReviewPointSelect: (id: string | number, page?: number, charPositions?: CharPosition[], value?: string) => void;
onStatusChange: (id: string | number, editAuditStatusId: string | number, status: string, message: string) => void;
onReviewPointSelect: (id: string | number, page?: number, charPositions?: CharPosition[], value?: string, bboxHighlight?: PdfBboxHighlight) => void;
onStatusChange: (id: string, editAuditStatusId: string | number, status: string, message: string) => void;
onConfirmResults: () => void;
onDownload: () => void;
auditStatus?: number;
@@ -45,63 +46,191 @@ interface DetailPanelProps {
showComparisonButton?: boolean;
}
type ExtractedFieldValue = {
value?: unknown;
page?: number | string;
};
function isValidQuad(value: unknown): value is [number, number, number, number] {
return Array.isArray(value) && value.length === 4 && value.every(item => typeof item === 'number' && Number.isFinite(item));
}
function hasNonZeroQuad(value: [number, number, number, number]): boolean {
return value.some(item => item !== 0);
}
function getFieldRawValue(value: ReviewPoint['content'][string]): unknown {
if (value == null) return null;
if (typeof value === 'object' && 'value' in value) {
return (value as { value?: unknown }).value ?? null;
}
return value;
}
function getFieldDisplayText(rawValue: unknown): string {
if (rawValue == null) return '缺失';
if (typeof rawValue === 'string' || typeof rawValue === 'number' || typeof rawValue === 'boolean') return String(rawValue);
try {
return JSON.stringify(rawValue);
} catch {
return String(rawValue);
}
}
function getFieldHighlightText(rawValue: unknown): string | undefined {
if (rawValue == null) return undefined;
if (typeof rawValue !== 'string' && typeof rawValue !== 'number' && typeof rawValue !== 'boolean') return undefined;
const text = String(rawValue).trim();
return text ? String(rawValue) : undefined;
}
function getFieldPage(point: ReviewPoint, key: string, value: ReviewPoint['content'][string]): number | undefined {
const contentPage = point.contentPage?.[key];
const parsedContentPage = Number(contentPage);
if (Number.isFinite(parsedContentPage) && parsedContentPage > 0) return parsedContentPage;
const inlinePage = typeof value === 'object' && value && 'page' in value
? Number((value as { page?: unknown }).page)
: NaN;
if (Number.isFinite(inlinePage) && inlinePage > 0) return inlinePage;
const pageNum = point.fieldPositions?.[key]?.page_num;
if (typeof pageNum === 'number' && Number.isFinite(pageNum) && pageNum >= 0) return pageNum + 1;
return undefined;
}
function getFieldConfidence(point: ReviewPoint, key: string): number | undefined {
const confidence = point.fieldPositions?.[key]?.confidence;
if (typeof confidence !== 'number' || !Number.isFinite(confidence)) return undefined;
return confidence;
}
function getFieldBboxHighlight(point: ReviewPoint, key: string, page?: number): PdfBboxHighlight | undefined {
const fieldPosition = point.fieldPositions?.[key];
if (!fieldPosition || !isValidQuad(fieldPosition.bbox) || !isValidQuad(fieldPosition.page_box)) return undefined;
if (!hasNonZeroQuad(fieldPosition.bbox) || !hasNonZeroQuad(fieldPosition.page_box)) return undefined;
return {
fieldKey: key,
bbox: [...fieldPosition.bbox],
pageBox: [...fieldPosition.page_box],
pageNum: fieldPosition.page_num,
page,
confidence: fieldPosition.confidence,
matchMethod: fieldPosition.match_method,
};
}
function ExtractedFieldsPanel({
reviewPoints,
onFieldClick,
}: {
reviewPoints: ReviewPoint[];
onFieldClick: (pointId: string | number, page: number) => void;
onFieldClick: (pointId: string | number, page?: number, value?: string, bboxHighlight?: PdfBboxHighlight) => void;
}) {
const fields: Array<{ key: string; value: string; page?: number; pointName: string; pointId: string | number }> = [];
const handleFieldNavigate = (pointId: string, page?: number, value?: string, bboxHighlight?: PdfBboxHighlight) => {
if (!page) return;
const selectedText = typeof window !== 'undefined' ? window.getSelection?.()?.toString().trim() : '';
if (selectedText) return;
onFieldClick(pointId, page, value, bboxHighlight);
};
const fields: Array<{
key: string;
displayValue: string;
highlightValue?: string;
isMissing: boolean;
confidence?: number;
page?: number;
pointId: string | number;
bboxHighlight?: PdfBboxHighlight;
}> = [];
reviewPoints.forEach((p) => {
if (p.content) {
Object.entries(p.content).forEach(([key, data]) => {
const fieldData = (data && typeof data === 'object' ? data : {}) as ExtractedFieldValue & { text?: string };
const val = fieldData.value;
const page = fieldData.page;
const text = typeof val === 'object' && val !== null
? ('text' in (val as Record<string, unknown>) ? String((val as Record<string, unknown>).text || '') : JSON.stringify(val))
: String(val || '');
Object.entries(p.content).forEach(([key, rawValue]) => {
const fieldRawValue = getFieldRawValue(rawValue);
const displayValue = getFieldDisplayText(fieldRawValue);
const highlightValue = getFieldHighlightText(fieldRawValue);
const page = getFieldPage(p, key, rawValue);
const confidence = getFieldConfidence(p, key);
const bboxHighlight = getFieldBboxHighlight(p, key, page);
fields.push({
key,
value: text,
page: page ? Number(page) : undefined,
pointName: p.pointName,
displayValue,
highlightValue,
isMissing: fieldRawValue == null,
confidence,
page,
pointId: p.id,
bboxHighlight,
});
});
}
});
return (
<div className="p-3">
<div className="text-[11px] font-medium text-slate-400 uppercase tracking-wider mb-2">
<span className="font-mono normal-case text-[10.5px]">{fields.length}</span>
<div className="h-full flex flex-col min-h-0">
<div className="shrink-0 px-4 py-3 border-b border-slate-100">
<div className="flex items-baseline justify-between gap-3">
<h3 className="text-[14px] font-semibold text-slate-900">
<span className="font-mono text-[11px] text-slate-400">{fields.length}</span>
</h3>
<div className="text-[11px] text-slate-400"> · </div>
</div>
</div>
{fields.length === 0 ? (
<div className="text-center py-6 text-[12px] text-slate-400"></div>
<div className="text-center py-10 text-[12px] text-slate-400"></div>
) : (
<div className="space-y-2">
<div className="flex-1 min-h-0 overflow-y-auto scroll-slim">
{fields.map((f, i) => (
<button
<div
key={`${f.key}-${i}`}
type="button"
className={`w-full border border-slate-200 rounded-md text-left hover:bg-slate-50 transition p-2.5 ${f.page ? 'cursor-pointer' : 'cursor-default opacity-70'}`}
onClick={() => f.page && onFieldClick(f.pointId, f.page)}
role={f.page ? 'button' : undefined}
tabIndex={f.page ? 0 : undefined}
className={`w-full flex items-start gap-2 px-3 py-2 border-b border-slate-100 text-left transition ${f.page ? 'cursor-pointer hover:bg-slate-50' : 'cursor-default opacity-80'}`}
onClick={() => handleFieldNavigate(f.pointId, f.page, f.highlightValue, f.bboxHighlight)}
onKeyDown={(event) => {
if (!f.page) return;
if (event.key === 'Enter' || event.key === ' ') {
event.preventDefault();
handleFieldNavigate(f.pointId, f.page, f.highlightValue, f.bboxHighlight);
}
}}
>
<div className="flex items-center justify-between gap-2">
<span className="text-[11px] text-slate-500 truncate font-medium">{f.key}</span>
{f.page && <span className="text-[10.5px] text-slate-400 shrink-0">P{f.page}</span>}
<div className="flex-1 min-w-0">
<div className="text-[12px] font-medium text-slate-800 leading-5 break-words">{f.key}</div>
<div className="mt-0.5 select-text cursor-text">
{f.isMissing ? (
<span className="text-[11px] text-red-500"></span>
) : (
<span className="text-[11px] text-slate-500 leading-5 whitespace-pre-wrap break-words">{f.displayValue}</span>
)}
</div>
</div>
{f.value && <div className="text-[12px] text-slate-700 mt-1 leading-relaxed line-clamp-2">{f.value}</div>}
<div className="text-[10px] text-slate-400 mt-0.5">{f.pointName}</div>
</button>
<div className="shrink-0 text-right min-w-[56px] pt-0.5">
<div className={`font-mono text-[10.5px] ${f.confidence == null ? 'text-slate-400' : f.confidence < 0.8 ? 'text-orange-600' : 'text-slate-500'}`}>
{f.confidence == null ? '-' : `${Math.round(f.confidence * 100)}%`}
</div>
{f.page ? (
<button
type="button"
className="mt-0.5 text-[10px] text-[#00684a] hover:underline"
onClick={(event) => {
event.stopPropagation();
handleFieldNavigate(f.pointId, f.page, f.highlightValue, f.bboxHighlight);
}}
>
p.{f.page}
</button>
) : (
<div className="mt-0.5 text-[10px] text-slate-300">-</div>
)}
</div>
</div>
))}
</div>
)}
@@ -120,6 +249,7 @@ export function DetailPanel({
onTabChange,
activeReviewPoint,
reviewPoints,
detailMode = 'legacy',
fileInfo,
reviewInfo,
onReviewPointSelect,
@@ -186,6 +316,7 @@ export function DetailPanel({
onReviewPointSelect={onReviewPointSelect}
onStatusChange={onStatusChange}
fileFormat={fileFormat}
detailMode={detailMode}
/>
</div>
) : (
@@ -199,8 +330,8 @@ export function DetailPanel({
{activeTab === 'fields' && (
<ExtractedFieldsPanel
reviewPoints={reviewPoints}
onFieldClick={(pointId, page) => {
onReviewPointSelect(pointId, page);
onFieldClick={(pointId, page, value, bboxHighlight) => {
onReviewPointSelect(pointId, page, undefined, value, bboxHighlight);
}}
/>
)}
@@ -5,13 +5,17 @@
import { useState, useEffect, useRef, useMemo } from 'react';
import { createPortal } from 'react-dom';
import { toastService } from '~/components/ui/Toast';
import type { ReviewPoint, CharPosition } from '../ReviewPointsList';
import type { ReviewPoint, CharPosition, PdfBboxHighlight } from '../ReviewPointsList';
import { CorporateInfoModal } from '../../corporate-information';
import type { BusinessInfoResult, DishonestyResult } from '../../corporate-information';
import { queryCompanyInfo } from '~/api/corporate-information/qichacha';
interface ReviewPointDetailCardProps {
reviewPoint: ReviewPoint;
onReviewPointSelect: (id: string | number, page?: number, charPositions?: CharPosition[], value?: string) => void;
onStatusChange: (id: string | number, editAuditStatusId: string | number, status: string, message: string) => void;
onReviewPointSelect: (id: string | number, page?: number, charPositions?: CharPosition[], value?: string, bboxHighlight?: PdfBboxHighlight) => void;
onStatusChange: (id: string, editAuditStatusId: string | number, status: string, message: string) => void;
fileFormat?: string;
detailMode?: 'legacy' | 'leaudit';
}
// ── 比较方法映射 ──
@@ -33,6 +37,58 @@ const getRuleTypeText = (type?: string): string => {
return ruleTypeMap[type] || type;
};
function normalizeActionContent(actionContent?: string | string[]): string {
if (typeof actionContent === 'string') return actionContent;
if (Array.isArray(actionContent)) {
return actionContent
.map(item => typeof item === 'string' ? item : JSON.stringify(item))
.filter(Boolean)
.join('\n');
}
return '';
}
function getLeauditNote(reviewPoint: ReviewPoint): string {
return reviewPoint.editAuditStatusMessage || normalizeActionContent(reviewPoint.actionContent) || reviewPoint.suggestion || '';
}
function getLeauditRawFieldValue(value: ReviewPoint['content'][string]): unknown {
if (value == null) return undefined;
if (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') return value;
if (typeof value === 'object' && 'value' in value) return value.value;
return value;
}
function isValidQuad(value: unknown): value is [number, number, number, number] {
return Array.isArray(value) && value.length === 4 && value.every(item => typeof item === 'number' && Number.isFinite(item));
}
function getLeauditTargetPage(reviewPoint: ReviewPoint, fieldKey: string): number | undefined {
const contentPage = reviewPoint.contentPage?.[fieldKey];
const parsedContentPage = Number(contentPage);
if (Number.isFinite(parsedContentPage) && parsedContentPage > 0) return parsedContentPage;
const pageNum = reviewPoint.fieldPositions?.[fieldKey]?.page_num;
if (typeof pageNum === 'number' && Number.isFinite(pageNum) && pageNum >= 0) return pageNum + 1;
return undefined;
}
function getLeauditBboxHighlight(reviewPoint: ReviewPoint, fieldKey: string, page?: number): PdfBboxHighlight | undefined {
const fieldPosition = reviewPoint.fieldPositions?.[fieldKey];
if (!fieldPosition || !isValidQuad(fieldPosition.bbox) || !isValidQuad(fieldPosition.page_box)) return undefined;
return {
fieldKey,
bbox: [...fieldPosition.bbox] as [number, number, number, number],
pageBox: [...fieldPosition.page_box] as [number, number, number, number],
pageNum: fieldPosition.page_num,
page,
confidence: fieldPosition.confidence,
matchMethod: fieldPosition.match_method,
};
}
// ── Tooltip 系统 ──
let activeTooltip = { show: false, content: null as React.ReactNode, position: { top: 0, left: 0 }, ready: false };
function TooltipPortal() {
@@ -223,7 +279,7 @@ function filterOtherRule(reviewPoint: ReviewPoint): MergedRule[] {
}
// ── renderOtherRule ──
function RenderOtherRule({ rule, reviewPoint, onReviewPointSelect }: { rule: MergedRule; reviewPoint: ReviewPoint; onReviewPointSelect: (id: string | number, page?: number, charPos?: CharPosition[], value?: string) => void }) {
function RenderOtherRule({ rule, reviewPoint, onReviewPointSelect }: { rule: MergedRule; reviewPoint: ReviewPoint; onReviewPointSelect: (id: string | number, page?: number, charPos?: CharPosition[], value?: string, bboxHighlight?: PdfBboxHighlight) => void }) {
const fieldKey = rule.fieldKey;
const fieldValue = rule.fieldValue;
const hasFailure = Object.values(fieldValue.type || {}).some(item => item.res === false);
@@ -273,7 +329,7 @@ function RenderOtherRule({ rule, reviewPoint, onReviewPointSelect }: { rule: Mer
// ── renderConsistencyRule ──
type ChainItem = { field: string; data: { key: string; page: number; value: string; char_positions?: CharPosition[] }; res: boolean; compareMethod?: string };
function RenderConsistencyRule({ rule, reviewPoint, onReviewPointSelect }: { rule: Record<string, unknown>; reviewPoint: ReviewPoint; onReviewPointSelect: (id: string | number, page?: number, charPos?: CharPosition[], value?: string) => void }) {
function RenderConsistencyRule({ rule, reviewPoint, onReviewPointSelect }: { rule: Record<string, unknown>; reviewPoint: ReviewPoint; onReviewPointSelect: (id: string | number, page?: number, charPos?: CharPosition[], value?: string, bboxHighlight?: PdfBboxHighlight) => void }) {
if (reviewPoint.result !== (rule.res as boolean)) return null;
const config = rule.config as { logic?: string; pairs?: Array<{ sourceField: Record<string, { page: number; value: string; char_positions?: CharPosition[] }>; targetField: Record<string, { page: number; value: string; char_positions?: CharPosition[] }>; res: boolean; compareMethod?: string }>; selectedFields?: string[] } | undefined;
if (!config || !config.pairs || !Array.isArray(config.pairs) || config.pairs.length === 0) return null;
@@ -389,7 +445,7 @@ function RenderConsistencyRule({ rule, reviewPoint, onReviewPointSelect }: { rul
}
// ── renderModelRule ──
function RenderModelRule({ rule, reviewPoint, onReviewPointSelect, fileFormat }: { rule: Record<string, unknown>; reviewPoint: ReviewPoint; onReviewPointSelect: (id: string | number, page?: number, charPos?: CharPosition[], value?: string) => void; fileFormat?: string }) {
function RenderModelRule({ rule, reviewPoint, onReviewPointSelect, fileFormat }: { rule: Record<string, unknown>; reviewPoint: ReviewPoint; onReviewPointSelect: (id: string | number, page?: number, charPos?: CharPosition[], value?: string, bboxHighlight?: PdfBboxHighlight) => void; fileFormat?: string }) {
const config = rule.config as { model?: string; fields?: Record<string, { page: number | string; value: string; char_positions?: CharPosition[]; res?: boolean }>; ai_suggestion?: { summary?: string; analysis?: { failure_reason?: string; solution_approach?: string; rule_understanding?: string }; suggestions?: Record<string, { reason: string; source: { page: number | null; type: string; field: string | null }; priority: string; confidence: number; suggested_value: string | null }>; generated_at?: string }; message?: string; res?: boolean } | undefined;
if (config?.res !== reviewPoint.result) return null;
@@ -434,167 +490,685 @@ function RenderModelRule({ rule, reviewPoint, onReviewPointSelect, fileFormat }:
return <>{fieldElements}</>;
}
function RenderGenericRule({
rule,
reviewPoint,
onReviewPointSelect,
}: {
rule: Record<string, unknown>;
// ── Main Component ──
function stringifyUnknown(value: unknown): string {
if (typeof value === 'string') return value;
if (typeof value === 'number' || typeof value === 'boolean') return String(value);
if (value == null) return '';
try {
return JSON.stringify(value, null, 2);
} catch {
return String(value);
}
}
function getLeauditFieldText(value: ReviewPoint['content'][string]): string {
if (value == null) return '未填写';
if (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') {
return String(value);
}
if (typeof value === 'object' && value && 'value' in value) {
return stringifyUnknown(value.value);
}
return stringifyUnknown(value);
}
function parseMissingArrayString(rawText: string): string[] {
const match = rawText.match(/missing[\w-]*\s*:\s*\[([\s\S]*?)\]/i) || rawText.match(/\[([\s\S]*?)\]/);
if (!match) return [];
const innerText = match[1].trim();
if (!innerText) return [];
const quotedItems = Array.from(innerText.matchAll(/['"]([^'"]+)['"]/g))
.map(item => item[1].trim())
.filter(Boolean);
if (quotedItems.length > 0) {
return quotedItems;
}
return innerText
.split(',')
.map(item => item.trim().replace(/^['"]|['"]$/g, ''))
.filter(Boolean);
}
function getLeauditMissingItems(reviewPoint: ReviewPoint): string[] {
const textCandidates = [
reviewPoint.skipReason,
typeof reviewPoint.evaluatedPointResultsLog?.skip_reason === 'string' ? reviewPoint.evaluatedPointResultsLog.skip_reason : '',
reviewPoint.suggestion,
].filter(Boolean) as string[];
for (const text of textCandidates) {
if (!/missing[\w-]*\s*:/i.test(text)) continue;
const items = parseMissingArrayString(text);
if (items.length > 0) return items;
}
return [];
}
function normalizeAiResponseItems(value: unknown, options?: { hideNone?: boolean }): string[] {
const hideNone = options?.hideNone === true;
if (Array.isArray(value)) {
return value
.map(item => String(item).trim())
.filter(item => item && (!hideNone || item !== '无'));
}
if (typeof value === 'string') {
const trimmed = value.trim();
if (!trimmed) return [];
if (hideNone && trimmed === '无') return [];
return [trimmed];
}
return [];
}
function LeauditReviewPointDetailCard({ reviewPoint, onReviewPointSelect, onStatusChange }: {
reviewPoint: ReviewPoint;
onReviewPointSelect: (id: string | number, page?: number, charPos?: CharPosition[], value?: string) => void;
onReviewPointSelect: (id: string | number, page?: number, charPos?: CharPosition[], value?: string, bboxHighlight?: PdfBboxHighlight) => void;
onStatusChange: (id: string, editAuditStatusId: string | number, status: string, message: string) => void;
}) {
const config = (rule.config && typeof rule.config === 'object' ? rule.config : {}) as Record<string, unknown>;
const detail = (config.detail && typeof config.detail === 'object' ? config.detail : {}) as Record<string, unknown>;
const fieldNames = Array.isArray(detail.fields)
? detail.fields.map((field) => String(field))
: Array.isArray((config as any).fields)
? (config as any).fields.map((field: unknown) => String(field))
: [];
const passed = typeof rule.res === 'boolean' ? rule.res : reviewPoint.result === true;
const reasonCandidates = passed
? [config.reason, detail.reason, reviewPoint.passMessage]
: [config.reason, detail.reason, reviewPoint.failMessage, reviewPoint.suggestion];
const reason = reasonCandidates.find((item) => typeof item === 'string' && item.trim()) as string | undefined;
const checkType = typeof config.check_type === 'string' ? config.check_type : '';
const primitiveType = typeof config.primitive_type === 'string' ? config.primitive_type : '';
const badgeText = checkType || primitiveType || '规则检查';
const [manualNote, setManualNote] = useState(() => getLeauditNote(reviewPoint));
const [corporateModalVisible, setCorporateModalVisible] = useState(false);
const [corporateCompanyName, setCorporateCompanyName] = useState('');
const [corporateLoading, setCorporateLoading] = useState(false);
const [corporateBusinessInfo, setCorporateBusinessInfo] = useState<BusinessInfoResult | null>(null);
const [corporateDishonestyInfo, setCorporateDishonestyInfo] = useState<DishonestyResult | null>(null);
const [corporateError, setCorporateError] = useState<string | null>(null);
const [corporateUpdatedAt, setCorporateUpdatedAt] = useState<string | null>(null);
const getFieldLocatorState = (fieldName: string) => {
const fieldData = reviewPoint.content?.[fieldName];
const page = fieldData?.page || reviewPoint.contentPage?.[fieldName];
const normalizedPage = page ? Number(page) : undefined;
const hasPage = !!(normalizedPage && Number.isFinite(normalizedPage));
const rawValue = fieldData?.value;
const normalizedValue =
typeof rawValue === 'string'
? rawValue.trim()
: rawValue == null
? ''
: String(rawValue);
useEffect(() => {
setManualNote(getLeauditNote(reviewPoint));
}, [reviewPoint.id, reviewPoint.editAuditStatusMessage, reviewPoint.actionContent, reviewPoint.suggestion]);
return {
fieldData,
normalizedPage: hasPage ? normalizedPage : undefined,
normalizedValue,
canLocate: hasPage || normalizedValue.length > 0,
};
const stages = Array.isArray(reviewPoint.evaluatedPointResultsLog?.stages)
? (reviewPoint.evaluatedPointResultsLog.stages as Array<Record<string, unknown>>)
: [];
const missingItems = getLeauditMissingItems(reviewPoint);
const legalBasisList = Array.isArray(reviewPoint.legalBasis)
? reviewPoint.legalBasis
: reviewPoint.legalBasis?.articles?.map(item => typeof item === 'string' ? item : (item.name || item.content || stringifyUnknown(item))) || [];
const riskLabelMap: Record<string, { cls: string; label: string }> = {
high: { cls: 'bg-red-50 text-red-700 border-red-200', label: '高风险' },
medium: { cls: 'bg-amber-50 text-amber-700 border-amber-200', label: '中风险' },
low: { cls: 'bg-emerald-50 text-emerald-700 border-emerald-200', label: '低风险' },
};
const riskMeta = riskLabelMap[reviewPoint.riskLevel || ''] || {
cls: 'bg-slate-100 text-slate-600 border-slate-200',
label: '未知风险',
};
const configConfidence = reviewPoint.evaluationConfig && typeof reviewPoint.evaluationConfig === 'object'
? reviewPoint.evaluationConfig.confidence
: undefined;
const confidencePct = typeof reviewPoint.confidence === 'number'
? `${Math.round(reviewPoint.confidence * 100)}%`
: typeof configConfidence === 'number'
? `${Math.round(configConfidence * 100)}%`
: null;
const isPass = reviewPoint.status === 'success' && reviewPoint.result === true;
const isWarning = reviewPoint.status === 'warning' || (reviewPoint.ruleStatus || '').startsWith('skipped_');
const statusChip = isPass
? { cls: 'bg-emerald-50 text-emerald-700 border-emerald-200', icon: 'ri-checkbox-circle-fill', label: '通过' }
: isWarning
? { cls: 'bg-amber-50 text-amber-700 border-amber-200', icon: 'ri-error-warning-fill', label: '提醒' }
: { cls: 'bg-red-50 text-red-700 border-red-200', icon: 'ri-close-circle-fill', label: '不通过' };
const summaryText = isPass
? (reviewPoint.passMessage || reviewPoint.suggestion || '校验通过')
: isWarning
? (reviewPoint.skipReason || reviewPoint.suggestion || '当前规则未执行或需人工关注')
: (reviewPoint.failMessage || reviewPoint.suggestion || '发现问题,请处理');
const partyANameRaw = getLeauditRawFieldValue(reviewPoint.content?.['甲方名称']);
const partyBNameRaw = getLeauditRawFieldValue(reviewPoint.content?.['乙方名称']);
const partyAName = typeof partyANameRaw === 'string' ? partyANameRaw.trim() : String(partyANameRaw || '').trim();
const partyBName = typeof partyBNameRaw === 'string' ? partyBNameRaw.trim() : String(partyBNameRaw || '').trim();
const shouldShowEnterpriseButtons = reviewPoint.groupName?.trim() === '合同主体';
const handleCorporateInfoClick = async (companyName: string, forceRefresh = false) => {
if (!companyName) {
toastService.warning('企业名称为空,无法查询');
return;
}
setCorporateModalVisible(true);
setCorporateCompanyName(companyName);
setCorporateLoading(true);
setCorporateError(null);
setCorporateBusinessInfo(null);
setCorporateDishonestyInfo(null);
setCorporateUpdatedAt(null);
try {
const response = await queryCompanyInfo({ keyword: companyName, force_refresh: forceRefresh });
if (response.success && response.data) {
setCorporateBusinessInfo(response.data.enterprise);
setCorporateUpdatedAt(response.data.updated_at);
if (response.data.dishonesty) {
setCorporateDishonestyInfo({
VerifyResult: response.data.dishonesty.VerifyResult,
Data: response.data.dishonesty.Data || [],
});
}
} else {
setCorporateError(response.message || '查询失败');
}
} catch (error) {
console.error('查询企业信息失败:', error);
setCorporateError(error instanceof Error ? error.message : '查询失败');
} finally {
setCorporateLoading(false);
}
};
const jumpToField = (fieldName: string) => {
const { fieldData, normalizedPage, normalizedValue } = getFieldLocatorState(fieldName);
if (normalizedPage) {
onReviewPointSelect(
reviewPoint.id,
normalizedPage,
fieldData?.char_positions,
normalizedValue || undefined,
const handleCorporateForceRefresh = async () => {
if (corporateCompanyName) {
await handleCorporateInfoClick(corporateCompanyName, true);
}
};
const handleCloseCorporateModal = () => {
setCorporateModalVisible(false);
setCorporateCompanyName('');
setCorporateBusinessInfo(null);
setCorporateDishonestyInfo(null);
setCorporateError(null);
setCorporateUpdatedAt(null);
};
const renderFieldCard = (fieldKey: string, fieldValue: string) => {
const page = getLeauditTargetPage(reviewPoint, fieldKey);
const bboxHighlight = getLeauditBboxHighlight(reviewPoint, fieldKey, page);
const enterpriseButton =
shouldShowEnterpriseButtons && fieldKey === '甲方名称' && partyAName
? renderEnterpriseInfoButton('甲方企业信息', partyAName)
: shouldShowEnterpriseButtons && fieldKey === '乙方名称' && partyBName
? renderEnterpriseInfoButton('乙方企业信息', partyBName)
: null;
return (
<button
key={fieldKey}
type="button"
className={`w-full border rounded-md text-left transition ${page ? 'hover:bg-[#f6ffed] hover:border-[#b7eb8f]' : 'opacity-90'} border-slate-200 bg-slate-50 field-btn`}
onClick={() => {
if (page) onReviewPointSelect(reviewPoint.id, page, undefined, fieldValue, bboxHighlight);
}}
>
<div className="p-2.5 flex-1 min-w-0">
<div className="flex items-center justify-between gap-2">
<div className="flex items-center gap-2 min-w-0">
<div className="inline-flex items-center text-[11px] text-slate-500 truncate font-medium">{fieldKey}</div>
{enterpriseButton && enterpriseButton}
</div>
<div className="flex items-center gap-2 shrink-0">
{page && <span className="text-[10.5px] text-slate-400 shrink-0">P{page}</span>}
</div>
</div>
<div
className="text-[12px] text-slate-700 mt-1 leading-relaxed whitespace-pre-wrap break-words select-text cursor-text"
onClick={(e) => e.stopPropagation()}
onMouseDown={(e) => e.stopPropagation()}
>
{fieldValue}
</div>
</div>
{/* <div className="w-8 shrink-0 flex items-center justify-center border-l border-slate-200">
<i className="ri-focus-3-line text-[#00684a] text-[16px]" />
</div> */}
</button>
);
};
const renderEnterpriseInfoButton = (label: string, companyName: string) => (
<button
type="button"
className={`inline-flex items-center gap-1 h-5 px-1.5 rounded border text-[10.5px] transition-colors flex-shrink-0 ${
companyName
? 'bg-[#00684a] text-white border-[#00684a] hover:bg-[#005a3f] hover:border-[#005a3f]'
: 'bg-slate-100 text-slate-400 border-slate-200 cursor-not-allowed'
}`}
disabled={!companyName}
onClick={(e) => {
e.stopPropagation();
if (companyName) {
void handleCorporateInfoClick(companyName);
}
}}
>
<i className="ri-building-4-line text-[11px]" />
{label}
</button>
);
const renderStageContent = (stage: Record<string, unknown>, index: number) => {
const detail = (stage.detail || {}) as Record<string, unknown>;
const checkType = String(stage.check_type || 'unknown');
const passed = stage.passed === true;
const hasPassedState = typeof stage.passed === 'boolean';
const stageCardClass = passed ? 'border-emerald-200 bg-emerald-50/60' : 'border-slate-200 bg-slate-50/60';
const stageBadgeClass = passed ? 'text-emerald-700' : 'text-slate-600';
const stageLabelMap: Record<string, string> = {
required: '字段必填',
match: '一致性比对',
ai: 'AI 评查',
contains: '包含校验',
compare: '比较校验',
};
const stageDisplayName = typeof stage.check_type_chinese === 'string' && stage.check_type_chinese.trim()
? stage.check_type_chinese.trim()
: (stageLabelMap[checkType] || checkType);
const stageReason = typeof stage.reason === 'string' ? stage.reason.trim() : '';
const getStageDisplayValue = (value: unknown) => {
if (value == null || value === '') return '—';
if (typeof value === 'string') return value;
if (typeof value === 'number' || typeof value === 'boolean') return String(value);
return stringifyUnknown(value);
};
const renderStageInfoRow = (
label: string,
value: unknown,
options?: { valueClassName?: string; mono?: boolean },
) => (
<div className="px-2.5 py-2 flex items-start justify-between gap-3 border-t border-slate-100">
<div className="text-[11px] text-slate-500 shrink-0">{label}</div>
<div className={`text-[11px] text-slate-700 text-left break-words whitespace-pre-wrap max-w-[72%] ml-auto ${options?.mono ? 'font-mono' : ''} ${options?.valueClassName || ''}`}>
{getStageDisplayValue(value)}
</div>
</div>
);
if (checkType === 'required') {
const fields = Array.isArray(detail.fields) ? detail.fields.map(item => String(item)) : [];
const missing = Array.isArray(detail.missing) ? detail.missing.map(item => String(item)) : [];
return (
<div key={`stage-${index}`} className={`border rounded-md p-3 ${stageCardClass}`}>
<div className="flex items-center justify-between gap-2 mb-2">
<div className="text-[11px] font-medium text-slate-500 uppercase tracking-wider">{stageDisplayName}</div>
<span className={`text-[10.5px] inline-flex items-center gap-1 ${stageBadgeClass}`}>
<i className={passed ? 'ri-checkbox-circle-fill' : 'ri-error-warning-fill'} />{passed ? '通过' : '缺失'}
</span>
</div>
{fields.length > 0 && <div className="text-[12px] text-slate-700">{`命中字段:${fields.join('、')}`}</div>}
{missing.length > 0 && <div className="text-[12px] text-amber-700 mt-1">{`缺失字段:${missing.join('、')}`}</div>}
</div>
);
return;
}
if (normalizedValue) {
onReviewPointSelect(
reviewPoint.id,
undefined,
fieldData?.char_positions,
normalizedValue,
if (checkType === 'match') {
const failures = Array.isArray(detail.failures)
? detail.failures.map(item => item as Record<string, unknown>)
: [];
return (
<div key={`stage-${index}`} className={`border rounded-md p-3 ${stageCardClass}`}>
<div className="flex items-center justify-between gap-2 mb-2">
<div className="text-[11px] font-medium text-slate-500 uppercase tracking-wider">{stageDisplayName}</div>
<span className={`text-[10.5px] inline-flex items-center gap-1 ${stageBadgeClass}`}>
<i className={passed ? 'ri-checkbox-circle-fill' : 'ri-error-warning-fill'} />{passed ? '通过' : '不一致'}
</span>
</div>
{failures.length > 0 ? (
<div className="space-y-2 mt-2">
{failures.map((failure, failureIndex) => {
const leftField = String(failure.a || '左侧字段');
const rightField = String(failure.b || '右侧字段');
const leftValue = failure.a_value == null ? '—' : String(failure.a_value);
const rightValue = failure.b_value == null ? '—' : String(failure.b_value);
return (
<div key={`failure-${index}-${failureIndex}`} className="border border-red-200 rounded-md bg-white/90 overflow-hidden">
<div className="px-3 py-1 flex items-center justify-between gap-2 border-b border-red-100 bg-red-50/70">
<div className="text-[11px] text-slate-500">{`差异项 ${failureIndex + 1}`}</div>
</div>
<div className="py-1">
<div className="px-2.5 py-1.5 flex items-start justify-between gap-3">
<div className="text-[11px] text-slate-500 max-w-[46%] basis-[46%] break-words whitespace-normal">{leftField}</div>
{failure.a_value == null ? (
<span className="text-[11px] text-red-600 inline-flex items-center justify-start gap-0.5 font-medium max-w-[46%] basis-[46%] whitespace-normal break-words text-left ml-auto">
<i className="ri-prohibited-line" />
{'未填写'}
</span>
) : (
<div className={`text-[11px] text-slate-800 max-w-[46%] basis-[46%] text-left break-words whitespace-normal ml-auto ${/^\d/.test(leftValue) ? 'font-mono' : ''}`}>
{leftValue}
</div>
)}
</div>
<div className="px-2.5 py-1.5 flex items-start justify-between gap-3 border-t border-slate-100">
<div className="text-[11px] text-slate-500 max-w-[46%] basis-[46%] break-words whitespace-normal">{rightField}</div>
{failure.b_value == null ? (
<span className="text-[11px] text-red-600 inline-flex items-center justify-start gap-0.5 font-medium max-w-[46%] basis-[46%] whitespace-normal break-words text-left ml-auto">
<i className="ri-prohibited-line" />
{'未填写'}
</span>
) : (
<div className={`text-[11px] text-slate-800 max-w-[46%] basis-[46%] text-left break-words whitespace-normal ml-auto ${/^\d/.test(rightValue) ? 'font-mono' : ''}`}>
{rightValue}
</div>
)}
</div>
</div>
</div>
);
})}
</div>
) : (
<div className="text-[12px] text-slate-600">{'未发现不一致项'}</div>
)}
</div>
);
toastService.info(`${fieldName} 当前没有页码,已改为按文本定位`);
return;
}
toastService.info(`${fieldName} 当前既没有页码,也没有可定位文本`);
if (checkType === 'compare') {
const opMap: Record<string, string> = {
'>=': '≥', '<=': '≤', '!=': '≠', '<>': '≠', '==': '=', '>': '>', '<': '<', '=': '=',
};
const displayOp = opMap[String(detail.op)] || String(detail.op);
const fmtNum = (v: unknown) => {
if (v == null || String(v).trim() === '') return getStageDisplayValue(v);
const n = Number(v);
return !isNaN(n) ? n.toLocaleString('zh-CN') : getStageDisplayValue(v);
};
const buildOperand = (field: unknown, value: unknown) => {
const fieldStr = getStageDisplayValue(field);
if (value == null || value === '') return fieldStr;
return `${fieldStr}(${fmtNum(value)})`;
};
const leftOperand = buildOperand(detail.left, detail.left_value);
const rightDisplay = typeof detail.right === 'number'
? fmtNum(detail.right)
: buildOperand(detail.right, detail.right_value);
return (
<div key={`stage-${index}`} className={`border rounded-md p-3 ${stageCardClass}`}>
<div className="flex items-center justify-between gap-2 mb-2">
<div className="text-[11px] font-medium text-slate-500 uppercase tracking-wider">{stageDisplayName}</div>
<span className={`text-[10.5px] inline-flex items-center gap-1 ${stageBadgeClass}`}>
<i className={passed ? 'ri-checkbox-circle-fill' : 'ri-close-circle-fill'} />
{passed ? '通过' : '未通过'}
</span>
</div>
<div className={`text-[12px] font-mono ${passed ? 'text-slate-800' : 'text-red-800'}`}>
{leftOperand} <span className="mx-1 font-bold">{displayOp}</span> {rightDisplay}
</div>
{stageReason && (
<div className="text-[11px] text-slate-500 mt-1">{stageReason}</div>
)}
</div>
);
}
if (checkType === 'ai') {
const response = (detail.response || {}) as Record<string, unknown>;
const reasonText = typeof response.reason === 'string' ? response.reason.trim() : '';
const strengthItems = normalizeAiResponseItems(response.strengths);
const suggestionItems = normalizeAiResponseItems(response.suggestion, { hideNone: true });
const dividerClass = passed ? 'border-emerald-200/70' : 'border-fuchsia-200/70';
return (
<section key={`stage-${index}`} className="px-0 pt-0">
<div className="text-[11px] font-medium text-slate-400 uppercase tracking-wider mb-2 flex items-center gap-1">
<i className="ri-sparkling-2-fill text-fuchsia-500 text-[12px]" />
{'AI 评查意见'}
</div>
<div className={`p-3 border rounded-md space-y-3 ${passed ? 'bg-emerald-50/70 border-emerald-200' : 'bg-fuchsia-50/60 border-fuchsia-200'}`}>
{reasonText && (
<div className="flex gap-2 text-[12.5px] text-slate-700 leading-relaxed">
<i className={`${passed ? 'ri-checkbox-circle-line text-emerald-500' : 'ri-error-warning-line text-fuchsia-500'} shrink-0 mt-0.5`} />
<div className="whitespace-pre-wrap break-words">{reasonText}</div>
</div>
)}
{strengthItems.length > 0 && (
<div className={`pt-2 border-t ${dividerClass}`}>
<div className="flex items-center gap-1 text-[11px] font-medium text-emerald-700 mb-1.5">
<i className="ri-medal-line" />
{'亮点'}
<span className="font-mono text-[10.5px] text-emerald-500">{strengthItems.length}</span>
</div>
<ul className="space-y-1 text-[12px] text-slate-700">
{strengthItems.map((item, itemIndex) => (
<li key={`strength-${index}-${itemIndex}`} className="flex gap-1.5">
<i className="ri-check-line text-emerald-500 mt-[1px] shrink-0" />
<span className="break-words">{item}</span>
</li>
))}
</ul>
</div>
)}
{suggestionItems.length > 0 && (
<div className={`pt-2 border-t ${dividerClass}`}>
<div className="flex items-center gap-1 text-[11px] font-medium text-fuchsia-700 mb-1.5">
<i className="ri-edit-2-line" />
{'修改建议'}
<span className="font-mono text-[10.5px] text-fuchsia-500">{suggestionItems.length}</span>
</div>
<ul className="space-y-1 text-[12px] text-slate-700">
{suggestionItems.map((item, itemIndex) => (
<li key={`suggestion-${index}-${itemIndex}`} className="flex gap-1.5">
<span className="text-fuchsia-400 shrink-0">{'•'}</span>
<span className="break-words">{item}</span>
</li>
))}
</ul>
</div>
)}
</div>
</section>
);
}
return (
<div key={`stage-${index}`} className={`border rounded-md p-3 ${stageCardClass}`}>
<div className="flex items-center justify-between gap-2 mb-2">
<div className="text-[11px] font-medium text-slate-500 uppercase tracking-wider">
{stageDisplayName}
</div>
{hasPassedState && (
<span className={`inline-flex items-center gap-1 px-1.5 h-5 rounded text-[10.5px] border ${
passed
? 'bg-emerald-50 text-emerald-700 border-emerald-200'
: 'bg-red-50 text-red-700 border-red-200'
}`}>
<i className={passed ? 'ri-checkbox-circle-fill' : 'ri-close-circle-fill'} />
{passed ? '通过' : '未通过'}
</span>
)}
</div>
<div className="rounded-md border border-slate-200 bg-white/80 overflow-hidden">
{/* {renderStageInfoRow('阶段类型', stageDisplayName)} */}
{/* {hasPassedState && renderStageInfoRow('结果', passed ? '通过' : '未通过', { valueClassName: passed ? 'text-emerald-700' : 'text-red-700' })} */}
{stageReason && renderStageInfoRow('原因', stageReason)}
</div>
</div>
);
};
return (
<div className={`mb-3 rounded-md border ${passed ? 'border-emerald-200 bg-emerald-50/60' : 'border-amber-200 bg-amber-50/70'} p-3`}>
<div className="flex items-center justify-between gap-2">
<div className="text-[11px] font-medium text-slate-600">{badgeText}</div>
<span className={`inline-flex items-center gap-1 rounded px-1.5 py-0.5 text-[10.5px] ${passed ? 'bg-emerald-100 text-emerald-700' : 'bg-amber-100 text-amber-700'}`}>
<i className={passed ? 'ri-checkbox-circle-line' : 'ri-error-warning-line'} />
{passed ? '通过' : '未通过'}
</span>
</div>
{reason && (
<div className="mt-2 text-[12px] leading-5 text-slate-700">
{reason}
<>
<article className="bg-white border border-slate-200 rounded-lg shadow-sm overflow-hidden">
<header className="px-4 pt-4 pb-3 border-b border-slate-100">
<div className="flex items-center gap-2 flex-wrap">
{reviewPoint.pointId && (
<span className="font-mono text-[11px] px-1.5 py-0.5 rounded bg-slate-100 text-slate-600 shrink-0">
{reviewPoint.pointId}
</span>
)}
<h2 className="text-[14.5px] font-semibold text-slate-900 break-all leading-snug">{reviewPoint.pointName}</h2>
</div>
<div className="mt-2 flex items-center justify-between gap-3 flex-wrap">
<div className="flex items-center gap-2 flex-wrap">
<span className={`inline-flex items-center gap-1 px-1.5 h-5 rounded text-[10.5px] border ${statusChip.cls}`}>
<i className={statusChip.icon} />{statusChip.label}
</span>
<span className={`inline-flex items-center gap-1 px-1.5 h-5 rounded text-[10.5px] border ${riskMeta.cls}`}>
<i className="ri-focus-3-line" />{riskMeta.label}
</span>
{reviewPoint.postAction === 'manual' && (
<span className="inline-flex items-center gap-1 px-1.5 h-5 rounded text-[10.5px] bg-slate-50 text-slate-600 border border-slate-200">
<i className="ri-user-line" />
</span>
)}
</div>
<div className="flex items-center gap-3 text-[11px] text-slate-500">
{reviewPoint.score != null && <span> <span className="font-mono text-slate-700">{reviewPoint.finalScore ?? reviewPoint.machineScore ?? reviewPoint.score}/{reviewPoint.score}</span></span>}
{confidencePct && <span> <span className="font-mono text-slate-700">{confidencePct}</span></span>}
</div>
</div>
</header>
{Object.keys(reviewPoint.content || {}).length > 0 && (
<section className="px-4 py-2">
<div className="text-[11px] font-medium text-slate-400 uppercase tracking-wider mb-2">
<span className="font-mono normal-case text-[10.5px]">{Object.keys(reviewPoint.content).length}</span>
</div>
<div className="space-y-2">
{Object.entries(reviewPoint.content).map(([key, value]) => renderFieldCard(key, getLeauditFieldText(value)))}
</div>
</section>
)}
{fieldNames.length > 0 && (
<div className="mt-3 flex flex-wrap gap-2">
{fieldNames.map((fieldName) => {
const { fieldData, normalizedPage, normalizedValue, canLocate } = getFieldLocatorState(fieldName);
const fieldValue = fieldData?.value;
const displayValue =
typeof fieldValue === 'string'
? fieldValue
: fieldValue == null
? '未抽取到值'
: JSON.stringify(fieldValue);
return (
<button
key={fieldName}
type="button"
className={`min-w-0 flex-1 rounded border px-2.5 py-2 text-left ${
canLocate
? 'border-slate-200 bg-white hover:border-[#00684a] hover:bg-[#f6fffb]'
: 'border-slate-200 bg-slate-50 text-slate-400 cursor-not-allowed'
}`}
onClick={() => canLocate && jumpToField(fieldName)}
disabled={!canLocate}
>
<div className="flex items-center justify-between gap-2 text-[11px] font-medium">
<span className={canLocate ? 'text-slate-500' : 'text-slate-400'}>{fieldName}</span>
{normalizedPage ? (
<span className="rounded bg-emerald-50 px-1.5 py-0.5 text-[10px] text-emerald-700">
{normalizedPage}
</span>
) : normalizedValue ? (
<span className="rounded bg-amber-50 px-1.5 py-0.5 text-[10px] text-amber-700">
</span>
) : (
<span className="rounded bg-slate-100 px-1.5 py-0.5 text-[10px] text-slate-500">
</span>
)}
{missingItems.length > 0 && (
<section className="px-4 py-2">
<div className="text-[11px] font-medium text-slate-400 uppercase tracking-wider mb-2">
<span className="font-mono normal-case text-[10.5px]">{missingItems.length}</span>
</div>
<div className="space-y-2">
{missingItems.map(item => (
<div key={item} className="w-full border border-red-200 rounded-md bg-red-50/60">
<div className="p-2.5 flex items-start justify-between gap-2">
<div className="min-w-0 text-[12px] text-slate-700 leading-relaxed break-words">{item}</div>
<span className="text-[10.5px] text-red-600 shrink-0"></span>
</div>
<div className="mt-1 text-[12px] leading-5 text-slate-700 break-all">{displayValue}</div>
</button>
);
})}
</div>
</div>
))}
</div>
</section>
)}
</div>
{stages.length > 0 && (
<section className="px-4 py-2">
<div className="text-[11px] font-medium text-slate-400 uppercase tracking-wider mb-2">
<span className="font-mono normal-case text-[10.5px]">{stages.length}</span>
</div>
<div className="space-y-2">
{stages.map((stage, index) => renderStageContent(stage, index))}
</div>
</section>
)}
{!isWarning && (
<section className="px-4 py-2">
<div className="text-[11px] font-medium text-slate-400 uppercase tracking-wider mb-2 flex items-center gap-1">
<i className={isPass ? 'ri-shield-check-line text-emerald-500 text-[12px]' : 'ri-close-circle-line text-red-500 text-[12px]'} />
{isPass ? '校验结果' : '问题说明'}
</div>
<div className={`p-3 rounded-md border text-[12.5px] leading-relaxed ${isPass ? 'bg-emerald-50/60 border-emerald-200 text-slate-700' : 'bg-red-50/60 border-red-200 text-slate-700'}`}>
{summaryText}
</div>
</section>
)}
{legalBasisList.length > 0 && (
<section className="px-4 py-2">
<div className="text-[11px] font-medium text-slate-400 uppercase tracking-wider mb-2"></div>
<div className="flex flex-wrap gap-2">
{legalBasisList.map((item, index) => (
<span key={`${item}-${index}`} className="bg-[#e6f4ff] border border-[#91caff] rounded-full px-2 py-0.5 text-xs text-[#0958d9]">
{item}
</span>
))}
</div>
</section>
)}
{reviewPoint.postAction === 'manual' && (
<section className="px-4 py-2">
<div className="text-[11px] font-medium text-slate-400 uppercase tracking-wider mb-2"></div>
<textarea
rows={2}
placeholder="请输入审核意见..."
className={`w-full p-2 border border-slate-200 rounded-md text-[12.5px] min-h-[56px] focus:outline-none focus:border-[#00684a] focus:ring-2 focus:ring-[#00684a]/15 resize-none placeholder:text-slate-400 ${reviewPoint.editAuditStatus !== 0 ? 'bg-gray-100 cursor-not-allowed' : 'bg-white'}`}
value={manualNote}
onChange={(e) => setManualNote(e.target.value)}
disabled={reviewPoint.editAuditStatus !== 0}
/>
</section>
)}
{reviewPoint.postAction === 'manual' && reviewPoint.editAuditStatus === 0 && (
<footer className="mt-3 px-4 py-3 flex items-center justify-end gap-2 border-t border-slate-100 bg-slate-50/60">
<button type="button" className="h-8 px-3 rounded-md text-[12.5px] bg-[#1890ff] text-white hover:bg-blue-600 flex items-center gap-1 font-medium" onClick={() => {
if (manualNote.trim() === '') { toastService.error('请输入审核意见'); return; }
onStatusChange(reviewPoint.id, reviewPoint.editAuditStatusId || '', 'true', manualNote);
}}>
<i className="ri-check-line" />
</button>
<button type="button" className="h-8 px-3 rounded-md text-[12.5px] bg-[#f5222d] text-white hover:bg-red-600 flex items-center gap-1 font-medium shadow-sm" onClick={() => {
if (manualNote.trim() === '') { toastService.error('请输入审核意见'); return; }
onStatusChange(reviewPoint.id, reviewPoint.editAuditStatusId || '', 'false', manualNote);
}}>
<i className="ri-close-line" />
</button>
</footer>
)}
{reviewPoint.postAction === 'manual' && reviewPoint.editAuditStatus !== 0 && (
<footer className="mt-3 px-4 py-3 flex items-center justify-end border-t border-slate-100 bg-slate-50/60">
<button type="button" className="h-8 px-3 rounded-md text-[12.5px] bg-purple-600 text-white hover:bg-purple-700 flex items-center gap-1 font-medium" onClick={() => {
onStatusChange(reviewPoint.id, reviewPoint.editAuditStatusId || '', 'review', '');
}}>
<i className="ri-refresh-line" />
</button>
</footer>
)}
{isPass && reviewPoint.postAction !== 'manual' && (
<footer className="px-4 py-3 flex items-center gap-2 border-t border-slate-100 bg-slate-50/60">
<span className="text-[11px] text-emerald-700 inline-flex items-center gap-1">
<i className="ri-verified-badge-fill" />
</span>
</footer>
)}
</article>
<CorporateInfoModal
visible={corporateModalVisible}
onClose={handleCloseCorporateModal}
companyName={corporateCompanyName}
businessInfo={corporateBusinessInfo}
dishonestyInfo={corporateDishonestyInfo}
businessLoading={corporateLoading}
dishonestyLoading={corporateLoading}
businessError={corporateError}
dishonestyError={corporateError}
updatedAt={corporateUpdatedAt}
onForceRefresh={handleCorporateForceRefresh}
/>
</>
);
}
// ── Main Component ──
export function ReviewPointDetailCard({ reviewPoint, onReviewPointSelect, onStatusChange, fileFormat }: ReviewPointDetailCardProps) {
const resolveManualNote = () => {
if (reviewPoint.editAuditStatusMessage) {
return reviewPoint.editAuditStatusMessage;
}
if (typeof reviewPoint.actionContent === 'string') {
return reviewPoint.actionContent;
}
if (reviewPoint.suggestion) {
return reviewPoint.suggestion;
}
return '';
};
const [manualNote, setManualNote] = useState(resolveManualNote);
function LegacyReviewPointDetailCard({ reviewPoint, onReviewPointSelect, onStatusChange, fileFormat }: ReviewPointDetailCardProps) {
const [manualNote, setManualNote] = useState(
() => reviewPoint.editAuditStatusMessage || normalizeActionContent(reviewPoint.actionContent) || reviewPoint.suggestion || ''
);
// reviewPoint 切换时重置默认值
useEffect(() => {
setManualNote(resolveManualNote());
setManualNote(reviewPoint.editAuditStatusMessage || normalizeActionContent(reviewPoint.actionContent) || reviewPoint.suggestion || '');
}, [reviewPoint.id, reviewPoint.editAuditStatusMessage, reviewPoint.actionContent, reviewPoint.suggestion]);
const otherRules = filterOtherRule(reviewPoint);
@@ -657,7 +1231,7 @@ export function ReviewPointDetailCard({ reviewPoint, onReviewPointSelect, onStat
if (rule.type === 'ai') {
return <div key={`rule-${i}`}>{otherRules.length > 0 && <div className="bg-gray-50 rounded border border-gray-200 text-xs mb-3" />}<RenderModelRule rule={rule} reviewPoint={reviewPoint} onReviewPointSelect={onReviewPointSelect} fileFormat={fileFormat} /></div>;
}
return <RenderGenericRule key={`rule-${i}`} rule={rule} reviewPoint={reviewPoint} onReviewPointSelect={onReviewPointSelect} />;
return null;
})}
</section>
@@ -727,3 +1301,17 @@ export function ReviewPointDetailCard({ reviewPoint, onReviewPointSelect, onStat
</article>
);
}
export function ReviewPointDetailCard(props: ReviewPointDetailCardProps) {
if (props.detailMode === 'leaudit') {
return (
<LeauditReviewPointDetailCard
reviewPoint={props.reviewPoint}
onReviewPointSelect={props.onReviewPointSelect}
onStatusChange={props.onStatusChange}
/>
);
}
return <LegacyReviewPointDetailCard {...props} />;
}
+278 -148
View File
@@ -45,8 +45,7 @@ import { PdfPreviewTest } from "~/components/reviews/previewComponents/PdfPrevie
import { DocxPreviewTest } from "~/components/reviews/previewComponents/DocxPreviewTest";
import { ComparePreview } from "~/components/reviews/previewComponents/ComparePreview";
import { type ReviewPoint } from '~/components/reviews';
import { messageService } from "~/components/ui/MessageModal";
import { type ReviewPoint, type PdfBboxHighlight } from '~/components/reviews';
import { loadingBarService } from "~/components/ui/LoadingBar";
/**
@@ -155,6 +154,8 @@ interface ReviewData {
aiAnalysis: AnalysisData;
}
type PreviewKind = 'pdf' | 'docx';
type PreviewDocument = {
path?: string;
attachments?: Array<{
@@ -163,10 +164,14 @@ type PreviewDocument = {
}>;
};
interface DefaultPreviewTarget {
page?: number;
highlightValue?: string;
bboxHighlight?: PdfBboxHighlight;
}
function resolvePreviewPath(document: PreviewDocument | null | undefined): string {
if (document?.path) {
return document.path;
}
if (document?.path) return document.path;
const primaryAttachment = Array.isArray(document?.attachments)
? document.attachments.find((item) => item?.fileRole === 'primary' && item?.ossUrl)
@@ -181,37 +186,124 @@ function resolvePreviewExtension(document: PreviewDocument | null | undefined):
return typeof suffix === 'string' ? suffix.toLowerCase() : '';
}
function buildReviewData(document: any, reviewPoints: ReviewPoint[], statistics: Statistics, reviewInfo: ReviewInfo): ReviewData | null {
if (!document) {
return null;
}
function isValidQuad(value: unknown): value is [number, number, number, number] {
return Array.isArray(value) && value.length === 4 && value.every(item => typeof item === 'number' && Number.isFinite(item));
}
const mockData = getMockReviewData();
const typeValue = document.type ?? document.type_id;
function hasNonZeroQuad(value: [number, number, number, number]): boolean {
return value.some(item => item !== 0);
}
function getReviewPointContentText(value: unknown): string | undefined {
if (value == null) return undefined;
if (typeof value === 'string' || typeof value === 'number' || typeof value === 'boolean') {
const text = String(value).trim();
return text || undefined;
}
if (typeof value === 'object' && value && 'value' in value) {
return getReviewPointContentText((value as { value?: unknown }).value);
}
return undefined;
}
function getReviewPointFieldPage(point: ReviewPoint, fieldKey: string, rawValue: unknown): number | undefined {
const contentPage = point.contentPage?.[fieldKey];
const normalizedContentPage = Number(contentPage);
if (Number.isFinite(normalizedContentPage) && normalizedContentPage > 0) return normalizedContentPage;
const inlinePage = typeof rawValue === 'object' && rawValue && 'page' in rawValue
? Number((rawValue as { page?: unknown }).page)
: NaN;
if (Number.isFinite(inlinePage) && inlinePage > 0) return inlinePage;
const pageNum = point.fieldPositions?.[fieldKey]?.page_num;
if (typeof pageNum === 'number' && Number.isFinite(pageNum) && pageNum >= 0) return pageNum + 1;
return undefined;
}
function getReviewPointFieldBbox(point: ReviewPoint, fieldKey: string, page?: number): PdfBboxHighlight | undefined {
const fieldPosition = point.fieldPositions?.[fieldKey];
if (!fieldPosition || !isValidQuad(fieldPosition.bbox) || !isValidQuad(fieldPosition.page_box)) return undefined;
if (!hasNonZeroQuad(fieldPosition.bbox) || !hasNonZeroQuad(fieldPosition.page_box)) return undefined;
return {
fileInfo: {
fileName: document.name || "未知文件名",
path: document.path || "未知路径",
contractNumber: document.documentNumber || document.document_number || "未知编号",
fileSize: document.size ? formatFileSize(document.size) : document.file_size ? formatFileSize(document.file_size) : "未知大小",
fileFormat: document.fileType ? document.fileType.toUpperCase() : "未知格式",
pageCount: document.pageCount || document.page_count || 0,
uploadTime: document.uploadTime || document.created_at || "未知时间",
uploadUser: document.uploadUser || "未知用户",
auditStatus: document.auditStatus || 0,
legalBasis: document.legalBasis || {},
fileType: typeValue !== undefined && typeValue !== null ? String(typeValue) : ""
},
contractInfo: mockData.contractInfo,
reviewInfo,
statistics,
fileContent: mockData.fileContent,
reviewPoints,
aiAnalysis: mockData.aiAnalysis,
fieldKey,
bbox: [...fieldPosition.bbox],
pageBox: [...fieldPosition.page_box],
pageNum: fieldPosition.page_num,
page,
confidence: fieldPosition.confidence,
matchMethod: fieldPosition.match_method,
};
}
function resolveDefaultPreviewTarget(point: ReviewPoint, previewKind: PreviewKind): DefaultPreviewTarget {
let firstPageCandidate: DefaultPreviewTarget | undefined;
let firstPageWithBboxCandidate: DefaultPreviewTarget | undefined;
let firstPageWithTextCandidate: DefaultPreviewTarget | undefined;
for (const [fieldKey, rawValue] of Object.entries(point.content || {})) {
const page = getReviewPointFieldPage(point, fieldKey, rawValue);
if (!page) continue;
const highlightValue = getReviewPointContentText(rawValue);
const bboxHighlight = getReviewPointFieldBbox(point, fieldKey, page);
const candidate: DefaultPreviewTarget = { page, highlightValue, bboxHighlight };
if (!firstPageCandidate) firstPageCandidate = candidate;
if (!firstPageWithBboxCandidate && bboxHighlight) firstPageWithBboxCandidate = candidate;
if (!firstPageWithTextCandidate && highlightValue) firstPageWithTextCandidate = candidate;
}
if (previewKind === 'pdf') {
return firstPageWithBboxCandidate || firstPageCandidate || {};
}
return firstPageWithTextCandidate || firstPageCandidate || {};
}
interface ReviewsTestLoaderSuccess {
previousRoute: string;
document: any;
reviewPoints: ReviewPoint[];
reviewInfo: ReviewInfo;
statistics: Statistics;
comparison_document: any;
userInfo: { sub: string; nick_name: string } | null;
frontendJWT: string | null;
flowType: 'legacy' | 'leaudit';
detailMode: 'legacy' | 'leaudit';
}
interface ReviewsTestLoaderError {
result: false;
message: string;
previousRoute: string;
}
type ReviewsTestLoaderData = ReviewsTestLoaderSuccess | ReviewsTestLoaderError;
const EMPTY_STATISTICS: Statistics = {
total: 0,
success: 0,
warning: 0,
error: 0,
notApplicable: 0,
score: 0,
};
const EMPTY_REVIEW_INFO: ReviewInfo = {
reviewTime: '',
reviewModel: '',
ruleGroup: '',
result: '',
issueCount: 0,
};
function isReviewsTestLoaderError(data: ReviewsTestLoaderData): data is ReviewsTestLoaderError {
return 'result' in data && data.result === false;
}
export const meta: MetaFunction = () => {
return [
@@ -233,49 +325,53 @@ export const handle = {
noPadding: true
};
export async function loader({ request }: LoaderFunctionArgs) {
export async function loader({ request }: LoaderFunctionArgs): Promise<Response> {
try {
const url = new URL(request.url);
const id = url.searchParams.get('id') || undefined;
const id = url.searchParams.get('id') || '';
const previousRoute = url.searchParams.get('previousRoute') || '';
// console.log("[Reviews Loader] 开始加载,id:", id, "previousRoute:", previousRoute);
if (!id) {
console.error("[Reviews Loader] 文件ID不能为空");
return Response.json({ result: false, message: '文件ID不能为空' });
return Response.json({ result: false, message: '文件ID不能为空', previousRoute });
}
// 获取用户会话信息
const { getUserSession } = await import("~/api/login/auth.server");
const { userInfo, frontendJWT } = await getUserSession(request);
// reviewsTest 正式链路统一只走新后端聚合接口,避免旧统一接口 404 噪音。
const reviewData = await getReviewPoints_fromApi(id, request);
if ('error' in reviewData && reviewData.error) {
console.error("[Reviews Loader] 获取评查点数据错误:", reviewData.error);
return Response.json({ result: false, message: reviewData.error });
return Response.json({
result: false,
message: reviewData.error,
previousRoute,
});
}
return Response.json({
previousRoute: previousRoute,
previousRoute,
document: reviewData.document,
reviewPoints: reviewData.data,
reviewInfo: reviewData.reviewInfo,
statistics: reviewData.stats,
statistics: { ...reviewData.stats, notApplicable: reviewData.stats?.notApplicable ?? 0 },
comparison_document: reviewData.comparison_document,
userInfo,
frontendJWT,
flowType: 'legacy',
scoredResults: null,
scoredSummary: null
userInfo:
userInfo?.sub && userInfo?.nick_name
? { sub: userInfo.sub, nick_name: userInfo.nick_name }
: null,
frontendJWT: frontendJWT ?? null,
flowType: 'leaudit',
detailMode: 'leaudit',
});
} catch (error) {
console.error('[Reviews Loader] 获取评查数据失败:', error);
console.error('[Reviews Loader] 错误堆栈:', error instanceof Error ? error.stack : '无堆栈信息');
return Response.json({ result: false, message: `获取评查数据失败: ${error instanceof Error ? error.message : '未知错误'}` });
console.error('[reviewsTest loader] Failed to load review data:', error);
return Response.json({
result: false,
message: `获取评查数据失败: ${error instanceof Error ? error.message : '未知错误'}`,
previousRoute: '',
});
}
}
// 添加 action 函数处理需要用户认证的操作
export async function action({ request }: ActionFunctionArgs) {
try {
const formData = await request.formData();
@@ -346,36 +442,27 @@ export async function action({ request }: ActionFunctionArgs) {
export default function ReviewDetails() {
const navigate = useNavigate();
const loaderData = useLoaderData<typeof loader>();
const normalizedLoaderData =
loaderData &&
typeof loaderData === 'object' &&
'reviewPoints' in loaderData &&
loaderData.reviewPoints &&
typeof loaderData.reviewPoints === 'object' &&
'data' in loaderData.reviewPoints &&
'document' in loaderData.reviewPoints
? {
...loaderData,
document: (loaderData.reviewPoints as any).document,
reviewPoints: (loaderData.reviewPoints as any).data,
reviewInfo: (loaderData.reviewPoints as any).reviewInfo,
statistics: (loaderData.reviewPoints as any).stats,
comparison_document: (loaderData.reviewPoints as any).comparison_document,
scoring_proposals: (loaderData.reviewPoints as any).scoring_proposals ?? [],
}
: loaderData;
const loaderData = useLoaderData<ReviewsTestLoaderData>();
const fetcher = useFetcher();
const { document, reviewPoints, statistics, reviewInfo, comparison_document, frontendJWT } = normalizedLoaderData as any;
const fallbackReviewData = buildReviewData(document, reviewPoints, statistics, reviewInfo);
const isLoaderError = isReviewsTestLoaderError(loaderData);
const successLoaderData = isLoaderError ? null : loaderData;
const document = successLoaderData?.document ?? null;
const reviewPoints = successLoaderData?.reviewPoints ?? [];
const statistics = successLoaderData?.statistics ?? EMPTY_STATISTICS;
const reviewInfo = successLoaderData?.reviewInfo ?? EMPTY_REVIEW_INFO;
const comparison_document = successLoaderData?.comparison_document ?? null;
const detailMode = successLoaderData?.detailMode ?? 'legacy';
const currentUserInfo = successLoaderData?.userInfo ?? null;
const frontendJWT = successLoaderData?.frontendJWT ?? null;
const [isLoading, setIsLoading] = useState(false); // 已经通过loader加载了数据,不需要再显示加载状态
const [rightActiveTab, setRightActiveTab] = useState<'result' | 'fields' | 'info'>('result');
const [reviewData, setReviewData] = useState<ReviewData | null>(fallbackReviewData);
const [reviewData, setReviewData] = useState<ReviewData | null>(null);
const [activeReviewPointResultId, setActiveReviewPointResultId] = useState<string | number | null>(null);
const [targetPage, setTargetPage] = useState<number | undefined>(undefined);
const [charPositions, setCharPositions] = useState<Array<{ box: number[][], char: string, score: number }> | undefined>(undefined);
const [bboxHighlight, setBboxHighlight] = useState<PdfBboxHighlight | undefined>(undefined);
const [highlightValue, setHighlightValue] = useState<string | undefined>(undefined);
const [pendingUpdate, setPendingUpdate] = useState<{
reviewPointResultId: string;
@@ -416,32 +503,14 @@ export default function ReviewDetails() {
// console.log('评查信息:', reviewInfo);
// console.log('比对文档:', comparison_document);
// console.log('用户信息:', loaderData.userInfo);
// console.log('JWT Token (前20位):', frontendJWT?.substring(0, 20) + '...');
// console.groupEnd();
// }
// }, [loaderData, document, reviewPoints, statistics, reviewInfo, comparison_document, frontendJWT]);
// }, [loaderData, document, reviewPoints, statistics, reviewInfo, comparison_document]);
// loader 数据加载出错
useEffect(()=>{
loadingBarService.hide();
// console.log('[Reviews Component] useEffect检查loaderData:', {
// hasResultKey: Object.keys(loaderData).find(key => key === 'result'),
// resultValue: loaderData.result,
// willNavigateBack: Object.keys(loaderData).find(key => key === 'result') && !loaderData.result
// });
if(Object.keys(loaderData).find(key => key === 'result') && !loaderData.result){
messageService.show({
title: '错误',
message: loaderData.message,
type: 'error',
confirmText: '确定',
cancelText: '',
onConfirm: () => {
navigate(-1);
}
})
}
},[loaderData, navigate]);
},[loaderData]);
// 当文档 ID 变化时,清空高亮相关的状态
@@ -450,17 +519,49 @@ export default function ReviewDetails() {
setActiveReviewPointResultId(null);
setTargetPage(undefined);
setCharPositions(undefined);
setBboxHighlight(undefined);
setHighlightValue(undefined);
}
}, [document?.id]);
// 使用 loader 数据同步本地评查页状态,避免首屏空白。
// 模拟获取评查数据
useEffect(() => {
setReviewData(buildReviewData(document, reviewPoints, statistics, reviewInfo));
if (!document) return;
// 构建文件信息对象
const fileInfo = {
fileName: document.name || "未知文件名",
path: document.path || "未知路径",
contractNumber: document.documentNumber || document.document_number || "未知编号",
fileSize: document.size ? formatFileSize(document.size) : document.file_size ? formatFileSize(document.file_size) : "未知大小",
// 文件格式类型
fileFormat: document.fileType ? document.fileType.toUpperCase() : "未知格式",
pageCount: document.pageCount || document.page_count || 0,
uploadTime: document.uploadTime || document.created_at || "未知时间",
uploadUser: document.uploadUser || "未知用户",
auditStatus: document.auditStatus || 0,
legalBasis: document.legalBasis || {},
// 文件类型(1:合同,2:卷宗。。。)
fileType: document.type || document.type_id ? document.type_id.toString() : ''
};
// 创建包含真实文档数据的评查数据对象
const reviewDataObj: ReviewData = {
// 使用真实文件信息
fileInfo: fileInfo,
// 其他字段暂时使用默认值
contractInfo: getMockReviewData().contractInfo,
reviewInfo: reviewInfo,
statistics: statistics,
fileContent: getMockReviewData().fileContent,
reviewPoints: reviewPoints,
aiAnalysis: getMockReviewData().aiAnalysis,
};
setReviewData(reviewDataObj);
setIsLoading(false);
}, [document, reviewPoints, statistics, reviewInfo]);
const effectiveReviewData = reviewData ?? fallbackReviewData;
const handleTabChange = (tabKey: 'result' | 'fields' | 'info') => {
setRightActiveTab(tabKey);
@@ -468,47 +569,36 @@ export default function ReviewDetails() {
// 从左栏选择评查点
const handleRuleSelect = (id: string | number) => {
setActiveReviewPointResultId(id);
setRightActiveTab('result');
// 查找评查点并尝试跳转到其页面
const point = effectiveReviewData?.reviewPoints.find(p => p.id === id);
const point = reviewData?.reviewPoints.find(p => p.id === id);
if (point) {
console.log('跳转到评查点页面:', point);
const page = getFirstPageFromPoint(point);
if (page) setTargetPage(page);
else setTargetPage(undefined);
setCharPositions(undefined);
setHighlightValue(undefined);
console.log('选择的评查点:', point);
const previewKind: PreviewKind = previewExtension === 'docx' ? 'docx' : 'pdf';
const defaultTarget = resolveDefaultPreviewTarget(point, previewKind);
handleReviewPointSelect(
id,
defaultTarget.page,
undefined,
defaultTarget.highlightValue,
previewKind === 'pdf' ? defaultTarget.bboxHighlight : undefined,
);
return;
}
handleReviewPointSelect(id);
};
// 从评查点中提取第一个有效页码
const getFirstPageFromPoint = (point: ReviewPoint): number | undefined => {
if (point.content) {
for (const data of Object.values(point.content)) {
const page = (data as any)?.page;
if (page && Number(page) > 0) return Number(page);
}
}
if (point.contentPage) {
for (const page of Object.values(point.contentPage)) {
if (page && Number(page) > 0) return Number(page);
}
}
return undefined;
};
// 下载文件
const handleDownloadFile = async () => {
try {
const downloadUrl = `/api/pdf-proxy?path=${encodeURIComponent(document?.path || '')}`;
const downloadUrl = `/api/pdf-proxy?path=${encodeURIComponent(previewPath)}`;
const response = await axios.get(downloadUrl, { responseType: 'blob' });
const blobUrl = URL.createObjectURL(response.data);
const a = window.document.createElement('a');
a.style.display = 'none';
a.href = blobUrl;
a.download = decodeURIComponent(document?.path?.split('/').pop() || 'document');
a.download = decodeURIComponent(previewPath.split('/').pop() || 'document');
window.document.body.appendChild(a);
a.click();
setTimeout(() => {
@@ -521,17 +611,19 @@ export default function ReviewDetails() {
}
};
const handleReviewPointSelect = (reviewPointId: string | number, page?: number, charPos?: Array<{ box: number[][], char: string, score: number }>, value?: string) => {
const handleReviewPointSelect = (reviewPointId: string | number, page?: number, charPos?: Array<{ box: number[][], char: string, score: number }>, value?: string, nextBboxHighlight?: PdfBboxHighlight) => {
// 如果点击的是相同的评查点,但有page参数,先重置targetPage以确保useEffect能够触发
if (reviewPointId === activeReviewPointResultId && page) {
setTargetPage(undefined);
setCharPositions(undefined);
setBboxHighlight(undefined);
setHighlightValue(undefined);
// 使用setTimeout确保状态更新后再设置新的targetPage、charPositions和highlightValue
setTimeout(() => {
setActiveReviewPointResultId(reviewPointId);
setTargetPage(page);
setCharPositions(charPos);
setBboxHighlight(nextBboxHighlight);
setHighlightValue(value);
}, 0);
} else {
@@ -539,10 +631,16 @@ export default function ReviewDetails() {
setActiveReviewPointResultId(reviewPointId);
setTargetPage(page);
setCharPositions(charPos);
setBboxHighlight(nextBboxHighlight);
setHighlightValue(value);
}
};
// 处理AI建议替换(保留空实现,右栏详情卡片中DOCX替换需要)
const handleAiSuggestionReplace = (_searchText: string, _replaceText: string, _pageNumber: number) => {
// PDF 文件不支持替换,暂不实现
};
// 刷新评审数据
// async function refreshReviewData(documentId: string) {
// // 设置加载状态
@@ -779,7 +877,7 @@ export default function ReviewDetails() {
};
// 获取当前激活的评查点对象
const activeReviewPoint = effectiveReviewData?.reviewPoints.find(p => p.id === activeReviewPointResultId) || null;
const activeReviewPoint = reviewData?.reviewPoints.find(p => p.id === activeReviewPointResultId) || null;
// ── 模板上传相关函数 ──
const handleOpenReuploadModal = () => {
@@ -821,7 +919,7 @@ export default function ReviewDetails() {
selectedTemplateFiles[0],
(document as any).id,
(comparison_document as any)?.comparisonId,
frontendJWT || undefined
frontendJWT || undefined,
);
if (uploadResult.error) throw new Error(uploadResult.error);
toastService.success('模板文件上传成功,即将返回上一页...');
@@ -833,14 +931,6 @@ export default function ReviewDetails() {
} finally { setIsUploading(false); }
};
const formatFileSize = (bytes: number) => {
if (bytes === 0) return '0 Bytes';
const k = 1024;
const sizes = ['Bytes', 'KB', 'MB', 'GB'];
const i = Math.floor(Math.log(bytes) / Math.log(k));
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
};
// ── 结构比对全页面视图 ──
if (showComparison) {
return (
@@ -849,7 +939,7 @@ export default function ReviewDetails() {
<button type="button" className="flex items-center gap-1 text-slate-600 hover:text-slate-900 text-[12.5px]" onClick={() => setShowComparison(false)}>
<i className="ri-arrow-left-line" />
</button>
<span className="font-medium text-sm text-slate-800 truncate">{effectiveReviewData?.fileInfo?.fileName}</span>
<span className="font-medium text-sm text-slate-800 truncate">{reviewData?.fileInfo?.fileName}</span>
</header>
<div className="flex-1 min-h-0 overflow-auto">
<Comparison comparison_document={comparison_document} />
@@ -858,6 +948,44 @@ export default function ReviewDetails() {
);
}
if (isLoaderError) {
return (
<div className="flex items-center justify-center min-h-screen bg-slate-50 px-6">
<div className="w-full max-w-xl rounded-xl border border-slate-200 bg-white shadow-sm p-8">
<div className="flex items-start gap-3">
<div className="w-10 h-10 rounded-full bg-amber-50 text-amber-600 flex items-center justify-center shrink-0">
<i className="ri-error-warning-line text-xl" />
</div>
<div className="min-w-0 flex-1">
<h1 className="text-lg font-semibold text-slate-900"></h1>
<p className="mt-2 text-sm text-slate-600 leading-6 break-words">
{loaderData.message || '文档不存在、当前账号无权限访问,或评查数据尚未准备完成。'}
</p>
<div className="mt-5 flex items-center gap-3">
<button
type="button"
className="inline-flex items-center gap-2 rounded-md bg-[#00684a] px-4 py-2 text-sm font-medium text-white hover:bg-[#00543c]"
onClick={() => navigate(getReturnUrl())}
>
<i className="ri-arrow-left-line" />
</button>
<button
type="button"
className="inline-flex items-center gap-2 rounded-md border border-slate-200 px-4 py-2 text-sm text-slate-600 hover:bg-slate-50"
onClick={() => window.location.reload()}
>
<i className="ri-refresh-line" />
</button>
</div>
</div>
</div>
</div>
</div>
);
}
return (
<div className="flex flex-col h-screen overflow-hidden">
{isLoading ? (
@@ -865,14 +993,14 @@ export default function ReviewDetails() {
<div className="loading-spinner"></div>
<span className="ml-3">...</span>
</div>
) : effectiveReviewData ? (
) : reviewData ? (
<main className="flex-1 min-h-0 grid grid-cols-[22%,1fr,30%] p-2">
{/* 左栏:规则目录 */}
<RulesDirectory
reviewPoints={effectiveReviewData.reviewPoints}
statistics={effectiveReviewData.statistics}
reviewPoints={reviewData.reviewPoints}
statistics={reviewData.statistics}
activeReviewPointResultId={activeReviewPointResultId}
fileName={effectiveReviewData.fileInfo.fileName}
fileName={reviewData.fileInfo.fileName}
onRuleSelect={handleRuleSelect}
onBack={() => navigate(getReturnUrl())}
/>
@@ -886,18 +1014,19 @@ export default function ReviewDetails() {
targetPage={targetPage}
charPositions={charPositions}
activeReviewPointResultId={activeReviewPointResultId}
reviewPoints={effectiveReviewData.reviewPoints}
reviewPoints={reviewData.reviewPoints}
highlightValue={highlightValue}
aiSuggestionReplace={aiSuggestionReplace}
userInfo={(normalizedLoaderData as any)?.userInfo}
userInfo={currentUserInfo || undefined}
/>
) : (
<PdfPreviewTest
filePath={previewPath}
targetPage={targetPage}
charPositions={charPositions}
bboxHighlight={bboxHighlight}
activeReviewPointResultId={activeReviewPointResultId}
reviewPoints={effectiveReviewData.reviewPoints}
reviewPoints={reviewData.reviewPoints}
/>
)}
</section>
@@ -907,15 +1036,16 @@ export default function ReviewDetails() {
activeTab={rightActiveTab}
onTabChange={handleTabChange}
activeReviewPoint={activeReviewPoint}
reviewPoints={effectiveReviewData.reviewPoints}
fileInfo={effectiveReviewData.fileInfo}
reviewInfo={effectiveReviewData.reviewInfo}
reviewPoints={reviewData.reviewPoints}
detailMode={detailMode}
fileInfo={reviewData.fileInfo}
reviewInfo={reviewData.reviewInfo}
onReviewPointSelect={handleReviewPointSelect}
onStatusChange={handleReviewPointStatusChange}
onConfirmResults={handleConfirmResults}
onDownload={handleDownloadFile}
auditStatus={document?.auditStatus}
fileFormat={effectiveReviewData.fileInfo.fileFormat}
fileFormat={reviewData.fileInfo.fileFormat}
onUploadTemplate={handleOpenReuploadModal}
onComparison={() => setShowComparison(true)}
showComparisonButton={showComparisonButton}
@@ -939,7 +1069,7 @@ export default function ReviewDetails() {
</div>
<div className="flex-1 min-h-0">
<ComparePreview
doc1Path={document?.path || ''}
doc1Path={previewPath}
doc2Path={comparison_document?.template_contract_path || ''}
/>
</div>