mirror of
https://github.com/MODSetter/SurfSense.git
synced 2026-04-25 00:36:31 +02:00
feat: optimize PDF viewer performance with improved page rendering and visibility management
This commit is contained in:
parent
706d5b9821
commit
abf758997a
1 changed files with 148 additions and 53 deletions
|
|
@ -18,24 +18,73 @@ interface PdfViewerProps {
|
|||
isPublic?: boolean;
|
||||
}
|
||||
|
||||
interface PageDimensions {
|
||||
width: number;
|
||||
height: number;
|
||||
}
|
||||
|
||||
const ZOOM_STEP = 0.15;
|
||||
const MIN_ZOOM = 0.5;
|
||||
const MAX_ZOOM = 3;
|
||||
const PAGE_GAP = 12;
|
||||
const SCROLL_DEBOUNCE_MS = 30;
|
||||
const BUFFER_PAGES = 1;
|
||||
|
||||
export function PdfViewer({ pdfUrl, isPublic = false }: PdfViewerProps) {
|
||||
const [numPages, setNumPages] = useState(0);
|
||||
const [scale, setScale] = useState(1);
|
||||
const [loading, setLoading] = useState(true);
|
||||
const [loadError, setLoadError] = useState<string | null>(null);
|
||||
const [currentPage, setCurrentPage] = useState(1);
|
||||
|
||||
const scrollContainerRef = useRef<HTMLDivElement>(null);
|
||||
const pagesContainerRef = useRef<HTMLDivElement>(null);
|
||||
const pdfDocRef = useRef<PDFDocumentProxy | null>(null);
|
||||
const canvasRefs = useRef<Map<number, HTMLCanvasElement>>(new Map());
|
||||
const renderTasksRef = useRef<Map<number, RenderTask>>(new Map());
|
||||
const renderedScalesRef = useRef<Map<number, number>>(new Map());
|
||||
const pageDimsRef = useRef<PageDimensions[]>([]);
|
||||
const visiblePagesRef = useRef<Set<number>>(new Set());
|
||||
const scrollTimerRef = useRef<ReturnType<typeof setTimeout> | null>(null);
|
||||
|
||||
const getScaledHeight = useCallback(
|
||||
(pageIndex: number) => {
|
||||
const dims = pageDimsRef.current[pageIndex];
|
||||
return dims ? Math.floor(dims.height * scale) : 0;
|
||||
},
|
||||
[scale],
|
||||
);
|
||||
|
||||
const getVisibleRange = useCallback(() => {
|
||||
const container = scrollContainerRef.current;
|
||||
if (!container || pageDimsRef.current.length === 0) return { first: 1, last: 1 };
|
||||
|
||||
const scrollTop = container.scrollTop;
|
||||
const viewportHeight = container.clientHeight;
|
||||
const scrollBottom = scrollTop + viewportHeight;
|
||||
|
||||
let cumTop = 16;
|
||||
let first = 1;
|
||||
let last = pageDimsRef.current.length;
|
||||
|
||||
for (let i = 0; i < pageDimsRef.current.length; i++) {
|
||||
const pageHeight = getScaledHeight(i);
|
||||
const pageBottom = cumTop + pageHeight;
|
||||
|
||||
if (pageBottom >= scrollTop && first === 1) {
|
||||
first = i + 1;
|
||||
}
|
||||
if (cumTop > scrollBottom) {
|
||||
last = i;
|
||||
break;
|
||||
}
|
||||
|
||||
cumTop = pageBottom + PAGE_GAP;
|
||||
}
|
||||
|
||||
first = Math.max(1, first - BUFFER_PAGES);
|
||||
last = Math.min(pageDimsRef.current.length, last + BUFFER_PAGES);
|
||||
|
||||
return { first, last };
|
||||
}, [getScaledHeight]);
|
||||
|
||||
const renderPage = useCallback(async (pageNum: number, currentScale: number) => {
|
||||
const pdf = pdfDocRef.current;
|
||||
|
|
@ -71,12 +120,51 @@ export function PdfViewer({ pdfUrl, isPublic = false }: PdfViewerProps) {
|
|||
await renderTask.promise;
|
||||
renderTasksRef.current.delete(pageNum);
|
||||
renderedScalesRef.current.set(pageNum, currentScale);
|
||||
page.cleanup();
|
||||
} catch (err: unknown) {
|
||||
if (err instanceof Error && err.message?.includes("cancelled")) return;
|
||||
console.error(`Failed to render page ${pageNum}:`, err);
|
||||
}
|
||||
}, []);
|
||||
|
||||
const cleanupPage = useCallback((pageNum: number) => {
|
||||
const existing = renderTasksRef.current.get(pageNum);
|
||||
if (existing) {
|
||||
existing.cancel();
|
||||
renderTasksRef.current.delete(pageNum);
|
||||
}
|
||||
|
||||
const canvas = canvasRefs.current.get(pageNum);
|
||||
if (canvas) {
|
||||
const ctx = canvas.getContext("2d");
|
||||
if (ctx) ctx.clearRect(0, 0, canvas.width, canvas.height);
|
||||
canvas.width = 0;
|
||||
canvas.height = 0;
|
||||
}
|
||||
|
||||
renderedScalesRef.current.delete(pageNum);
|
||||
}, []);
|
||||
|
||||
const renderVisiblePages = useCallback(() => {
|
||||
if (!pdfDocRef.current || pageDimsRef.current.length === 0) return;
|
||||
|
||||
const { first, last } = getVisibleRange();
|
||||
const newVisible = new Set<number>();
|
||||
|
||||
for (let i = first; i <= last; i++) {
|
||||
newVisible.add(i);
|
||||
renderPage(i, scale);
|
||||
}
|
||||
|
||||
for (const pageNum of visiblePagesRef.current) {
|
||||
if (!newVisible.has(pageNum)) {
|
||||
cleanupPage(pageNum);
|
||||
}
|
||||
}
|
||||
|
||||
visiblePagesRef.current = newVisible;
|
||||
}, [getVisibleRange, renderPage, cleanupPage, scale]);
|
||||
|
||||
useEffect(() => {
|
||||
let cancelled = false;
|
||||
|
||||
|
|
@ -84,7 +172,7 @@ export function PdfViewer({ pdfUrl, isPublic = false }: PdfViewerProps) {
|
|||
setLoading(true);
|
||||
setLoadError(null);
|
||||
setNumPages(0);
|
||||
setCurrentPage(1);
|
||||
pageDimsRef.current = [];
|
||||
|
||||
try {
|
||||
const loadingTask = pdfjsLib.getDocument({
|
||||
|
|
@ -98,7 +186,21 @@ export function PdfViewer({ pdfUrl, isPublic = false }: PdfViewerProps) {
|
|||
return;
|
||||
}
|
||||
|
||||
const dims: PageDimensions[] = [];
|
||||
for (let i = 1; i <= pdf.numPages; i++) {
|
||||
const page = await pdf.getPage(i);
|
||||
const viewport = page.getViewport({ scale: 1 });
|
||||
dims.push({ width: viewport.width, height: viewport.height });
|
||||
page.cleanup();
|
||||
}
|
||||
|
||||
if (cancelled) {
|
||||
pdf.destroy();
|
||||
return;
|
||||
}
|
||||
|
||||
pdfDocRef.current = pdf;
|
||||
pageDimsRef.current = dims;
|
||||
setNumPages(pdf.numPages);
|
||||
setLoading(false);
|
||||
} catch (err: unknown) {
|
||||
|
|
@ -118,52 +220,42 @@ export function PdfViewer({ pdfUrl, isPublic = false }: PdfViewerProps) {
|
|||
}
|
||||
renderTasksRef.current.clear();
|
||||
renderedScalesRef.current.clear();
|
||||
visiblePagesRef.current.clear();
|
||||
pdfDocRef.current?.destroy();
|
||||
pdfDocRef.current = null;
|
||||
};
|
||||
}, [pdfUrl]);
|
||||
|
||||
useEffect(() => {
|
||||
if (!pdfDocRef.current || numPages === 0) return;
|
||||
if (numPages === 0) return;
|
||||
|
||||
renderedScalesRef.current.clear();
|
||||
visiblePagesRef.current.clear();
|
||||
|
||||
for (let i = 1; i <= numPages; i++) {
|
||||
renderPage(i, scale);
|
||||
}
|
||||
}, [scale, numPages, renderPage]);
|
||||
const frame = requestAnimationFrame(() => {
|
||||
renderVisiblePages();
|
||||
});
|
||||
|
||||
return () => cancelAnimationFrame(frame);
|
||||
}, [numPages, renderVisiblePages]);
|
||||
|
||||
useEffect(() => {
|
||||
const container = scrollContainerRef.current;
|
||||
if (!container || numPages <= 1) return;
|
||||
if (!container || numPages === 0) return;
|
||||
|
||||
const handleScroll = () => {
|
||||
const canvases = canvasRefs.current;
|
||||
const containerTop = container.scrollTop;
|
||||
const containerMid = containerTop + container.clientHeight / 2;
|
||||
|
||||
let closest = 1;
|
||||
let closestDist = Number.POSITIVE_INFINITY;
|
||||
|
||||
for (let i = 1; i <= numPages; i++) {
|
||||
const canvas = canvases.get(i);
|
||||
if (!canvas) continue;
|
||||
const rect = canvas.getBoundingClientRect();
|
||||
const containerRect = container.getBoundingClientRect();
|
||||
const canvasMid = rect.top - containerRect.top + containerTop + rect.height / 2;
|
||||
const dist = Math.abs(canvasMid - containerMid);
|
||||
if (dist < closestDist) {
|
||||
closestDist = dist;
|
||||
closest = i;
|
||||
}
|
||||
}
|
||||
|
||||
setCurrentPage(closest);
|
||||
if (scrollTimerRef.current) clearTimeout(scrollTimerRef.current);
|
||||
scrollTimerRef.current = setTimeout(() => {
|
||||
renderVisiblePages();
|
||||
}, SCROLL_DEBOUNCE_MS);
|
||||
};
|
||||
|
||||
container.addEventListener("scroll", handleScroll, { passive: true });
|
||||
return () => container.removeEventListener("scroll", handleScroll);
|
||||
}, [numPages]);
|
||||
return () => {
|
||||
container.removeEventListener("scroll", handleScroll);
|
||||
if (scrollTimerRef.current) clearTimeout(scrollTimerRef.current);
|
||||
};
|
||||
}, [numPages, renderVisiblePages]);
|
||||
|
||||
const setCanvasRef = useCallback((pageNum: number, el: HTMLCanvasElement | null) => {
|
||||
if (el) {
|
||||
|
|
@ -184,7 +276,7 @@ export function PdfViewer({ pdfUrl, isPublic = false }: PdfViewerProps) {
|
|||
if (loadError) {
|
||||
return (
|
||||
<div className="flex flex-col items-center justify-center h-full gap-3 p-6 text-center">
|
||||
<p className="font-medium text-foreground">Failed to load resume preview</p>
|
||||
<p className="font-medium text-foreground">Failed to load PDF</p>
|
||||
<p className="text-sm text-muted-foreground">{loadError}</p>
|
||||
</div>
|
||||
);
|
||||
|
|
@ -194,14 +286,6 @@ export function PdfViewer({ pdfUrl, isPublic = false }: PdfViewerProps) {
|
|||
<div className="flex flex-col h-full">
|
||||
{numPages > 0 && (
|
||||
<div className={`flex items-center justify-center gap-2 px-4 py-2 border-b shrink-0 ${isPublic ? "bg-main-panel" : "bg-sidebar"}`}>
|
||||
{numPages > 1 && (
|
||||
<>
|
||||
<span className="text-xs text-muted-foreground tabular-nums min-w-[60px] text-center">
|
||||
{currentPage} / {numPages}
|
||||
</span>
|
||||
<div className="w-px h-4 bg-border mx-1" />
|
||||
</>
|
||||
)}
|
||||
<Button variant="ghost" size="icon" onClick={zoomOut} disabled={scale <= MIN_ZOOM} className="size-7">
|
||||
<ZoomOutIcon className="size-4" />
|
||||
</Button>
|
||||
|
|
@ -223,18 +307,29 @@ export function PdfViewer({ pdfUrl, isPublic = false }: PdfViewerProps) {
|
|||
<Spinner size="md" />
|
||||
</div>
|
||||
) : (
|
||||
<div
|
||||
ref={pagesContainerRef}
|
||||
className="flex flex-col items-center py-4"
|
||||
style={{ gap: `${PAGE_GAP}px` }}
|
||||
>
|
||||
{Array.from({ length: numPages }, (_, i) => i + 1).map((pageNum) => (
|
||||
<canvas
|
||||
key={pageNum}
|
||||
ref={(el) => setCanvasRef(pageNum, el)}
|
||||
className="shadow-lg"
|
||||
/>
|
||||
))}
|
||||
<div className="flex flex-col items-center py-4" style={{ gap: `${PAGE_GAP}px` }}>
|
||||
{pageDimsRef.current.map((dims, i) => {
|
||||
const pageNum = i + 1;
|
||||
const scaledWidth = Math.floor(dims.width * scale);
|
||||
const scaledHeight = Math.floor(dims.height * scale);
|
||||
return (
|
||||
<div
|
||||
key={pageNum}
|
||||
className="relative shrink-0"
|
||||
style={{ width: scaledWidth, height: scaledHeight }}
|
||||
>
|
||||
<canvas
|
||||
ref={(el) => setCanvasRef(pageNum, el)}
|
||||
className="shadow-lg absolute inset-0"
|
||||
/>
|
||||
{numPages > 1 && (
|
||||
<span className="absolute bottom-2 right-3 text-[10px] tabular-nums text-white/80 bg-black/50 px-1.5 py-0.5 rounded pointer-events-none">
|
||||
Page {pageNum}/{numPages}
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue