mirror of
https://github.com/Frooodle/Stirling-PDF.git
synced 2026-03-04 02:20:19 +01:00
# Description of Changes A new universal file context rather than the splintered ones for the main views, tools and manager we had before (manager still has its own but its better integreated with the core context) File context has been split it into a handful of different files managing various file related issues separately to reduce the monolith - FileReducer.ts - State management fileActions.ts - File operations fileSelectors.ts - Data access patterns lifecycle.ts - Resource cleanup and memory management fileHooks.ts - React hooks interface contexts.ts - Context providers Improved thumbnail generation Improved indexxedb handling Stopped handling files as blobs were not necessary to improve performance A new library handling drag and drop https://github.com/atlassian/pragmatic-drag-and-drop (Out of scope yes but I broke the old one with the new filecontext and it needed doing so it was a might as well) A new library handling virtualisation on page editor @tanstack/react-virtual, as above. Quickly ripped out the last remnants of the old URL params stuff and replaced with the beginnings of what will later become the new URL navigation system (for now it just restores the tool name in url behavior) Fixed selected file not regestered when opening a tool Fixed png thumbnails Closes #(issue_number) --- ## Checklist ### General - [ ] I have read the [Contribution Guidelines](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/CONTRIBUTING.md) - [ ] I have read the [Stirling-PDF Developer Guide](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/DeveloperGuide.md) (if applicable) - [ ] I have read the [How to add new languages to Stirling-PDF](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/HowToAddNewLanguage.md) (if applicable) - [ ] I have performed a self-review of my own code - [ ] My changes generate no new warnings ### Documentation - [ ] I have updated relevant docs on [Stirling-PDF's doc repo](https://github.com/Stirling-Tools/Stirling-Tools.github.io/blob/main/docs/) (if functionality has heavily changed) - [ ] I have read the section [Add New Translation Tags](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/HowToAddNewLanguage.md#add-new-translation-tags) (for new translation tags only) ### UI Changes (if applicable) - [ ] Screenshots or videos demonstrating the UI changes are attached (e.g., as comments or direct attachments in the PR) ### Testing (if applicable) - [ ] I have tested my changes locally. Refer to the [Testing Guide](https://github.com/Stirling-Tools/Stirling-PDF/blob/main/devGuide/DeveloperGuide.md#6-testing) for more details. --------- Co-authored-by: Reece Browne <you@example.com>
429 lines
17 KiB
TypeScript
429 lines
17 KiB
TypeScript
import { pdfWorkerManager } from '../services/pdfWorkerManager';
|
|
|
|
export interface ThumbnailWithMetadata {
|
|
thumbnail: string; // Always returns a thumbnail (placeholder if needed)
|
|
pageCount: number;
|
|
}
|
|
|
|
interface ColorScheme {
|
|
bgTop: string;
|
|
bgBottom: string;
|
|
border: string;
|
|
icon: string;
|
|
badge: string;
|
|
textPrimary: string;
|
|
textSecondary: string;
|
|
}
|
|
|
|
/**
|
|
* Calculate thumbnail scale based on file size (modern 2024 scaling)
|
|
*/
|
|
export function calculateScaleFromFileSize(fileSize: number): number {
|
|
const MB = 1024 * 1024;
|
|
if (fileSize < 10 * MB) return 1.0; // Full quality for small files
|
|
if (fileSize < 50 * MB) return 0.8; // High quality for common file sizes
|
|
if (fileSize < 200 * MB) return 0.6; // Good quality for typical large files
|
|
if (fileSize < 500 * MB) return 0.4; // Readable quality for large but manageable files
|
|
return 0.3; // Still usable quality, not tiny
|
|
}
|
|
|
|
|
|
/**
|
|
* Generate encrypted PDF thumbnail with lock icon
|
|
*/
|
|
function generateEncryptedPDFThumbnail(file: File): string {
|
|
const canvas = document.createElement('canvas');
|
|
canvas.width = 120;
|
|
canvas.height = 150;
|
|
const ctx = canvas.getContext('2d')!;
|
|
|
|
// Use PDF color scheme but with encrypted styling
|
|
const colorScheme = getFileTypeColorScheme('PDF');
|
|
|
|
// Create gradient background
|
|
const gradient = ctx.createLinearGradient(0, 0, 0, canvas.height);
|
|
gradient.addColorStop(0, colorScheme.bgTop);
|
|
gradient.addColorStop(1, colorScheme.bgBottom);
|
|
|
|
// Rounded rectangle background
|
|
drawRoundedRect(ctx, 8, 8, canvas.width - 16, canvas.height - 16, 8);
|
|
ctx.fillStyle = gradient;
|
|
ctx.fill();
|
|
|
|
// Border with dashed pattern for encrypted indicator
|
|
ctx.strokeStyle = colorScheme.border;
|
|
ctx.lineWidth = 2;
|
|
ctx.setLineDash([4, 4]);
|
|
ctx.stroke();
|
|
ctx.setLineDash([]); // Reset dash pattern
|
|
|
|
// Large lock icon as main element
|
|
drawLargeLockIcon(ctx, canvas.width / 2, canvas.height / 2 - 10, colorScheme);
|
|
|
|
// "PDF" text under the lock
|
|
ctx.font = 'bold 14px -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif';
|
|
ctx.fillStyle = colorScheme.icon;
|
|
ctx.textAlign = 'center';
|
|
ctx.fillText('PDF', canvas.width / 2, canvas.height / 2 + 35);
|
|
|
|
// File size with subtle styling
|
|
const sizeText = formatFileSize(file.size);
|
|
ctx.font = '11px -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif';
|
|
ctx.fillStyle = colorScheme.textSecondary;
|
|
ctx.textAlign = 'center';
|
|
ctx.fillText(sizeText, canvas.width / 2, canvas.height - 15);
|
|
|
|
return canvas.toDataURL();
|
|
}
|
|
|
|
/**
|
|
* Generate modern placeholder thumbnail with file extension
|
|
*/
|
|
function generatePlaceholderThumbnail(file: File): string {
|
|
const canvas = document.createElement('canvas');
|
|
canvas.width = 120;
|
|
canvas.height = 150;
|
|
const ctx = canvas.getContext('2d')!;
|
|
|
|
// Get file extension for color theming
|
|
const extension = file.name.split('.').pop()?.toUpperCase() || 'FILE';
|
|
const colorScheme = getFileTypeColorScheme(extension);
|
|
|
|
// Create gradient background
|
|
const gradient = ctx.createLinearGradient(0, 0, 0, canvas.height);
|
|
gradient.addColorStop(0, colorScheme.bgTop);
|
|
gradient.addColorStop(1, colorScheme.bgBottom);
|
|
|
|
// Rounded rectangle background
|
|
drawRoundedRect(ctx, 8, 8, canvas.width - 16, canvas.height - 16, 8);
|
|
ctx.fillStyle = gradient;
|
|
ctx.fill();
|
|
|
|
// Subtle shadow/border
|
|
ctx.strokeStyle = colorScheme.border;
|
|
ctx.lineWidth = 1.5;
|
|
ctx.stroke();
|
|
|
|
// Modern document icon
|
|
drawModernDocumentIcon(ctx, canvas.width / 2, 45, colorScheme.icon);
|
|
|
|
// Extension badge
|
|
drawExtensionBadge(ctx, canvas.width / 2, canvas.height / 2 + 15, extension, colorScheme);
|
|
|
|
// File size with subtle styling
|
|
const sizeText = formatFileSize(file.size);
|
|
ctx.font = '11px -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif';
|
|
ctx.fillStyle = colorScheme.textSecondary;
|
|
ctx.textAlign = 'center';
|
|
ctx.fillText(sizeText, canvas.width / 2, canvas.height - 15);
|
|
|
|
return canvas.toDataURL();
|
|
}
|
|
|
|
/**
|
|
* Get color scheme based on file extension
|
|
*/
|
|
function getFileTypeColorScheme(extension: string): ColorScheme {
|
|
const schemes: Record<string, ColorScheme> = {
|
|
// Documents
|
|
'PDF': { bgTop: '#FF6B6B20', bgBottom: '#FF6B6B10', border: '#FF6B6B40', icon: '#FF6B6B', badge: '#FF6B6B', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'DOC': { bgTop: '#4ECDC420', bgBottom: '#4ECDC410', border: '#4ECDC440', icon: '#4ECDC4', badge: '#4ECDC4', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'DOCX': { bgTop: '#4ECDC420', bgBottom: '#4ECDC410', border: '#4ECDC440', icon: '#4ECDC4', badge: '#4ECDC4', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'ODT': { bgTop: '#4ECDC420', bgBottom: '#4ECDC410', border: '#4ECDC440', icon: '#4ECDC4', badge: '#4ECDC4', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'TXT': { bgTop: '#95A5A620', bgBottom: '#95A5A610', border: '#95A5A640', icon: '#95A5A6', badge: '#95A5A6', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'RTF': { bgTop: '#95A5A620', bgBottom: '#95A5A610', border: '#95A5A640', icon: '#95A5A6', badge: '#95A5A6', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
|
|
// Spreadsheets
|
|
'XLS': { bgTop: '#2ECC7120', bgBottom: '#2ECC7110', border: '#2ECC7140', icon: '#2ECC71', badge: '#2ECC71', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'XLSX': { bgTop: '#2ECC7120', bgBottom: '#2ECC7110', border: '#2ECC7140', icon: '#2ECC71', badge: '#2ECC71', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'ODS': { bgTop: '#2ECC7120', bgBottom: '#2ECC7110', border: '#2ECC7140', icon: '#2ECC71', badge: '#2ECC71', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'CSV': { bgTop: '#2ECC7120', bgBottom: '#2ECC7110', border: '#2ECC7140', icon: '#2ECC71', badge: '#2ECC71', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
|
|
// Presentations
|
|
'PPT': { bgTop: '#E67E2220', bgBottom: '#E67E2210', border: '#E67E2240', icon: '#E67E22', badge: '#E67E22', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'PPTX': { bgTop: '#E67E2220', bgBottom: '#E67E2210', border: '#E67E2240', icon: '#E67E22', badge: '#E67E22', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'ODP': { bgTop: '#E67E2220', bgBottom: '#E67E2210', border: '#E67E2240', icon: '#E67E22', badge: '#E67E22', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
|
|
// Images
|
|
'JPG': { bgTop: '#FF9F4320', bgBottom: '#FF9F4310', border: '#FF9F4340', icon: '#FF9F43', badge: '#FF9F43', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'JPEG': { bgTop: '#FF9F4320', bgBottom: '#FF9F4310', border: '#FF9F4340', icon: '#FF9F43', badge: '#FF9F43', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'PNG': { bgTop: '#FF9F4320', bgBottom: '#FF9F4310', border: '#FF9F4340', icon: '#FF9F43', badge: '#FF9F43', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'GIF': { bgTop: '#FF9F4320', bgBottom: '#FF9F4310', border: '#FF9F4340', icon: '#FF9F43', badge: '#FF9F43', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'BMP': { bgTop: '#FF9F4320', bgBottom: '#FF9F4310', border: '#FF9F4340', icon: '#FF9F43', badge: '#FF9F43', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'TIFF': { bgTop: '#FF9F4320', bgBottom: '#FF9F4310', border: '#FF9F4340', icon: '#FF9F43', badge: '#FF9F43', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'WEBP': { bgTop: '#FF9F4320', bgBottom: '#FF9F4310', border: '#FF9F4340', icon: '#FF9F43', badge: '#FF9F43', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'SVG': { bgTop: '#FF9F4320', bgBottom: '#FF9F4310', border: '#FF9F4340', icon: '#FF9F43', badge: '#FF9F43', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
|
|
// Web
|
|
'HTML': { bgTop: '#FD79A820', bgBottom: '#FD79A810', border: '#FD79A840', icon: '#FD79A8', badge: '#FD79A8', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'XML': { bgTop: '#FD79A820', bgBottom: '#FD79A810', border: '#FD79A840', icon: '#FD79A8', badge: '#FD79A8', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
|
|
// Text/Markup
|
|
'MD': { bgTop: '#6C5CE720', bgBottom: '#6C5CE710', border: '#6C5CE740', icon: '#6C5CE7', badge: '#6C5CE7', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
|
|
// Email
|
|
'EML': { bgTop: '#A29BFE20', bgBottom: '#A29BFE10', border: '#A29BFE40', icon: '#A29BFE', badge: '#A29BFE', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
|
|
// Archives
|
|
'ZIP': { bgTop: '#9B59B620', bgBottom: '#9B59B610', border: '#9B59B640', icon: '#9B59B6', badge: '#9B59B6', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'RAR': { bgTop: '#9B59B620', bgBottom: '#9B59B610', border: '#9B59B640', icon: '#9B59B6', badge: '#9B59B6', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
'7Z': { bgTop: '#9B59B620', bgBottom: '#9B59B610', border: '#9B59B640', icon: '#9B59B6', badge: '#9B59B6', textPrimary: '#FFFFFF', textSecondary: '#666666' },
|
|
|
|
// Default
|
|
'DEFAULT': { bgTop: '#74B9FF20', bgBottom: '#74B9FF10', border: '#74B9FF40', icon: '#74B9FF', badge: '#74B9FF', textPrimary: '#FFFFFF', textSecondary: '#666666' }
|
|
};
|
|
|
|
return schemes[extension] || schemes['DEFAULT'];
|
|
}
|
|
|
|
/**
|
|
* Draw rounded rectangle
|
|
*/
|
|
function drawRoundedRect(ctx: CanvasRenderingContext2D, x: number, y: number, width: number, height: number, radius: number) {
|
|
ctx.beginPath();
|
|
ctx.moveTo(x + radius, y);
|
|
ctx.lineTo(x + width - radius, y);
|
|
ctx.quadraticCurveTo(x + width, y, x + width, y + radius);
|
|
ctx.lineTo(x + width, y + height - radius);
|
|
ctx.quadraticCurveTo(x + width, y + height, x + width - radius, y + height);
|
|
ctx.lineTo(x + radius, y + height);
|
|
ctx.quadraticCurveTo(x, y + height, x, y + height - radius);
|
|
ctx.lineTo(x, y + radius);
|
|
ctx.quadraticCurveTo(x, y, x + radius, y);
|
|
ctx.closePath();
|
|
}
|
|
|
|
/**
|
|
* Draw modern document icon
|
|
*/
|
|
function drawModernDocumentIcon(ctx: CanvasRenderingContext2D, centerX: number, centerY: number, color: string) {
|
|
const size = 24;
|
|
ctx.fillStyle = color;
|
|
ctx.strokeStyle = color;
|
|
ctx.lineWidth = 2;
|
|
|
|
// Document body
|
|
drawRoundedRect(ctx, centerX - size/2, centerY - size/2, size, size * 1.2, 3);
|
|
ctx.fill();
|
|
|
|
// Folded corner
|
|
ctx.beginPath();
|
|
ctx.moveTo(centerX + size/2 - 6, centerY - size/2);
|
|
ctx.lineTo(centerX + size/2, centerY - size/2 + 6);
|
|
ctx.lineTo(centerX + size/2 - 6, centerY - size/2 + 6);
|
|
ctx.closePath();
|
|
ctx.fillStyle = '#FFFFFF40';
|
|
ctx.fill();
|
|
}
|
|
|
|
/**
|
|
* Draw large lock icon for encrypted PDFs
|
|
*/
|
|
function drawLargeLockIcon(ctx: CanvasRenderingContext2D, centerX: number, centerY: number, colorScheme: ColorScheme) {
|
|
const size = 48;
|
|
ctx.fillStyle = colorScheme.icon;
|
|
ctx.strokeStyle = colorScheme.icon;
|
|
ctx.lineWidth = 3;
|
|
|
|
// Lock body (rectangle)
|
|
const bodyWidth = size;
|
|
const bodyHeight = size * 0.75;
|
|
const bodyX = centerX - bodyWidth / 2;
|
|
const bodyY = centerY - bodyHeight / 4;
|
|
|
|
drawRoundedRect(ctx, bodyX, bodyY, bodyWidth, bodyHeight, 4);
|
|
ctx.fill();
|
|
|
|
// Lock shackle (semicircle)
|
|
const shackleRadius = size * 0.32;
|
|
const shackleY = centerY - size * 0.25;
|
|
|
|
ctx.beginPath();
|
|
ctx.arc(centerX, shackleY, shackleRadius, Math.PI, 2 * Math.PI);
|
|
ctx.stroke();
|
|
|
|
// Keyhole
|
|
const keyholeX = centerX;
|
|
const keyholeY = bodyY + bodyHeight * 0.4;
|
|
ctx.fillStyle = colorScheme.textPrimary;
|
|
ctx.beginPath();
|
|
ctx.arc(keyholeX, keyholeY, 4, 0, 2 * Math.PI);
|
|
ctx.fill();
|
|
ctx.fillRect(keyholeX - 2, keyholeY, 4, 8);
|
|
}
|
|
|
|
/**
|
|
* Generate standard PDF thumbnail by rendering first page
|
|
*/
|
|
async function generateStandardPDFThumbnail(pdf: any, scale: number): Promise<string> {
|
|
const page = await pdf.getPage(1);
|
|
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;
|
|
return canvas.toDataURL();
|
|
}
|
|
|
|
/**
|
|
* Draw extension badge
|
|
*/
|
|
function drawExtensionBadge(ctx: CanvasRenderingContext2D, centerX: number, centerY: number, extension: string, colorScheme: ColorScheme) {
|
|
const badgeWidth = Math.max(extension.length * 8 + 16, 40);
|
|
const badgeHeight = 22;
|
|
|
|
// Badge background
|
|
drawRoundedRect(ctx, centerX - badgeWidth/2, centerY - badgeHeight/2, badgeWidth, badgeHeight, 11);
|
|
ctx.fillStyle = colorScheme.badge;
|
|
ctx.fill();
|
|
|
|
// Badge text
|
|
ctx.font = 'bold 11px -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, sans-serif';
|
|
ctx.fillStyle = colorScheme.textPrimary;
|
|
ctx.textAlign = 'center';
|
|
ctx.fillText(extension, centerX, centerY + 4);
|
|
}
|
|
|
|
/**
|
|
* Format file size for display
|
|
*/
|
|
function formatFileSize(bytes: number): string {
|
|
if (bytes === 0) return '0 B';
|
|
const k = 1024;
|
|
const sizes = ['B', 'KB', 'MB', 'GB'];
|
|
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(1)) + ' ' + sizes[i];
|
|
}
|
|
|
|
async function generatePDFThumbnail(arrayBuffer: ArrayBuffer, file: File, scale: number): Promise<string> {
|
|
try {
|
|
const pdf = await pdfWorkerManager.createDocument(arrayBuffer, {
|
|
disableAutoFetch: true,
|
|
disableStream: true
|
|
});
|
|
|
|
const thumbnail = await generateStandardPDFThumbnail(pdf, scale);
|
|
|
|
// Immediately clean up memory after thumbnail generation using worker manager
|
|
pdfWorkerManager.destroyDocument(pdf);
|
|
return thumbnail;
|
|
} catch (error) {
|
|
if (error instanceof Error) {
|
|
// Check if PDF is encrypted
|
|
if (error.name === "PasswordException") {
|
|
return generateEncryptedPDFThumbnail(file);
|
|
}
|
|
}
|
|
throw error; // Not an encryption issue, re-throw
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Generate thumbnail for any file type - always returns a thumbnail (placeholder if needed)
|
|
*/
|
|
export async function generateThumbnailForFile(file: File): Promise<string> {
|
|
// Skip very large files
|
|
if (file.size >= 100 * 1024 * 1024) {
|
|
return generatePlaceholderThumbnail(file);
|
|
}
|
|
|
|
// Handle image files - convert to data URL for persistence
|
|
if (file.type.startsWith('image/')) {
|
|
return new Promise((resolve, reject) => {
|
|
const reader = new FileReader();
|
|
reader.onload = () => resolve(reader.result as string);
|
|
reader.onerror = () => reject(reader.error);
|
|
reader.readAsDataURL(file);
|
|
});
|
|
}
|
|
|
|
// Handle PDF files
|
|
if (file.type.startsWith('application/pdf')) {
|
|
const scale = calculateScaleFromFileSize(file.size);
|
|
|
|
// Only read first 2MB for thumbnail generation to save memory
|
|
const chunkSize = 2 * 1024 * 1024; // 2MB
|
|
const chunk = file.slice(0, Math.min(chunkSize, file.size));
|
|
const arrayBuffer = await chunk.arrayBuffer();
|
|
|
|
try {
|
|
return await generatePDFThumbnail(arrayBuffer, file, scale);
|
|
} catch (error) {
|
|
if (error instanceof Error && error.name === 'InvalidPDFException') {
|
|
console.warn(`PDF structure issue for ${file.name} - trying with full file`);
|
|
try {
|
|
// Try with full file instead of chunk
|
|
const fullArrayBuffer = await file.arrayBuffer();
|
|
return await generatePDFThumbnail(fullArrayBuffer, file, scale);
|
|
} catch (fullFileError) {
|
|
console.warn(`Full file PDF processing also failed for ${file.name} - using placeholder`);
|
|
return generatePlaceholderThumbnail(file);
|
|
}
|
|
}
|
|
console.warn(`PDF processing failed for ${file.name} - using placeholder:`, error);
|
|
return generatePlaceholderThumbnail(file);
|
|
}
|
|
}
|
|
|
|
// All other files get placeholder
|
|
return generatePlaceholderThumbnail(file);
|
|
}
|
|
|
|
/**
|
|
* Generate thumbnail and extract page count for a PDF file - always returns a valid thumbnail
|
|
*/
|
|
export async function generateThumbnailWithMetadata(file: File): Promise<ThumbnailWithMetadata> {
|
|
// Non-PDF files have no page count
|
|
if (!file.type.startsWith('application/pdf')) {
|
|
const thumbnail = await generateThumbnailForFile(file);
|
|
return { thumbnail, pageCount: 0 };
|
|
}
|
|
|
|
// Skip very large files
|
|
if (file.size >= 100 * 1024 * 1024) {
|
|
const thumbnail = generatePlaceholderThumbnail(file);
|
|
return { thumbnail, pageCount: 1 };
|
|
}
|
|
|
|
const scale = calculateScaleFromFileSize(file.size);
|
|
|
|
try {
|
|
const arrayBuffer = await file.arrayBuffer();
|
|
const pdf = await pdfWorkerManager.createDocument(arrayBuffer);
|
|
|
|
const pageCount = pdf.numPages;
|
|
const page = await pdf.getPage(1);
|
|
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) {
|
|
pdfWorkerManager.destroyDocument(pdf);
|
|
throw new Error('Could not get canvas context');
|
|
}
|
|
|
|
await page.render({ canvasContext: context, viewport }).promise;
|
|
const thumbnail = canvas.toDataURL();
|
|
|
|
pdfWorkerManager.destroyDocument(pdf);
|
|
return { thumbnail, pageCount };
|
|
|
|
} catch (error) {
|
|
if (error instanceof Error && error.name === "PasswordException") {
|
|
// Handle encrypted PDFs
|
|
const thumbnail = generateEncryptedPDFThumbnail(file);
|
|
return { thumbnail, pageCount: 1 };
|
|
}
|
|
|
|
const thumbnail = generatePlaceholderThumbnail(file);
|
|
return { thumbnail, pageCount: 1 };
|
|
}
|
|
} |