From a0483e46b8ba5f31827334fd6eb40dc6b81241ca Mon Sep 17 00:00:00 2001 From: John Date: Sun, 5 Jan 2025 13:59:05 -0300 Subject: [PATCH] optimize pdf rendering --- app/react/V2/Components/PDFViewer/PDFPage.tsx | 154 ++++++++---------- 1 file changed, 72 insertions(+), 82 deletions(-) diff --git a/app/react/V2/Components/PDFViewer/PDFPage.tsx b/app/react/V2/Components/PDFViewer/PDFPage.tsx index d8a124e700..d7f604934c 100644 --- a/app/react/V2/Components/PDFViewer/PDFPage.tsx +++ b/app/react/V2/Components/PDFViewer/PDFPage.tsx @@ -1,9 +1,8 @@ /* eslint-disable max-statements */ import React, { useEffect, useRef, useState } from 'react'; -import { PDFDocumentProxy, PDFPageProxy } from 'pdfjs-dist'; -import { PDFPageView } from 'pdfjs-dist/web/pdf_viewer.mjs'; +import { PDFDocumentProxy } from 'pdfjs-dist'; import { Highlight } from '@huridocs/react-text-selection-handler'; -import { useAtom } from 'jotai'; +import { useAtom, useSetAtom } from 'jotai'; import { pdfScaleAtom } from 'V2/atoms'; import { EventBus, PDFJSViewer, PDFJS } from './pdfjs'; import { TextHighlight } from './types'; @@ -19,105 +18,96 @@ interface PDFPageProps { } const PDFPage = ({ pdf, page, eventBus, containerWidth, highlights }: PDFPageProps) => { - const pageContainerRef = useRef(null); - const [isVisible, setIsVisible] = useState(false); - const [pdfPage, setPdfPage] = useState(); const [error, setError] = useState(); const [pdfScale, setPdfScale] = useAtom(pdfScaleAtom); - const pageViewer = useRef(); - - useEffect(() => { - pdf - .getPage(page) - .then(result => setPdfPage(result)) - .catch((e: Error) => setError(e.message)); - }, [page, pdf]); - - useEffect(() => { - if (pageContainerRef.current && pdfPage) { - const currentContainer = pageContainerRef.current; - const originalViewport = pdfPage.getViewport({ scale: 1 }); - const scale = calculateScaling( - originalViewport.width * PDFJS.PixelsPerInch.PDF_TO_CSS_UNITS, - containerWidth - ); - const defaultViewport = pdfPage.getViewport({ scale }); - - if (scale !== pdfScale) { - setPdfScale(scale); - } - - if (!pageViewer.current) { - pageViewer.current = new PDFJSViewer.PDFPageView({ - container: currentContainer, - id: page, - scale, - defaultViewport, - annotationMode: 0, - eventBus, - }); - - pageViewer.current.setPdfPage(pdfPage); - } - } - }, [containerWidth, eventBus, page, pdfPage]); + const pageContainerRef = useRef(null); useEffect(() => { const currentContainer = pageContainerRef.current; + let observer: IntersectionObserver; - const handleIntersection: IntersectionObserverCallback = entries => { - const [entry] = entries; - setIsVisible(entry.isIntersecting); - }; + pdf + .getPage(page) + .then(pdfPage => { + if (currentContainer && pdfPage) { + const originalViewport = pdfPage.getViewport({ scale: 1 }); + const scale = calculateScaling( + originalViewport.width * PDFJS.PixelsPerInch.PDF_TO_CSS_UNITS, + containerWidth + ); + const defaultViewport = pdfPage.getViewport({ scale }); + + if (scale !== pdfScale) { + setPdfScale(scale); + } + + const pageViewer = new PDFJSViewer.PDFPageView({ + container: currentContainer, + id: page, + scale, + defaultViewport, + annotationMode: 0, + eventBus, + }); + + pageViewer.setPdfPage(pdfPage); + + const handleIntersection: IntersectionObserverCallback = entries => { + const [entry] = entries; + if (entry.isIntersecting) { + if (pageViewer.renderingState === PDFJSViewer.RenderingStates.INITIAL) { + pageViewer.draw().catch(e => { + setError(e.message); + }); + } + } else { + if (pageViewer.renderingState === PDFJSViewer.RenderingStates.INITIAL) { + pageViewer.cancelRendering(); + } + if (pageViewer.renderingState === PDFJSViewer.RenderingStates.FINISHED) { + pageViewer.destroy(); + } + } + }; - const observer = new IntersectionObserver(handleIntersection, { - root: null, - threshold: 0.1, - }); + observer = new IntersectionObserver(handleIntersection, { + root: null, + threshold: 0.1, + }); - if (currentContainer) { - observer.observe(currentContainer); - } + observer.observe(currentContainer); + } + }) + .catch((e: Error) => { + setError(e.message); + }); return () => { if (currentContainer) observer.unobserve(currentContainer); - return undefined; }; + //pdf rendering is expensive and we want to make sure there's a single effect that runs only on mount + // eslint-disable-next-line react-hooks/exhaustive-deps }, []); - useEffect(() => { - if (pageViewer.current && pdfPage) { - if (isVisible) { - if (pageViewer.current.renderingState === PDFJSViewer.RenderingStates.INITIAL) { - pageViewer.current.draw().catch(e => setError(e.message)); - } - } - if (!isVisible) { - pageViewer.current.destroy(); - } - } - }, [isVisible, pdfPage]); - if (error) { return
{error}
; } return (
- {isVisible && - highlights?.map(highlight => { - const scaledHightlight = { - ...highlight, - textSelection: adjustSelectionsToScale(highlight.textSelection, pdfScale), - }; - return ( - - ); - })} + {highlights?.map(highlight => { + const scaledHightlight = { + ...highlight, + textSelection: adjustSelectionsToScale(highlight.textSelection, pdfScale), + }; + return ( + + ); + })}
); };