UNPKG

@anaralabs/lector

Version:

Headless PDF viewer for React

1,542 lines (1,527 loc) 161 kB
'use client'; import React, { createContext, forwardRef, useRef, useState, useCallback, useContext, useEffect, cloneElement, createRef, useLayoutEffect } from 'react'; import { createPortal } from 'react-dom'; import { useFloating, autoUpdate, offset, shift, useDismiss, useInteractions, inline, flip } from '@floating-ui/react'; import { useVirtualizer, elementScroll, debounce } from '@tanstack/react-virtual'; import { create, useStore, createStore } from 'zustand'; import { jsx, Fragment, jsxs } from 'react/jsx-runtime'; import { AnnotationLayer, TextLayer, getDocument } from 'pdfjs-dist'; import { useDebounce } from 'use-debounce'; import { v4 } from 'uuid'; import { Slot } from '@radix-ui/react-slot'; // src/components/annotation-tooltip.tsx // src/lib/clamp.ts var clamp = (value, min, max) => { return Math.min(Math.max(value, min), max); }; // src/lib/zoom.ts var getFitWidthZoom = (containerWidth, viewports, zoomOptions) => { const { minZoom, maxZoom } = zoomOptions; const maxPageWidth = Math.max(...viewports.map((viewport) => viewport.width)); const targetZoom = containerWidth / maxPageWidth; const clampedZoom = Math.min(Math.max(targetZoom, minZoom), maxZoom); return clampedZoom; }; var createZustandContext = (getStore) => { const Context = React.createContext(null); const Provider = (props) => { const [store] = React.useState(() => getStore(props.initialValue)); return /* @__PURE__ */ jsx(Context.Provider, { value: store, children: props.children }); }; return { useContext: () => React.useContext(Context), Context, Provider }; }; // src/internal.ts var PDFStore = createZustandContext( (initialState) => { return createStore((set, get) => ({ pdfDocumentProxy: initialState.pdfDocumentProxy, zoom: initialState.zoom, isZoomFitWidth: initialState.isZoomFitWidth ?? false, zoomOptions: { minZoom: initialState.zoomOptions?.minZoom ?? 0.5, maxZoom: initialState.zoomOptions?.maxZoom ?? 10 }, viewportRef: createRef(), viewports: initialState.viewports, updateZoom: (zoom, isZoomFitWidth = false) => { const { minZoom, maxZoom } = get().zoomOptions; set((state) => { if (typeof zoom === "function") { const newZoom2 = clamp(zoom(state.zoom), minZoom, maxZoom); return { zoom: newZoom2, isZoomFitWidth }; } const newZoom = clamp(zoom, minZoom, maxZoom); return { zoom: newZoom, isZoomFitWidth }; }); }, zoomFitWidth: () => { const { viewportRef, zoomOptions, viewports } = get(); if (!viewportRef.current) return; const clampedZoom = getFitWidthZoom( viewportRef.current.clientWidth, viewports, zoomOptions ); set({ zoom: clampedZoom, isZoomFitWidth: true }); return clampedZoom; }, currentPage: 1, setCurrentPage: (val) => { set({ currentPage: val }); }, isPinching: false, setIsPinching: (val) => { set({ isPinching: val }); }, virtualizer: null, setVirtualizer: (val) => { set({ virtualizer: val }); }, pageProxies: initialState.pageProxies, getPdfPageProxy: (pageNumber) => { const proxy = get().pageProxies[pageNumber - 1]; if (!proxy) throw new Error(`Page ${pageNumber} does not exist`); return proxy; }, textContent: [], setTextContent: (val) => { set({ textContent: val }); }, highlights: [], setHighlight: (val) => { set({ highlights: val }); }, customSelectionRects: [], setCustomSelectionRects: (val) => { set({ customSelectionRects: val }); }, coloredHighlights: [], addColoredHighlight: (value) => set((prevState) => ({ coloredHighlights: [...prevState.coloredHighlights, value] })), deleteColoredHighlight: (uuid) => set((prevState) => ({ coloredHighlights: prevState.coloredHighlights.filter( (rect) => rect.uuid !== uuid ) })) })); } ); var usePdf = (selector) => useStore(PDFStore.useContext(), selector); // src/hooks/useAnnotationTooltip.ts var useAnnotationTooltip = ({ annotation, onOpenChange, position = "top", isOpen: controlledIsOpen }) => { const isNewAnnotation = Date.now() - new Date(annotation.createdAt).getTime() < 1e3; const [isPositionCalculated, setIsPositionCalculated] = useState(false); const [isOpen, setIsOpen] = useState(false); const viewportRef = usePdf((state) => state.viewportRef); const scale = usePdf((state) => state.zoom); const effectiveIsOpen = (isOpen && isPositionCalculated || controlledIsOpen) ?? false; const { refs, floatingStyles, context } = useFloating({ placement: position, open: effectiveIsOpen, onOpenChange: (open) => { setIsOpen(open); onOpenChange?.(open); }, whileElementsMounted: autoUpdate, middleware: [ inline(), offset(10), flip({ crossAxis: false, fallbackAxisSideDirection: "end" }), shift({ padding: 8 }) ] }); const dismiss = useDismiss(context); const { getReferenceProps, getFloatingProps } = useInteractions([dismiss]); const updateTooltipPosition = useCallback(() => { if (!annotation.highlights.length) { setIsPositionCalculated(false); return; } const highlightRects = annotation.highlights; let minLeft = Infinity; let maxRight = -Infinity; let minTop = Infinity; let maxBottom = -Infinity; const viewportElement = viewportRef.current; if (!viewportElement) { setIsPositionCalculated(false); return; } const pageElement = viewportElement.querySelector(`[data-page-number="${annotation.pageNumber}"]`); if (!pageElement) { setIsPositionCalculated(false); return; } refs.setReference({ getBoundingClientRect() { const pageRect = pageElement.getBoundingClientRect(); highlightRects.forEach((highlight) => { const scaledLeft = highlight.left * scale; const scaledWidth = highlight.width * scale; const scaledTop = highlight.top * scale; const scaledHeight = highlight.height * scale; const left = pageRect.left + scaledLeft; const right = left + scaledWidth; const top = pageRect.top + scaledTop; const bottom = top + scaledHeight; minLeft = Math.min(minLeft, left); maxRight = Math.max(maxRight, right); minTop = Math.min(minTop, top); maxBottom = Math.max(maxBottom, bottom); }); const width = maxRight - minLeft; const height = maxBottom - minTop; const centerX = minLeft + width / 2; const centerY = minTop + height / 2; const rect = { width, height, x: centerX - width / 2, y: centerY - height / 2, top: centerY - height / 2, right: centerX + width / 2, bottom: centerY + height / 2, left: centerX - width / 2 }; return rect; }, getClientRects() { return [this.getBoundingClientRect()]; }, contextElement: viewportRef.current || void 0 }); setIsPositionCalculated(true); if (isNewAnnotation) { setIsOpen(true); } }, [annotation.highlights, annotation.pageNumber, refs, viewportRef, scale, isNewAnnotation]); useEffect(() => { const viewport = viewportRef.current; setIsPositionCalculated(false); requestAnimationFrame(() => { updateTooltipPosition(); }); const handleScroll = () => { requestAnimationFrame(updateTooltipPosition); }; const handleResize = () => { requestAnimationFrame(updateTooltipPosition); }; if (viewport) { viewport.addEventListener("scroll", handleScroll, { passive: true }); } window.addEventListener("resize", handleResize, { passive: true }); return () => { if (viewport) { viewport.removeEventListener("scroll", handleScroll); } window.removeEventListener("resize", handleResize); }; }, [updateTooltipPosition, viewportRef, scale, controlledIsOpen]); return { isOpen: effectiveIsOpen, setIsOpen, refs, floatingStyles, getFloatingProps, getReferenceProps }; }; var AnnotationTooltip = ({ annotation, children, renderTooltipContent, hoverTooltipContent, onOpenChange, className, focusedOpenId, focusedHoverOpenId, hoverClassName, isOpen: controlledIsOpen, hoverIsOpen: controlledHoverIsOpen }) => { const viewportRef = usePdf((state) => state.viewportRef); const closeTimeoutRef = useRef(null); const isMouseInTooltipRef = useRef(false); const [triggeredPosition, setTriggeredPosition] = useState(); const { isOpen: uncontrolledIsOpen, setIsOpen, refs, floatingStyles, getFloatingProps, getReferenceProps } = useAnnotationTooltip({ annotation, onOpenChange, position: triggeredPosition, isOpen: controlledIsOpen }); const { isOpen: uncontrolledHoverIsOpen, setIsOpen: setHoverIsOpen, refs: hoverRefs, floatingStyles: hoverFloatingStyles, getFloatingProps: getHoverFloatingProps, getReferenceProps: getHoverReferenceProps } = useAnnotationTooltip({ position: "bottom", annotation, isOpen: controlledHoverIsOpen }); const isOpen = controlledIsOpen ?? uncontrolledIsOpen; const hoverIsOpen = controlledHoverIsOpen || uncontrolledHoverIsOpen; const handleClick = useCallback(() => { if (controlledIsOpen === void 0) { setIsOpen(!isOpen); } }, [controlledIsOpen, isOpen, setIsOpen]); const handleMouseEnter = useCallback(() => { if (focusedOpenId && focusedOpenId !== annotation.id) return; if (focusedHoverOpenId && focusedHoverOpenId !== annotation.id) return; if (hoverTooltipContent) { if (closeTimeoutRef.current) { clearTimeout(closeTimeoutRef.current); closeTimeoutRef.current = null; } setHoverIsOpen(true); } }, [hoverTooltipContent, setHoverIsOpen]); const closeTooltip = useCallback(() => { if (!isMouseInTooltipRef.current) { setHoverIsOpen(false); } }, [setHoverIsOpen]); const handleMouseLeave = useCallback(() => { if (!hoverTooltipContent) return; closeTimeoutRef.current = setTimeout(closeTooltip, 100); }, [hoverTooltipContent, closeTooltip]); const handleTooltipMouseEnter = useCallback(() => { if (focusedOpenId && focusedOpenId !== annotation.id) return; if (focusedHoverOpenId && focusedHoverOpenId !== annotation.id) return; isMouseInTooltipRef.current = true; if (closeTimeoutRef.current) { clearTimeout(closeTimeoutRef.current); closeTimeoutRef.current = null; } }, []); const handleTooltipMouseLeave = useCallback(() => { isMouseInTooltipRef.current = false; setHoverIsOpen(false); }, [setHoverIsOpen]); return /* @__PURE__ */ jsxs(Fragment, { children: [ /* @__PURE__ */ jsx( "div", { ref: (node) => { refs.setReference(node); hoverRefs.setReference(node); }, onClick: handleClick, onMouseEnter: handleMouseEnter, onMouseLeave: handleMouseLeave, ...getReferenceProps(), ...getHoverReferenceProps(), children } ), isOpen && viewportRef.current && createPortal( /* @__PURE__ */ jsx( "div", { ref: refs.setFloating, className, "data-annotation-tooltip": "click", style: { ...floatingStyles, position: "absolute", pointerEvents: "auto", zIndex: 50 }, ...getFloatingProps(), children: renderTooltipContent({ annotation, onClose: () => setIsOpen(false), setPosition: (position) => setTriggeredPosition(position) }) } ), viewportRef.current ), !isOpen && hoverIsOpen && annotation.comment && hoverTooltipContent && viewportRef.current && createPortal( /* @__PURE__ */ jsx( "div", { ref: hoverRefs.setFloating, className: hoverClassName, "data-annotation-tooltip": "hover", style: { ...hoverFloatingStyles, position: "absolute", pointerEvents: "auto", zIndex: 51 }, onMouseEnter: handleTooltipMouseEnter, onMouseLeave: handleTooltipMouseLeave, ...getHoverFloatingProps(), children: hoverTooltipContent } ), viewportRef.current ) ] }); }; var useAnnotations = create((set) => ({ annotations: [], addAnnotation: (annotation) => set((state) => ({ annotations: [ ...state.annotations, annotation ] })), updateAnnotation: (id, updates) => set((state) => ({ annotations: state.annotations.map( (annotation) => annotation.id === id ? { ...annotation, ...updates } : annotation ) })), deleteAnnotation: (id) => set((state) => ({ annotations: state.annotations.filter((annotation) => annotation.id !== id) })), setAnnotations: (annotations) => set({ annotations }) })); var PDFPageNumberContext = createContext(0); var usePDFPageNumber = () => { return useContext(PDFPageNumberContext); }; var AnnotationHighlightLayer = ({ className, style, renderTooltipContent, renderHoverTooltipContent, tooltipClassName, highlightClassName, underlineClassName, commentIconPosition, commmentIcon, commentIconClassName, focusedAnnotationId, focusedHoverAnnotationId, onAnnotationClick, onAnnotationTooltipClose, hoverTooltipClassName }) => { const { annotations } = useAnnotations(); const pageNumber = usePDFPageNumber(); const pageAnnotations = annotations.filter( (annotation) => annotation.pageNumber === pageNumber ); const getCommentIconPosition = (highlights) => { if (!highlights.length) return { top: 0, right: 10 }; const sortedHighlights = [...highlights].sort((a, b) => { const topDiff = a.top - b.top; return Math.abs(topDiff) < 3 ? a.left - b.left : topDiff; }); const lines = []; let currentLine = []; sortedHighlights.forEach((highlight) => { if (currentLine.length === 0) { currentLine.push(highlight); } else { const firstInLine = currentLine[0]; if (Math.abs(highlight.top - firstInLine.top) <= 3) { currentLine.push(highlight); } else { lines.push([...currentLine]); currentLine = [highlight]; } } }); if (currentLine.length > 0) { lines.push(currentLine); } const PAGE_WIDTH = 600; const hasLongLine = lines.some((line) => { if (line.length === 0) return false; const rightmost2 = Math.max(...line.map((h) => h.left + h.width)); return rightmost2 > PAGE_WIDTH * 0.8; }); const firstHighlight = highlights[0]; const firstLine = lines[0] || []; const leftmost = Math.min(...firstLine.map((h) => h.left)); const rightmost = Math.max(...firstLine.map((h) => h.left + h.width)); const lineCenter = leftmost + (rightmost - leftmost) / 2; const shouldPositionRight = hasLongLine || lineCenter > PAGE_WIDTH * 0.5; const rightPosition = commentIconPosition === "highlight" ? { left: rightmost + 8 } : { right: 10 }; const leftPosition = commentIconPosition === "highlight" ? { left: leftmost - 18 } : { left: 20 }; return { top: firstHighlight.top + firstHighlight.height / 2 - 6, ...shouldPositionRight ? rightPosition : leftPosition }; }; return /* @__PURE__ */ jsx("div", { className, style, children: pageAnnotations.map((annotation) => { return /* @__PURE__ */ jsx( AnnotationTooltip, { annotation, className: tooltipClassName, hoverClassName: hoverTooltipClassName, focusedOpenId: focusedAnnotationId, focusedHoverOpenId: focusedHoverAnnotationId, isOpen: focusedAnnotationId === annotation.id, hoverIsOpen: focusedHoverAnnotationId === annotation.id, onOpenChange: (open) => { if (open && onAnnotationClick) { onAnnotationClick(annotation); } else if (!open && onAnnotationTooltipClose) { onAnnotationTooltipClose(annotation); } }, renderTooltipContent, hoverTooltipContent: renderHoverTooltipContent({ annotation, onClose: () => { } }), children: /* @__PURE__ */ jsxs( "div", { style: { cursor: "pointer" }, onClick: () => onAnnotationClick?.(annotation), children: [ annotation.highlights.map((highlight, index) => /* @__PURE__ */ jsx( "div", { className: highlightClassName, style: { position: "absolute", top: highlight.top, left: highlight.left, width: highlight.width, height: highlight.height, backgroundColor: annotation.color }, "data-highlight-id": annotation.id }, `highlight-${index}` )), annotation.comment && annotation.underlines?.map((rect, index) => /* @__PURE__ */ jsx( "div", { className: underlineClassName, style: { position: "absolute", top: rect.top, left: rect.left, width: rect.width, height: 1.1, backgroundColor: annotation.borderColor }, "data-comment-id": annotation.id }, `underline-${index}` )), annotation.comment && commmentIcon && /* @__PURE__ */ jsx( "div", { className: commentIconClassName, style: { position: "absolute", ...getCommentIconPosition(annotation.highlights), color: "gray", cursor: "pointer", zIndex: 10 }, "data-comment-icon-id": annotation.id, children: commmentIcon } ) ] } ) }, annotation.id ); }) }); }; // ../../node_modules/.pnpm/clsx@2.1.1/node_modules/clsx/dist/clsx.mjs function r(e) { var t, f, n = ""; if ("string" == typeof e || "number" == typeof e) n += e; else if ("object" == typeof e) if (Array.isArray(e)) { var o = e.length; for (t = 0; t < o; t++) e[t] && (f = r(e[t])) && (n && (n += " "), n += f); } else for (f in e) e[f] && (n && (n += " "), n += f); return n; } function clsx() { for (var e, t, f = 0, n = "", o = arguments.length; f < o; f++) (e = arguments[f]) && (t = r(e)) && (n && (n += " "), n += t); return n; } var clsx_default = clsx; // src/lib/cancellable.ts var cancellable = (promise) => { let isCancelled = false; const wrappedPromise = new Promise((resolve, reject) => { promise.then( (value) => { if (!isCancelled) { resolve(value); } }, (error) => { if (!isCancelled) { reject(error); } } ); }); return { promise: wrappedPromise, cancel() { isCancelled = true; } }; }; // src/hooks/pages/usePdfJump.tsx var usePdfJump = () => { const virtualizer = usePdf((state) => state.virtualizer); const setHighlight = usePdf((state) => state.setHighlight); const jumpToPage = (pageIndex, options) => { if (!virtualizer) return; const defaultOptions = { align: "start", behavior: "smooth" }; const finalOptions = { ...defaultOptions, ...options }; virtualizer.scrollToIndex(pageIndex - 1, finalOptions); }; const jumpToOffset = (offset3) => { if (!virtualizer) return; virtualizer.scrollToOffset(offset3, { align: "start", behavior: "smooth" }); }; const scrollToHighlightRects = (rects, type, align = "start", additionalOffset = 0) => { if (!virtualizer) return; const firstPage = Math.min(...rects.map((rect) => rect.pageNumber)); const pageOffset = virtualizer.getOffsetForIndex(firstPage - 1, "start"); if (pageOffset === null) return; const targetRect = rects.find((rect) => rect.pageNumber === firstPage); if (!targetRect) return; const isNumber = pageOffset?.[0] != null; if (!isNumber) return; const pageStart = pageOffset[0] ?? 0; let rectTop; let rectHeight; if (type === "percent") { const estimatePageHeight = virtualizer.options.estimateSize( firstPage - 1 ); rectTop = targetRect.top / 100 * estimatePageHeight; rectHeight = targetRect.height / 100 * estimatePageHeight; } else { rectTop = targetRect.top; rectHeight = targetRect.height; } let scrollOffset; if (align === "center") { const viewportHeight = virtualizer.scrollElement?.clientHeight || 0; const rectCenter = pageStart + rectTop + rectHeight / 2; scrollOffset = rectCenter - viewportHeight / 2; } else { scrollOffset = pageStart + rectTop; } scrollOffset += additionalOffset; const adjustedOffset = Math.max(0, scrollOffset); virtualizer.scrollToOffset(adjustedOffset, { align: "start", // Always use start when we've calculated our own centering behavior: "smooth" }); }; const jumpToHighlightRects = (rects, type, align = "start", additionalOffset = 0) => { if (!virtualizer) return; setHighlight(rects); scrollToHighlightRects(rects, type, align, additionalOffset); }; return { jumpToPage, jumpToOffset, jumpToHighlightRects, scrollToHighlightRects }; }; var LinkService = class { _pdfDocumentProxy; externalLinkEnabled = true; isInPresentationMode = false; _currentPageNumber = 0; _pageNavigationCallback; get pdfDocumentProxy() { if (!this._pdfDocumentProxy) { throw new Error("pdfDocumentProxy is not set"); } return this._pdfDocumentProxy; } constructor() { } get pagesCount() { return this._pdfDocumentProxy?.numPages || 0; } get page() { return this._currentPageNumber; } set page(value) { this._currentPageNumber = value; if (this._pageNavigationCallback) { this._pageNavigationCallback(value); } } // Required for link annotations to work setDocument(pdfDocument) { this._pdfDocumentProxy = pdfDocument; } setViewer() { } getDestinationHash(dest) { if (!dest) return ""; const destRef = dest[0]; if (dest.length > 1 && typeof dest[1] === "object" && dest[1] !== null && "url" in dest[1]) { const urlDest = dest[1]; return urlDest.url; } if (destRef && typeof destRef === "object") { if ("num" in destRef) { const numRef = destRef; return `#page=${numRef.num + 1}`; } if ("gen" in destRef) { const genRef = destRef; const refNum = genRef.num ?? 0; return `#page=${refNum + 1}`; } } if (typeof destRef === "number") { return `#page=${destRef + 1}`; } return `#dest-${String(dest)}`; } getAnchorUrl(hash) { if (hash.startsWith("http://") || hash.startsWith("https://")) { return hash; } return `#${hash}`; } addLinkAttributes(link, url, newWindow) { if (!link) return; const isExternalLink = url.startsWith("http://") || url.startsWith("https://"); if (isExternalLink && this.externalLinkEnabled) { link.href = url; link.target = newWindow === false ? "" : "_blank"; link.rel = "noopener noreferrer"; } else if (!isExternalLink) { link.href = url; link.target = ""; } else { link.href = "#"; link.target = ""; } } async goToDestination(dest) { let explicitDest; if (typeof dest === "string") { explicitDest = await this.pdfDocumentProxy.getDestination(dest); } else if (Array.isArray(dest)) { explicitDest = dest; } else { explicitDest = await dest; } if (!explicitDest) { return; } if (explicitDest.length > 1 && typeof explicitDest[1] === "object" && explicitDest[1] !== null && "url" in explicitDest[1]) { return; } const destRef = explicitDest[0]; let pageIndex; if (destRef && typeof destRef === "object") { if ("num" in destRef) { try { const refProxy = destRef; pageIndex = await this.pdfDocumentProxy.getPageIndex(refProxy); } catch (_error) { return; } } else { return; } } else if (typeof destRef === "number") { pageIndex = destRef; } else { return; } const pageNumber = pageIndex + 1; if (this._pageNavigationCallback) { this._pageNavigationCallback(pageNumber); } } executeNamedAction() { } navigateTo(dest) { this.goToDestination(dest); } get rotation() { return 0; } set rotation(value) { } goToPage(pageNumber) { if (pageNumber >= 1 && pageNumber <= this.pagesCount) { if (this._pageNavigationCallback) { this._pageNavigationCallback(pageNumber); } } } setHash(hash) { if (hash.startsWith("#page=")) { const pageNumber = parseInt(hash.substring(6), 10); if (!isNaN(pageNumber)) { this.goToPage(pageNumber); } } } executeSetOCGState() { } // Method to register navigation callback registerPageNavigationCallback(callback) { this._pageNavigationCallback = callback; } // Method to unregister navigation callback unregisterPageNavigationCallback() { this._pageNavigationCallback = void 0; } }; var defaultLinkService = new LinkService(); var PDFLinkServiceContext = createContext(defaultLinkService); var usePDFLinkService = () => { return useContext(PDFLinkServiceContext); }; var useVisibility = ({ elementRef }) => { const [visible, setVisible] = useState(false); useEffect(() => { if (!elementRef.current) { return; } const observer = new IntersectionObserver(([entry]) => { setVisible(entry?.isIntersecting ?? false); }); observer.observe(elementRef.current); return () => { observer.disconnect(); }; }, [elementRef]); return { visible }; }; // src/hooks/layers/useAnnotationLayer.tsx var defaultAnnotationLayerParams = { renderForms: true, externalLinksEnabled: true, jumpOptions: { behavior: "smooth", align: "start" } }; var useAnnotationLayer = (params) => { const mergedParams = { ...defaultAnnotationLayerParams, ...params }; const annotationLayerRef = useRef(null); const annotationLayerObjectRef = useRef(null); const linkService = usePDFLinkService(); const { visible } = useVisibility({ elementRef: annotationLayerRef }); const pageNumber = usePDFPageNumber(); const pdfPageProxy = usePdf((state) => state.getPdfPageProxy(pageNumber)); const pdfDocumentProxy = usePdf((state) => state.pdfDocumentProxy); useEffect(() => { linkService.externalLinkEnabled = mergedParams.externalLinksEnabled; }, [linkService, mergedParams.externalLinksEnabled]); const { jumpToPage } = usePdfJump(); useEffect(() => { if (!jumpToPage) return; const handlePageNavigation = (pageNumber2) => { jumpToPage(pageNumber2, mergedParams.jumpOptions); }; linkService.registerPageNavigationCallback(handlePageNavigation); return () => { linkService.unregisterPageNavigationCallback(); }; }, [jumpToPage, linkService, mergedParams.jumpOptions]); useEffect(() => { const style = document.createElement("style"); style.textContent = ` .annotationLayer { position: absolute; left: 0; top: 0; right: 0; bottom: 0; overflow: hidden; opacity: 1; z-index: 3; } .annotationLayer section { position: absolute; } .annotationLayer .linkAnnotation > a, .annotationLayer .buttonWidgetAnnotation.pushButton > a { position: absolute; font-size: 1em; top: 0; left: 0; width: 100%; height: 100%; background: url("") 0 0 repeat; cursor: pointer; } .annotationLayer .linkAnnotation > a:hover, .annotationLayer .buttonWidgetAnnotation.pushButton > a:hover { opacity: 0.2; background: rgba(255, 255, 0, 1); box-shadow: 0 2px 10px rgba(255, 255, 0, 1); } `; document.head.appendChild(style); return () => { document.head.removeChild(style); }; }, []); useEffect(() => { if (!annotationLayerRef.current) return; const element = annotationLayerRef.current; const handleLinkClick = (e) => { if (!e.target || !(e.target instanceof HTMLAnchorElement)) return; const target = e.target; const href = target.getAttribute("href") || ""; if (href.startsWith("#page=")) { e.preventDefault(); const pageNumber2 = parseInt(href.substring(6), 10); if (!isNaN(pageNumber2)) { linkService.goToPage(pageNumber2); } } }; element.addEventListener("click", handleLinkClick); return () => { element.removeEventListener("click", handleLinkClick); }; }, [linkService]); useEffect(() => { if (!annotationLayerRef.current) { return; } if (visible) { annotationLayerRef.current.style.contentVisibility = "visible"; } else { annotationLayerRef.current.style.contentVisibility = "hidden"; } }, [visible]); useEffect(() => { if (!annotationLayerRef.current || !pdfPageProxy || !pdfDocumentProxy) { return; } if (linkService._pdfDocumentProxy !== pdfDocumentProxy) { linkService.setDocument(pdfDocumentProxy); } annotationLayerRef.current.innerHTML = ""; annotationLayerRef.current.className = "annotationLayer"; const viewport = pdfPageProxy.getViewport({ scale: 1 }); const annotationLayerConfig = { div: annotationLayerRef.current, viewport, page: pdfPageProxy, linkService, accessibilityManager: void 0, annotationCanvasMap: void 0, annotationEditorUIManager: void 0, structTreeLayer: void 0 }; const annotationLayer = new AnnotationLayer(annotationLayerConfig); annotationLayerObjectRef.current = annotationLayer; const { cancel } = cancellable( (async () => { try { const annotations = await pdfPageProxy.getAnnotations(); await annotationLayer.render({ ...annotationLayerConfig, ...mergedParams, annotations, linkService }); } catch (_error) { } })() ); return () => { cancel(); }; }, [pdfPageProxy, pdfDocumentProxy, mergedParams, linkService]); return { annotationLayerRef }; }; var AnnotationLayer2 = ({ renderForms = true, externalLinksEnabled = true, jumpOptions = { behavior: "smooth", align: "start" }, className, style, ...props }) => { const { annotationLayerRef } = useAnnotationLayer({ renderForms, externalLinksEnabled, jumpOptions }); return /* @__PURE__ */ jsx( "div", { className: clsx_default("annotationLayer", className), style: { ...style, position: "absolute", top: 0, left: 0 }, ...props, ref: annotationLayerRef } ); }; var useDpr = () => { const [dpr, setDPR] = useState( !window ? 1 : Math.min(window.devicePixelRatio, 2) ); useEffect(() => { if (!window) { return; } const handleDPRChange = () => { setDPR(window.devicePixelRatio); }; const windowMatch = window.matchMedia( `screen and (min-resolution: ${dpr}dppx)` ); windowMatch.addEventListener("change", handleDPRChange); return () => { windowMatch.removeEventListener("change", handleDPRChange); }; }, []); return dpr; }; // src/hooks/layers/useCanvasLayer.tsx var useCanvasLayer = ({ background }) => { const canvasRef = useRef(null); const pageNumber = usePDFPageNumber(); const dpr = useDpr(); const bouncyZoom = usePdf((state) => state.zoom); const pdfPageProxy = usePdf((state) => state.getPdfPageProxy(pageNumber)); const [zoom] = useDebounce(bouncyZoom, 100); useLayoutEffect(() => { if (!canvasRef.current) { return; } const viewport = pdfPageProxy.getViewport({ scale: 1 }); const canvas = canvasRef.current; const scale = dpr * zoom; canvas.height = viewport.height * scale; canvas.width = viewport.width * scale; canvas.style.height = `${viewport.height}px`; canvas.style.width = `${viewport.width}px`; const canvasContext = canvas.getContext("2d"); canvasContext.scale(scale, scale); const renderingTask = pdfPageProxy.render({ canvasContext, viewport, background }); renderingTask.promise.catch((error) => { if (error.name === "RenderingCancelledException") { return; } throw error; }); return () => { void renderingTask.cancel(); }; }, [pdfPageProxy, dpr, zoom]); return { canvasRef }; }; var CanvasLayer = ({ style, background, ...props }) => { const { canvasRef } = useCanvasLayer({ background }); return /* @__PURE__ */ jsx( "canvas", { style: { ...style, position: "absolute", top: 0, left: 0, width: "100%", height: "100%" }, ...props, ref: canvasRef } ); }; // src/hooks/useSelectionDimensions.tsx var MERGE_THRESHOLD = 2; var shouldMergeRects = (rect1, rect2) => { const verticalOverlap = !(rect1.top > rect2.top + rect2.height || rect2.top > rect1.top + rect1.height); const horizontallyConnected = Math.abs(rect1.left + rect1.width - rect2.left) <= MERGE_THRESHOLD || Math.abs(rect2.left + rect2.width - rect1.left) <= MERGE_THRESHOLD || rect1.left < rect2.left + rect2.width && rect2.left < rect1.left + rect1.width; return verticalOverlap && horizontallyConnected; }; var consolidateHighlightRects = (rects) => { if (rects.length <= 1) return rects; const consolidated = []; const sorted = [...rects].sort((a, b) => { const pageCompare = a.pageNumber - b.pageNumber; if (pageCompare !== 0) return pageCompare; const topDiff = a.top - b.top; return Math.abs(topDiff) < 2 ? a.left - b.left : topDiff; }); let current = sorted[0]; if (!current) return rects; for (let i = 1; i < sorted.length; i++) { const next = sorted[i]; if (!next) continue; const samePageAndLine = current.pageNumber === next.pageNumber && Math.abs(current.top - next.top) < Math.max(current.height, next.height) * 0.5; const horizontallyConnected = samePageAndLine && // Adjacent (touching or very close) (Math.abs(current.left + current.width - next.left) <= MERGE_THRESHOLD || Math.abs(next.left + next.width - current.left) <= MERGE_THRESHOLD || // Overlapping current.left < next.left + next.width && next.left < current.left + current.width || // Very close (small gap) Math.abs(current.left + current.width - next.left) <= current.height * 0.2); if (horizontallyConnected) { const newLeft = Math.min(current.left, next.left); const newRight = Math.max(current.left + current.width, next.left + next.width); const newTop = Math.min(current.top, next.top); const newBottom = Math.max(current.top + current.height, next.top + next.height); current = { left: newLeft, top: newTop, width: newRight - newLeft, height: newBottom - newTop, pageNumber: current.pageNumber }; } else { consolidated.push(current); current = next; } } if (current) { consolidated.push(current); } return consolidated; }; var consolidateRects = (rects) => { if (rects.length <= 1) return rects; const result = []; const visited = /* @__PURE__ */ new Set(); for (let i = 0; i < rects.length; i++) { if (visited.has(i)) continue; const currentRect = rects[i]; if (!currentRect) continue; const currentGroup = [currentRect]; visited.add(i); let foundNew = true; while (foundNew) { foundNew = false; for (let j = 0; j < rects.length; j++) { if (visited.has(j)) continue; const candidateRect = rects[j]; if (!candidateRect) continue; const shouldMergeWithGroup = currentGroup.some( (groupRect) => doRectsOverlap(groupRect, candidateRect) ); if (shouldMergeWithGroup) { currentGroup.push(candidateRect); visited.add(j); foundNew = true; } } } result.push(mergeRectGroup(currentGroup)); } return result; }; var doRectsOverlap = (rect1, rect2) => { const horizontalOverlap = rect1.left < rect2.left + rect2.width && rect2.left < rect1.left + rect1.width; const verticalOverlap = rect1.top < rect2.top + rect2.height && rect2.top < rect1.top + rect1.height; const closeEnough = shouldMergeRects(rect1, rect2); return horizontalOverlap && verticalOverlap || closeEnough; }; var mergeRectGroup = (rects) => { if (rects.length === 1) { const rect = rects[0]; if (!rect) throw new Error("Invalid rect in group"); return rect; } const firstRect = rects[0]; if (!firstRect) throw new Error("Invalid first rect in group"); let minLeft = firstRect.left; let minTop = firstRect.top; let maxRight = firstRect.left + firstRect.width; let maxBottom = firstRect.top + firstRect.height; rects.forEach((rect) => { if (!rect) return; minLeft = Math.min(minLeft, rect.left); minTop = Math.min(minTop, rect.top); maxRight = Math.max(maxRight, rect.left + rect.width); maxBottom = Math.max(maxBottom, rect.top + rect.height); }); return { left: minLeft, top: minTop, width: maxRight - minLeft, height: maxBottom - minTop, pageNumber: firstRect.pageNumber }; }; var useSelectionDimensions = () => { const store = PDFStore.useContext(); const getAnnotationDimension = () => { const selection = window.getSelection(); if (!selection || selection.isCollapsed) return; const range = selection.getRangeAt(0); const highlightRects = []; const underlineRects = []; const textLayerMapHighlight = /* @__PURE__ */ new Map(); const textLayerMapUnderline = /* @__PURE__ */ new Map(); const clientRects = Array.from(range.getClientRects()).filter( (rect) => rect.width > 2 && rect.height > 2 ); clientRects.forEach((clientRect) => { let element = document.elementFromPoint( clientRect.left + 1, clientRect.top + clientRect.height / 2 ); if (!element) { element = document.elementFromPoint( clientRect.left + clientRect.width / 2, clientRect.top + clientRect.height / 2 ); } if (!element) { element = document.elementFromPoint( clientRect.right - 1, clientRect.top + clientRect.height / 2 ); } if (!element) { element = document.elementFromPoint( clientRect.left + clientRect.width / 2, clientRect.top + 1 ); } if (!element) { element = document.elementFromPoint( clientRect.left + clientRect.width / 2, clientRect.bottom - 1 ); } const textLayer = element?.closest(".textLayer"); if (!textLayer) return; const isSuperOrSubScript = (el) => { if (!el) return false; if (el.tagName.toLowerCase() === "sup" || el.tagName.toLowerCase() === "sub") { return true; } const classes = el.className; if (typeof classes === "string") { const superSubClasses = ["superscript", "subscript", "sup", "sub"]; if (superSubClasses.some((c) => classes.includes(c))) { return true; } } const elementRect = el.getBoundingClientRect(); if (elementRect.height < 6 && elementRect.width < 15) { const textContent = el.textContent?.trim() || ""; if (textContent.length <= 2 && /^[\d\w]{1,2}$/.test(textContent)) { const parentRect = el.parentElement?.getBoundingClientRect(); if (parentRect && parentRect.height > elementRect.height * 2) { const elementCenter = elementRect.top + elementRect.height / 2; const parentCenter = parentRect.top + parentRect.height / 2; const verticalOffset = Math.abs(elementCenter - parentCenter); if (verticalOffset > parentRect.height * 0.4) { return true; } } } } return false; }; const pageNumber = parseInt( textLayer.getAttribute("data-page-number") || "1", 10 ); const textLayerRect = textLayer.getBoundingClientRect(); const zoom = store.getState().zoom; const highlightRect = { width: clientRect.width / zoom, height: clientRect.height / zoom, top: (clientRect.top - textLayerRect.top) / zoom, left: (clientRect.left - textLayerRect.left) / zoom, pageNumber }; if (!textLayerMapHighlight.has(pageNumber)) { textLayerMapHighlight.set(pageNumber, []); } textLayerMapHighlight.get(pageNumber)?.push(highlightRect); const shouldCreateUnderline = !isSuperOrSubScript(element); if (shouldCreateUnderline) { const baselineOffset = clientRect.height * 0.85; const underlineHeight = 2; const underlineRect = { width: clientRect.width / zoom, height: underlineHeight / zoom, top: (clientRect.top - textLayerRect.top + baselineOffset) / zoom, left: (clientRect.left - textLayerRect.left) / zoom, pageNumber }; if (!textLayerMapUnderline.has(pageNumber)) { textLayerMapUnderline.set(pageNumber, []); } textLayerMapUnderline.get(pageNumber)?.push(underlineRect); } }); textLayerMapHighlight.forEach((rects) => { if (rects.length > 0) { if (rects.length === 1) { highlightRects.push(...rects); } else { const consolidated = consolidateHighlightRects(rects); highlightRects.push(...consolidated); } } }); textLayerMapUnderline.forEach((rects) => { if (rects.length > 0) { const lineGroups = groupRectsByLine(rects); lineGroups.forEach((group) => { if (group.length === 0) return; group.sort((a, b) => a.left - b.left); let i = 0; while (i < group.length) { const startRect = group[i]; if (!startRect) { i++; continue; } let endIndex = i; while (endIndex + 1 < group.length) { const currentRect = group[endIndex]; const nextRect = group[endIndex + 1]; if (!currentRect || !nextRect) break; const gap = nextRect.left - (currentRect.left + currentRect.width); const maxGapAllowed = Math.max(MERGE_THRESHOLD, currentRect.height * 0.3); if (gap <= maxGapAllowed) { endIndex++; } else { break; } } const endRect = group[endIndex]; if (!endRect) { i++; continue; } const lineRect = { width: endRect.left + endRect.width - startRect.left, height: 1.5, top: startRect.top, left: startRect.left, pageNumber: startRect.pageNumber }; underlineRects.push(lineRect); i = endIndex + 1; } }); } }); if (underlineRects.length === 0 && highlightRects.length > 0) { highlightRects.forEach((highlightRect) => { const baselineOffset = highlightRect.height * 0.85; const underlineHeight = 1.5; const underlineRect = { width: highlightRect.width, height: underlineHeight, top: highlightRect.top + baselineOffset, left: highlightRect.left, pageNumber: highlightRect.pageNumber }; underlineRects.push(underlineRect); }); } return { highlights: highlightRects.sort((a, b) => a.pageNumber - b.pageNumber), underlines: consolidateUnderlines(underlineRects).sort((a, b) => a.pageNumber - b.pageNumber), text: range.toString().trim(), isCollapsed: false }; }; const groupRectsByLine = (rects) => { const VERTICAL_TOLERANCE = 3; const groups = []; rects.forEach((rect) => { const centerY = rect.top + rect.height / 2; let foundGroup = false; for (const group of groups) { if (group.length === 0) continue; const firstRect = group[0]; if (!firstRect) continue; const groupCenterY = firstRect.top + firstRect.height / 2; if (Math.abs(centerY - groupCenterY) <= VERTICAL_TOLERANCE) { group.push(rect); foundGroup = true; break; } } if (!foundGroup) { groups.push([rect]); } }); return groups; }; const consolidateUnderlines = (underlines) => { if (underlines.length <= 1) return underlines; const consolidated = []; const sorted = [...underlines].sort((a, b) => { const pageCompare = a.pageNumber - b.pageNumber; if (pageCompare !== 0) return pageCompare; const topCompare = a.top - b.top; return Math.abs(topCompare) < 1 ? a.left - b.left : topCompare; }); let current = sorted[0]; for (let i = 1; i < sorted.length; i++) { const next = sorted[i]; const samePageAndLine = current.pageNumber === next.pageNumber && Math.abs(current.top - next.top) < 1; const horizontallyConnected = samePageAndLine && (Math.abs(current.left + current.width - next.left) <= MERGE_THRESHOLD || current.left < next.left + next.width && next.left < current.left + current.width); if (horizontallyConnected) { const newWidth = Math.max(current.left + current.width, next.left + next.width) - Math.min(current.left, next.left); current = { ...current, left: Math.min(current.left, next.left), width: newWidth }; } else { consolidated.push(current); current = next; } } consolidated.push(current); return consolidated; }; const getDimension = () => { const selection = window.getSelection(); if (!selection || selection.isCollapsed) return; const range = selection.getRangeAt(0); const highlights = []; const textLayerMap = /* @__PURE__ */ new Map(); const clientRects = Array.from(range.getClientRects()).filter( (rect) => rect.width > 2 && rect.height > 2 ); clientRects.forEach((clientRect) => { const element = document.elementFromPoint( clientRect.left + 1, clientRect.top + clientRect.height / 2 ); const textLayer = element?.closest(".textLayer"); if (!textLayer) return; const pageNumber = parseInt( textLayer.getAttribute("data-page-number") || "1", 10 ); const textLayerRect = textLayer.getBoundingClientRect(); const zoom = store.getState().zoom; const rect = { width: clientRect.width / zoom, height: clientRect.height / zoom, top: (clientRect.top - textLayerRect.top) / zoom, left: (clientRect.left - textLayerRect.left) / zoom, pageNumber }; if (!textLayerMap.has(pageNumber)) { textLayerMap.set(pageNumber, []); } textLayerMap.get(pageNumber)?.push(rect); }); textLayerMap.forEach((rects) => { if (rects.length > 0) { const consolidated = consolidateRects(rects); highlights.push(...consolidated); } }); return { highlights: highlights.sort((a, b) => a.pageNumber - b.pageNumber), text: range.toString().trim(), isCollapsed: false }; }; const getSelection = () => getDimension(); return { getDimension, getSelection, getAnnotationDimension }; }; var SelectionTooltip = ({ children }) => { const [isOpen, setIsOpen] = useState(false); const lastSelectionRef = useRef(null); const viewportRef = usePdf((state) => state.viewportRef); const { refs, floatingStyles, context } = useFloating({ placement: "bottom", open: isOpen, onOpenChange: setIsOpen, whileElementsMounted: autoUpdate, middleware: [offset(10), shift({ padding: 8 })] }); const dismiss = useDismiss(context); const { getFloatingProps } = useInteractions([dismiss]); const updateTooltipPosition = useCallback(() => { const selection = document.getSelection(); if (!selection || selection.isCollapsed) { setIsOpen(false); lastSelecti