mirror of
https://github.com/Stirling-Tools/Stirling-PDF.git
synced 2025-08-02 18:45:21 +00:00

🔄 Dynamic Processing Strategies - Adaptive routing: Same tool uses different backend endpoints based on file analysis - Combined vs separate processing: Intelligently chooses between merge operations and individual file processing - Cross-format workflows: Enable complex conversions like "mixed files → PDF" that other tools can't handle ⚙️ Format-Specific Intelligence Each conversion type gets tailored options: - HTML/ZIP → PDF: Zoom controls (0.1-3.0 increments) with live preview - Email → PDF: Attachment handling, size limits, recipient control - PDF → PDF/A: Digital signature detection with warnings - Images → PDF: Smart combining vs individual file options File Architecture Core Implementation: ├── Convert.tsx # Main stepped workflow UI ├── ConvertSettings.tsx # Centralized settings with smart detection ├── GroupedFormatDropdown.tsx # Enhanced format selector with grouping ├── useConvertParameters.ts # Smart detection & parameter management ├── useConvertOperation.ts # Multi-strategy processing logic └── Settings Components: ├── ConvertFromWebSettings.tsx # HTML zoom controls ├── ConvertFromEmailSettings.tsx # Email attachment options ├── ConvertToPdfaSettings.tsx # PDF/A with signature detection ├── ConvertFromImageSettings.tsx # Image PDF options └── ConvertToImageSettings.tsx # PDF to image options Utility Layer Utils & Services: ├── convertUtils.ts # Format detection & endpoint routing ├── fileResponseUtils.ts # Generic API response handling └── setupTests.ts # Enhanced test environment with crypto mocks Testing & Quality Comprehensive Test Coverage Test Suite: ├── useConvertParameters.test.ts # Parameter logic & smart detection ├── useConvertParametersAutoDetection.test.ts # File type analysis ├── ConvertIntegration.test.tsx # End-to-end conversion workflows ├── ConvertSmartDetectionIntegration.test.tsx # Mixed file scenarios ├── ConvertE2E.spec.ts # Playwright browser tests ├── convertUtils.test.ts # Utility function validation └── fileResponseUtils.test.ts # API response handling Advanced Test Features - Crypto API mocking: Proper test environment for file hashing - File.arrayBuffer() polyfills: Complete browser API simulation - Multi-file scenario testing: Complex batch processing validation - CI/CD integration: Vitest runs in GitHub Actions with proper artifacts --------- Co-authored-by: Connor Yoh <connor@stirlingpdf.com> Co-authored-by: Anthony Stirling <77850077+Frooodle@users.noreply.github.com>
149 lines
6.3 KiB
TypeScript
149 lines
6.3 KiB
TypeScript
|
|
export const COLOR_TYPES = {
|
|
COLOR: 'color',
|
|
GREYSCALE: 'greyscale',
|
|
BLACK_WHITE: 'blackwhite'
|
|
} as const;
|
|
|
|
export const OUTPUT_OPTIONS = {
|
|
SINGLE: 'single',
|
|
MULTIPLE: 'multiple'
|
|
} as const;
|
|
|
|
export const FIT_OPTIONS = {
|
|
FIT_PAGE: 'fitDocumentToPage',
|
|
MAINTAIN_ASPECT: 'maintainAspectRatio',
|
|
FILL_PAGE: 'fillPage'
|
|
} as const;
|
|
|
|
|
|
export const CONVERSION_ENDPOINTS = {
|
|
'office-pdf': '/api/v1/convert/file/pdf',
|
|
'pdf-image': '/api/v1/convert/pdf/img',
|
|
'image-pdf': '/api/v1/convert/img/pdf',
|
|
'pdf-office-word': '/api/v1/convert/pdf/word',
|
|
'pdf-office-presentation': '/api/v1/convert/pdf/presentation',
|
|
'pdf-office-text': '/api/v1/convert/pdf/text',
|
|
'pdf-csv': '/api/v1/convert/pdf/csv',
|
|
'pdf-markdown': '/api/v1/convert/pdf/markdown',
|
|
'pdf-html': '/api/v1/convert/pdf/html',
|
|
'pdf-xml': '/api/v1/convert/pdf/xml',
|
|
'pdf-pdfa': '/api/v1/convert/pdf/pdfa',
|
|
'html-pdf': '/api/v1/convert/html/pdf',
|
|
'markdown-pdf': '/api/v1/convert/markdown/pdf',
|
|
'eml-pdf': '/api/v1/convert/eml/pdf'
|
|
} as const;
|
|
|
|
export const ENDPOINT_NAMES = {
|
|
'office-pdf': 'file-to-pdf',
|
|
'pdf-image': 'pdf-to-img',
|
|
'image-pdf': 'img-to-pdf',
|
|
'pdf-office-word': 'pdf-to-word',
|
|
'pdf-office-presentation': 'pdf-to-presentation',
|
|
'pdf-office-text': 'pdf-to-text',
|
|
'pdf-csv': 'pdf-to-csv',
|
|
'pdf-markdown': 'pdf-to-markdown',
|
|
'pdf-html': 'pdf-to-html',
|
|
'pdf-xml': 'pdf-to-xml',
|
|
'pdf-pdfa': 'pdf-to-pdfa',
|
|
'html-pdf': 'html-to-pdf',
|
|
'markdown-pdf': 'markdown-to-pdf',
|
|
'eml-pdf': 'eml-to-pdf'
|
|
} as const;
|
|
|
|
|
|
// Grouped file extensions for dropdowns
|
|
export const FROM_FORMAT_OPTIONS = [
|
|
{ value: 'any', label: 'Any', group: 'Multiple Files' },
|
|
{ value: 'image', label: 'Images', group: 'Multiple Files' },
|
|
{ value: 'pdf', label: 'PDF', group: 'Document' },
|
|
{ value: 'docx', label: 'DOCX', group: 'Document' },
|
|
{ value: 'doc', label: 'DOC', group: 'Document' },
|
|
{ value: 'odt', label: 'ODT', group: 'Document' },
|
|
{ value: 'xlsx', label: 'XLSX', group: 'Spreadsheet' },
|
|
{ value: 'xls', label: 'XLS', group: 'Spreadsheet' },
|
|
{ value: 'ods', label: 'ODS', group: 'Spreadsheet' },
|
|
{ value: 'pptx', label: 'PPTX', group: 'Presentation' },
|
|
{ value: 'ppt', label: 'PPT', group: 'Presentation' },
|
|
{ value: 'odp', label: 'ODP', group: 'Presentation' },
|
|
{ value: 'jpg', label: 'JPG', group: 'Image' },
|
|
{ value: 'jpeg', label: 'JPEG', group: 'Image' },
|
|
{ value: 'png', label: 'PNG', group: 'Image' },
|
|
{ value: 'gif', label: 'GIF', group: 'Image' },
|
|
{ value: 'bmp', label: 'BMP', group: 'Image' },
|
|
{ value: 'tiff', label: 'TIFF', group: 'Image' },
|
|
{ value: 'webp', label: 'WEBP', group: 'Image' },
|
|
{ value: 'svg', label: 'SVG', group: 'Image' },
|
|
{ value: 'html', label: 'HTML', group: 'Web' },
|
|
{ value: 'zip', label: 'ZIP', group: 'Web' },
|
|
{ value: 'md', label: 'MD', group: 'Text' },
|
|
{ value: 'txt', label: 'TXT', group: 'Text' },
|
|
{ value: 'rtf', label: 'RTF', group: 'Text' },
|
|
{ value: 'eml', label: 'EML', group: 'Email' },
|
|
];
|
|
|
|
export const TO_FORMAT_OPTIONS = [
|
|
{ value: 'pdf', label: 'PDF', group: 'Document' },
|
|
{ value: 'pdfa', label: 'PDF/A', group: 'Document' },
|
|
{ value: 'docx', label: 'DOCX', group: 'Document' },
|
|
{ value: 'odt', label: 'ODT', group: 'Document' },
|
|
{ value: 'csv', label: 'CSV', group: 'Spreadsheet' },
|
|
{ value: 'pptx', label: 'PPTX', group: 'Presentation' },
|
|
{ value: 'odp', label: 'ODP', group: 'Presentation' },
|
|
{ value: 'txt', label: 'TXT', group: 'Text' },
|
|
{ value: 'rtf', label: 'RTF', group: 'Text' },
|
|
{ value: 'md', label: 'MD', group: 'Text' },
|
|
{ value: 'png', label: 'PNG', group: 'Image' },
|
|
{ value: 'jpg', label: 'JPG', group: 'Image' },
|
|
{ value: 'gif', label: 'GIF', group: 'Image' },
|
|
{ value: 'tiff', label: 'TIFF', group: 'Image' },
|
|
{ value: 'bmp', label: 'BMP', group: 'Image' },
|
|
{ value: 'webp', label: 'WEBP', group: 'Image' },
|
|
{ value: 'html', label: 'HTML', group: 'Web' },
|
|
{ value: 'xml', label: 'XML', group: 'Web' },
|
|
];
|
|
|
|
// Conversion matrix - what each source format can convert to
|
|
export const CONVERSION_MATRIX: Record<string, string[]> = {
|
|
'any': ['pdf'], // Mixed files always convert to PDF
|
|
'image': ['pdf'], // Multiple images always convert to PDF
|
|
'pdf': ['png', 'jpg', 'gif', 'tiff', 'bmp', 'webp', 'docx', 'odt', 'pptx', 'odp', 'csv', 'txt', 'rtf', 'md', 'html', 'xml', 'pdfa'],
|
|
'docx': ['pdf'], 'doc': ['pdf'], 'odt': ['pdf'],
|
|
'xlsx': ['pdf'], 'xls': ['pdf'], 'ods': ['pdf'],
|
|
'pptx': ['pdf'], 'ppt': ['pdf'], 'odp': ['pdf'],
|
|
'jpg': ['pdf'], 'jpeg': ['pdf'], 'png': ['pdf'], 'gif': ['pdf'], 'bmp': ['pdf'], 'tiff': ['pdf'], 'webp': ['pdf'], 'svg': ['pdf'],
|
|
'html': ['pdf'],
|
|
'zip': ['pdf'],
|
|
'md': ['pdf'],
|
|
'txt': ['pdf'], 'rtf': ['pdf'],
|
|
'eml': ['pdf']
|
|
};
|
|
|
|
// Map extensions to endpoint keys
|
|
export const EXTENSION_TO_ENDPOINT: Record<string, Record<string, string>> = {
|
|
'any': { 'pdf': 'file-to-pdf' }, // Mixed files use file-to-pdf endpoint
|
|
'image': { 'pdf': 'img-to-pdf' }, // Multiple images use img-to-pdf endpoint
|
|
'pdf': {
|
|
'png': 'pdf-to-img', 'jpg': 'pdf-to-img', 'gif': 'pdf-to-img', 'tiff': 'pdf-to-img', 'bmp': 'pdf-to-img', 'webp': 'pdf-to-img',
|
|
'docx': 'pdf-to-word', 'odt': 'pdf-to-word',
|
|
'pptx': 'pdf-to-presentation', 'odp': 'pdf-to-presentation',
|
|
'csv': 'pdf-to-csv',
|
|
'txt': 'pdf-to-text', 'rtf': 'pdf-to-text', 'md': 'pdf-to-markdown',
|
|
'html': 'pdf-to-html', 'xml': 'pdf-to-xml',
|
|
'pdfa': 'pdf-to-pdfa'
|
|
},
|
|
'docx': { 'pdf': 'file-to-pdf' }, 'doc': { 'pdf': 'file-to-pdf' }, 'odt': { 'pdf': 'file-to-pdf' },
|
|
'xlsx': { 'pdf': 'file-to-pdf' }, 'xls': { 'pdf': 'file-to-pdf' }, 'ods': { 'pdf': 'file-to-pdf' },
|
|
'pptx': { 'pdf': 'file-to-pdf' }, 'ppt': { 'pdf': 'file-to-pdf' }, 'odp': { 'pdf': 'file-to-pdf' },
|
|
'jpg': { 'pdf': 'img-to-pdf' }, 'jpeg': { 'pdf': 'img-to-pdf' }, 'png': { 'pdf': 'img-to-pdf' },
|
|
'gif': { 'pdf': 'img-to-pdf' }, 'bmp': { 'pdf': 'img-to-pdf' }, 'tiff': { 'pdf': 'img-to-pdf' }, 'webp': { 'pdf': 'img-to-pdf' }, 'svg': { 'pdf': 'img-to-pdf' },
|
|
'html': { 'pdf': 'html-to-pdf' },
|
|
'zip': { 'pdf': 'html-to-pdf' },
|
|
'md': { 'pdf': 'markdown-to-pdf' },
|
|
'txt': { 'pdf': 'file-to-pdf' }, 'rtf': { 'pdf': 'file-to-pdf' },
|
|
'eml': { 'pdf': 'eml-to-pdf' }
|
|
};
|
|
|
|
export type ColorType = typeof COLOR_TYPES[keyof typeof COLOR_TYPES];
|
|
export type OutputOption = typeof OUTPUT_OPTIONS[keyof typeof OUTPUT_OPTIONS];
|
|
export type FitOption = typeof FIT_OPTIONS[keyof typeof FIT_OPTIONS]; |