2025-09-16 19:36:36 +01:00
|
|
|
import React, { createContext, useContext, useState, ReactNode, useRef } from 'react';
|
2025-09-15 13:33:39 +01:00
|
|
|
|
2025-09-16 19:36:36 +01:00
|
|
|
// State interfaces - represent the shape of data from each bridge
|
|
|
|
interface ScrollState {
|
|
|
|
currentPage: number;
|
|
|
|
totalPages: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ZoomState {
|
|
|
|
currentZoom: number;
|
|
|
|
zoomPercent: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface PanState {
|
|
|
|
isPanning: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface SelectionState {
|
|
|
|
hasSelection: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface SpreadState {
|
|
|
|
spreadMode: any;
|
|
|
|
isDualPage: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
interface RotationState {
|
|
|
|
rotation: number;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Bridge registration interface - bridges register with state and API
|
|
|
|
interface BridgeRef {
|
|
|
|
state: any;
|
|
|
|
api: any;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* ViewerContext provides a unified interface to EmbedPDF functionality.
|
|
|
|
*
|
|
|
|
* Architecture:
|
|
|
|
* - Bridges store their own state locally and register with this context
|
|
|
|
* - Context provides read-only access to bridge state via getter functions
|
|
|
|
* - Actions call EmbedPDF APIs directly through bridge references
|
|
|
|
* - No circular dependencies - bridges don't call back into this context
|
|
|
|
*/
|
2025-09-15 13:33:39 +01:00
|
|
|
interface ViewerContextType {
|
2025-09-16 19:36:36 +01:00
|
|
|
// UI state managed by this context
|
2025-09-15 13:33:39 +01:00
|
|
|
isThumbnailSidebarVisible: boolean;
|
|
|
|
toggleThumbnailSidebar: () => void;
|
2025-09-16 19:36:36 +01:00
|
|
|
|
|
|
|
// State getters - read current state from bridges
|
|
|
|
getScrollState: () => ScrollState;
|
|
|
|
getZoomState: () => ZoomState;
|
|
|
|
getPanState: () => PanState;
|
|
|
|
getSelectionState: () => SelectionState;
|
|
|
|
getSpreadState: () => SpreadState;
|
|
|
|
getRotationState: () => RotationState;
|
|
|
|
getSearchResults: () => any[] | null;
|
|
|
|
getSearchActiveIndex: () => number;
|
|
|
|
getThumbnailAPI: () => any;
|
|
|
|
|
|
|
|
// Action handlers - call EmbedPDF APIs directly
|
|
|
|
scrollActions: {
|
|
|
|
scrollToPage: (page: number) => void;
|
|
|
|
scrollToFirstPage: () => void;
|
|
|
|
scrollToPreviousPage: () => void;
|
|
|
|
scrollToNextPage: () => void;
|
|
|
|
scrollToLastPage: () => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
zoomActions: {
|
|
|
|
zoomIn: () => void;
|
|
|
|
zoomOut: () => void;
|
|
|
|
toggleMarqueeZoom: () => void;
|
|
|
|
requestZoom: (level: number) => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
panActions: {
|
|
|
|
enablePan: () => void;
|
|
|
|
disablePan: () => void;
|
|
|
|
togglePan: () => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
selectionActions: {
|
|
|
|
copyToClipboard: () => void;
|
|
|
|
getSelectedText: () => string;
|
|
|
|
getFormattedSelection: () => any;
|
|
|
|
};
|
|
|
|
|
|
|
|
spreadActions: {
|
|
|
|
setSpreadMode: (mode: any) => void;
|
|
|
|
getSpreadMode: () => any;
|
|
|
|
toggleSpreadMode: () => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
rotationActions: {
|
|
|
|
rotateForward: () => void;
|
|
|
|
rotateBackward: () => void;
|
|
|
|
setRotation: (rotation: number) => void;
|
|
|
|
getRotation: () => number;
|
|
|
|
};
|
|
|
|
|
|
|
|
searchActions: {
|
|
|
|
search: (query: string) => Promise<void>;
|
|
|
|
next: () => void;
|
|
|
|
previous: () => void;
|
|
|
|
clear: () => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Bridge registration - internal use by bridges
|
|
|
|
registerBridge: (type: string, ref: BridgeRef) => void;
|
2025-09-15 13:33:39 +01:00
|
|
|
}
|
|
|
|
|
2025-09-16 19:36:36 +01:00
|
|
|
export const ViewerContext = createContext<ViewerContextType | null>(null);
|
2025-09-15 13:33:39 +01:00
|
|
|
|
|
|
|
interface ViewerProviderProps {
|
|
|
|
children: ReactNode;
|
|
|
|
}
|
|
|
|
|
|
|
|
export const ViewerProvider: React.FC<ViewerProviderProps> = ({ children }) => {
|
2025-09-16 19:36:36 +01:00
|
|
|
// UI state - only state directly managed by this context
|
2025-09-15 13:33:39 +01:00
|
|
|
const [isThumbnailSidebarVisible, setIsThumbnailSidebarVisible] = useState(false);
|
2025-09-16 19:36:36 +01:00
|
|
|
|
|
|
|
// Bridge registry - bridges register their state and APIs here
|
|
|
|
const bridgeRefs = useRef({
|
|
|
|
scroll: null as BridgeRef | null,
|
|
|
|
zoom: null as BridgeRef | null,
|
|
|
|
pan: null as BridgeRef | null,
|
|
|
|
selection: null as BridgeRef | null,
|
|
|
|
search: null as BridgeRef | null,
|
|
|
|
spread: null as BridgeRef | null,
|
|
|
|
rotation: null as BridgeRef | null,
|
|
|
|
thumbnail: null as BridgeRef | null,
|
|
|
|
});
|
|
|
|
|
|
|
|
const registerBridge = (type: string, ref: BridgeRef) => {
|
|
|
|
bridgeRefs.current[type as keyof typeof bridgeRefs.current] = ref;
|
|
|
|
};
|
2025-09-15 13:33:39 +01:00
|
|
|
|
|
|
|
const toggleThumbnailSidebar = () => {
|
|
|
|
setIsThumbnailSidebarVisible(prev => !prev);
|
|
|
|
};
|
|
|
|
|
2025-09-16 19:36:36 +01:00
|
|
|
// State getters - read from bridge refs
|
|
|
|
const getScrollState = (): ScrollState => {
|
|
|
|
return bridgeRefs.current.scroll?.state || { currentPage: 1, totalPages: 0 };
|
|
|
|
};
|
|
|
|
|
|
|
|
const getZoomState = (): ZoomState => {
|
|
|
|
return bridgeRefs.current.zoom?.state || { currentZoom: 1.4, zoomPercent: 140 };
|
|
|
|
};
|
|
|
|
|
|
|
|
const getPanState = (): PanState => {
|
|
|
|
return bridgeRefs.current.pan?.state || { isPanning: false };
|
|
|
|
};
|
|
|
|
|
|
|
|
const getSelectionState = (): SelectionState => {
|
|
|
|
return bridgeRefs.current.selection?.state || { hasSelection: false };
|
|
|
|
};
|
|
|
|
|
|
|
|
const getSpreadState = (): SpreadState => {
|
|
|
|
return bridgeRefs.current.spread?.state || { spreadMode: null, isDualPage: false };
|
|
|
|
};
|
|
|
|
|
|
|
|
const getRotationState = (): RotationState => {
|
|
|
|
return bridgeRefs.current.rotation?.state || { rotation: 0 };
|
|
|
|
};
|
|
|
|
|
|
|
|
const getSearchResults = () => {
|
|
|
|
return bridgeRefs.current.search?.state?.results || null;
|
|
|
|
};
|
|
|
|
|
|
|
|
const getSearchActiveIndex = () => {
|
|
|
|
return bridgeRefs.current.search?.state?.activeIndex || 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
const getThumbnailAPI = () => {
|
|
|
|
return bridgeRefs.current.thumbnail?.api || null;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Action handlers - call APIs directly
|
|
|
|
const scrollActions = {
|
|
|
|
scrollToPage: (page: number) => {
|
|
|
|
const api = bridgeRefs.current.scroll?.api;
|
|
|
|
if (api?.scrollToPage) {
|
|
|
|
api.scrollToPage({ pageNumber: page });
|
|
|
|
}
|
|
|
|
},
|
|
|
|
scrollToFirstPage: () => {
|
|
|
|
const api = bridgeRefs.current.scroll?.api;
|
|
|
|
if (api?.scrollToPage) {
|
|
|
|
api.scrollToPage({ pageNumber: 1 });
|
|
|
|
}
|
|
|
|
},
|
|
|
|
scrollToPreviousPage: () => {
|
|
|
|
const api = bridgeRefs.current.scroll?.api;
|
|
|
|
if (api?.scrollToPreviousPage) {
|
|
|
|
api.scrollToPreviousPage();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
scrollToNextPage: () => {
|
|
|
|
const api = bridgeRefs.current.scroll?.api;
|
|
|
|
if (api?.scrollToNextPage) {
|
|
|
|
api.scrollToNextPage();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
scrollToLastPage: () => {
|
|
|
|
const scrollState = getScrollState();
|
|
|
|
const api = bridgeRefs.current.scroll?.api;
|
|
|
|
if (api?.scrollToPage && scrollState.totalPages > 0) {
|
|
|
|
api.scrollToPage({ pageNumber: scrollState.totalPages });
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const zoomActions = {
|
|
|
|
zoomIn: () => {
|
|
|
|
const api = bridgeRefs.current.zoom?.api;
|
|
|
|
if (api?.zoomIn) {
|
|
|
|
api.zoomIn();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
zoomOut: () => {
|
|
|
|
const api = bridgeRefs.current.zoom?.api;
|
|
|
|
if (api?.zoomOut) {
|
|
|
|
api.zoomOut();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
toggleMarqueeZoom: () => {
|
|
|
|
const api = bridgeRefs.current.zoom?.api;
|
|
|
|
if (api?.toggleMarqueeZoom) {
|
|
|
|
api.toggleMarqueeZoom();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
requestZoom: (level: number) => {
|
|
|
|
const api = bridgeRefs.current.zoom?.api;
|
|
|
|
if (api?.requestZoom) {
|
|
|
|
api.requestZoom(level);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const panActions = {
|
|
|
|
enablePan: () => {
|
|
|
|
const api = bridgeRefs.current.pan?.api;
|
|
|
|
if (api?.enable) {
|
|
|
|
api.enable();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
disablePan: () => {
|
|
|
|
const api = bridgeRefs.current.pan?.api;
|
|
|
|
if (api?.disable) {
|
|
|
|
api.disable();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
togglePan: () => {
|
|
|
|
const api = bridgeRefs.current.pan?.api;
|
|
|
|
if (api?.toggle) {
|
|
|
|
api.toggle();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const selectionActions = {
|
|
|
|
copyToClipboard: () => {
|
|
|
|
const api = bridgeRefs.current.selection?.api;
|
|
|
|
if (api?.copyToClipboard) {
|
|
|
|
api.copyToClipboard();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
getSelectedText: () => {
|
|
|
|
const api = bridgeRefs.current.selection?.api;
|
|
|
|
if (api?.getSelectedText) {
|
|
|
|
return api.getSelectedText();
|
|
|
|
}
|
|
|
|
return '';
|
|
|
|
},
|
|
|
|
getFormattedSelection: () => {
|
|
|
|
const api = bridgeRefs.current.selection?.api;
|
|
|
|
if (api?.getFormattedSelection) {
|
|
|
|
return api.getFormattedSelection();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const spreadActions = {
|
|
|
|
setSpreadMode: (mode: any) => {
|
|
|
|
const api = bridgeRefs.current.spread?.api;
|
|
|
|
if (api?.setSpreadMode) {
|
|
|
|
api.setSpreadMode(mode);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
getSpreadMode: () => {
|
|
|
|
const api = bridgeRefs.current.spread?.api;
|
|
|
|
if (api?.getSpreadMode) {
|
|
|
|
return api.getSpreadMode();
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
toggleSpreadMode: () => {
|
|
|
|
const api = bridgeRefs.current.spread?.api;
|
|
|
|
if (api?.toggleSpreadMode) {
|
|
|
|
api.toggleSpreadMode();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const rotationActions = {
|
|
|
|
rotateForward: () => {
|
|
|
|
const api = bridgeRefs.current.rotation?.api;
|
|
|
|
if (api?.rotateForward) {
|
|
|
|
api.rotateForward();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
rotateBackward: () => {
|
|
|
|
const api = bridgeRefs.current.rotation?.api;
|
|
|
|
if (api?.rotateBackward) {
|
|
|
|
api.rotateBackward();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
setRotation: (rotation: number) => {
|
|
|
|
const api = bridgeRefs.current.rotation?.api;
|
|
|
|
if (api?.setRotation) {
|
|
|
|
api.setRotation(rotation);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
getRotation: () => {
|
|
|
|
const api = bridgeRefs.current.rotation?.api;
|
|
|
|
if (api?.getRotation) {
|
|
|
|
return api.getRotation();
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const searchActions = {
|
|
|
|
search: async (query: string) => {
|
|
|
|
const api = bridgeRefs.current.search?.api;
|
|
|
|
if (api?.search) {
|
|
|
|
return api.search(query);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
next: () => {
|
|
|
|
const api = bridgeRefs.current.search?.api;
|
|
|
|
if (api?.next) {
|
|
|
|
api.next();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
previous: () => {
|
|
|
|
const api = bridgeRefs.current.search?.api;
|
|
|
|
if (api?.previous) {
|
|
|
|
api.previous();
|
|
|
|
}
|
|
|
|
},
|
|
|
|
clear: () => {
|
|
|
|
const api = bridgeRefs.current.search?.api;
|
|
|
|
if (api?.clear) {
|
|
|
|
api.clear();
|
|
|
|
}
|
|
|
|
}
|
2025-09-15 13:33:39 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
const value: ViewerContextType = {
|
2025-09-16 19:36:36 +01:00
|
|
|
// UI state
|
2025-09-15 13:33:39 +01:00
|
|
|
isThumbnailSidebarVisible,
|
|
|
|
toggleThumbnailSidebar,
|
2025-09-16 19:36:36 +01:00
|
|
|
|
|
|
|
// State getters
|
|
|
|
getScrollState,
|
|
|
|
getZoomState,
|
|
|
|
getPanState,
|
|
|
|
getSelectionState,
|
|
|
|
getSpreadState,
|
|
|
|
getRotationState,
|
|
|
|
getSearchResults,
|
|
|
|
getSearchActiveIndex,
|
|
|
|
getThumbnailAPI,
|
|
|
|
|
|
|
|
// Actions
|
|
|
|
scrollActions,
|
|
|
|
zoomActions,
|
|
|
|
panActions,
|
|
|
|
selectionActions,
|
|
|
|
spreadActions,
|
|
|
|
rotationActions,
|
|
|
|
searchActions,
|
|
|
|
|
|
|
|
// Bridge registration
|
|
|
|
registerBridge,
|
2025-09-15 13:33:39 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
return (
|
|
|
|
<ViewerContext.Provider value={value}>
|
|
|
|
{children}
|
|
|
|
</ViewerContext.Provider>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export const useViewer = (): ViewerContextType => {
|
|
|
|
const context = useContext(ViewerContext);
|
|
|
|
if (!context) {
|
|
|
|
throw new Error('useViewer must be used within a ViewerProvider');
|
|
|
|
}
|
|
|
|
return context;
|
|
|
|
};
|