2025-06-10 11:19:54 +01:00
|
|
|
import { useState, useCallback } from 'react';
|
|
|
|
import { getDocument } from 'pdfjs-dist';
|
|
|
|
import { PDFDocument, PDFPage } from '../types/pageEditor';
|
|
|
|
|
|
|
|
export function usePDFProcessor() {
|
|
|
|
const [loading, setLoading] = useState(false);
|
|
|
|
const [error, setError] = useState<string | null>(null);
|
|
|
|
|
|
|
|
const generatePageThumbnail = useCallback(async (
|
|
|
|
file: File,
|
|
|
|
pageNumber: number,
|
|
|
|
scale: number = 0.5
|
|
|
|
): Promise<string> => {
|
|
|
|
try {
|
|
|
|
const arrayBuffer = await file.arrayBuffer();
|
|
|
|
const pdf = await getDocument({ data: arrayBuffer }).promise;
|
|
|
|
const page = await pdf.getPage(pageNumber);
|
|
|
|
|
|
|
|
const viewport = page.getViewport({ scale });
|
|
|
|
const canvas = document.createElement('canvas');
|
|
|
|
canvas.width = viewport.width;
|
|
|
|
canvas.height = viewport.height;
|
|
|
|
|
|
|
|
const context = canvas.getContext('2d');
|
|
|
|
if (!context) {
|
|
|
|
throw new Error('Could not get canvas context');
|
|
|
|
}
|
|
|
|
|
|
|
|
await page.render({ canvasContext: context, viewport }).promise;
|
|
|
|
const thumbnail = canvas.toDataURL();
|
|
|
|
|
|
|
|
// Clean up
|
|
|
|
pdf.destroy();
|
|
|
|
|
|
|
|
return thumbnail;
|
|
|
|
} catch (error) {
|
|
|
|
console.error('Failed to generate thumbnail:', error);
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
const processPDFFile = useCallback(async (file: File): Promise<PDFDocument> => {
|
|
|
|
setLoading(true);
|
|
|
|
setError(null);
|
|
|
|
|
|
|
|
try {
|
|
|
|
const arrayBuffer = await file.arrayBuffer();
|
|
|
|
const pdf = await getDocument({ data: arrayBuffer }).promise;
|
|
|
|
const totalPages = pdf.numPages;
|
|
|
|
|
|
|
|
const pages: PDFPage[] = [];
|
|
|
|
|
2025-07-16 17:53:50 +01:00
|
|
|
// Create pages without thumbnails initially - load them lazily
|
2025-06-10 11:19:54 +01:00
|
|
|
for (let i = 1; i <= totalPages; i++) {
|
|
|
|
pages.push({
|
|
|
|
id: `${file.name}-page-${i}`,
|
|
|
|
pageNumber: i,
|
2025-07-16 17:53:50 +01:00
|
|
|
thumbnail: null, // Will be loaded lazily
|
2025-06-10 11:19:54 +01:00
|
|
|
rotation: 0,
|
|
|
|
selected: false
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2025-07-16 17:53:50 +01:00
|
|
|
// Generate thumbnails for first 10 pages immediately for better UX
|
|
|
|
const priorityPages = Math.min(10, totalPages);
|
|
|
|
for (let i = 1; i <= priorityPages; i++) {
|
|
|
|
try {
|
|
|
|
const thumbnail = await generatePageThumbnail(file, i);
|
|
|
|
pages[i - 1].thumbnail = thumbnail;
|
|
|
|
} catch (error) {
|
|
|
|
console.warn(`Failed to generate thumbnail for page ${i}:`, error);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2025-06-10 11:19:54 +01:00
|
|
|
// Clean up
|
|
|
|
pdf.destroy();
|
|
|
|
|
|
|
|
const document: PDFDocument = {
|
|
|
|
id: `${Date.now()}-${Math.random().toString(36).substr(2, 9)}`,
|
|
|
|
name: file.name,
|
|
|
|
file,
|
|
|
|
pages,
|
|
|
|
totalPages
|
|
|
|
};
|
|
|
|
|
|
|
|
return document;
|
|
|
|
} catch (error) {
|
|
|
|
const errorMessage = error instanceof Error ? error.message : 'Failed to process PDF';
|
|
|
|
setError(errorMessage);
|
|
|
|
throw error;
|
|
|
|
} finally {
|
|
|
|
setLoading(false);
|
|
|
|
}
|
|
|
|
}, [generatePageThumbnail]);
|
|
|
|
|
|
|
|
return {
|
|
|
|
processPDFFile,
|
|
|
|
generatePageThumbnail,
|
|
|
|
loading,
|
|
|
|
error
|
|
|
|
};
|
|
|
|
}
|