mirror of
https://github.com/Stirling-Tools/Stirling-PDF.git
synced 2025-08-22 04:09:22 +00: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>
364 lines
10 KiB
TypeScript
364 lines
10 KiB
TypeScript
import React, { useState, useCallback, useRef, useEffect } from 'react';
|
|
import { Text, Checkbox, Tooltip, ActionIcon, Badge } from '@mantine/core';
|
|
import { useTranslation } from 'react-i18next';
|
|
import CloseIcon from '@mui/icons-material/Close';
|
|
import PushPinIcon from '@mui/icons-material/PushPin';
|
|
import PushPinOutlinedIcon from '@mui/icons-material/PushPinOutlined';
|
|
import DragIndicatorIcon from '@mui/icons-material/DragIndicator';
|
|
import { draggable, dropTargetForElements } from '@atlaskit/pragmatic-drag-and-drop/element/adapter';
|
|
import styles from './PageEditor.module.css';
|
|
import { useFileContext } from '../../contexts/FileContext';
|
|
|
|
interface FileItem {
|
|
id: string;
|
|
name: string;
|
|
pageCount: number;
|
|
thumbnail: string;
|
|
size: number;
|
|
splitBefore?: boolean;
|
|
}
|
|
|
|
interface FileThumbnailProps {
|
|
file: FileItem;
|
|
index: number;
|
|
totalFiles: number;
|
|
selectedFiles: string[];
|
|
selectionMode: boolean;
|
|
onToggleFile: (fileId: string) => void;
|
|
onDeleteFile: (fileId: string) => void;
|
|
onViewFile: (fileId: string) => void;
|
|
onSetStatus: (status: string) => void;
|
|
onReorderFiles?: (sourceFileId: string, targetFileId: string, selectedFileIds: string[]) => void;
|
|
toolMode?: boolean;
|
|
isSupported?: boolean;
|
|
}
|
|
|
|
const FileThumbnail = ({
|
|
file,
|
|
index,
|
|
totalFiles,
|
|
selectedFiles,
|
|
selectionMode,
|
|
onToggleFile,
|
|
onDeleteFile,
|
|
onViewFile,
|
|
onSetStatus,
|
|
onReorderFiles,
|
|
toolMode = false,
|
|
isSupported = true,
|
|
}: FileThumbnailProps) => {
|
|
const { t } = useTranslation();
|
|
const { pinnedFiles, pinFile, unpinFile, isFilePinned, activeFiles } = useFileContext();
|
|
|
|
// Drag and drop state
|
|
const [isDragging, setIsDragging] = useState(false);
|
|
const dragElementRef = useRef<HTMLDivElement | null>(null);
|
|
|
|
// Find the actual File object that corresponds to this FileItem
|
|
const actualFile = activeFiles.find(f => f.name === file.name && f.size === file.size);
|
|
|
|
const formatFileSize = (bytes: number) => {
|
|
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];
|
|
};
|
|
|
|
// Setup drag and drop using @atlaskit/pragmatic-drag-and-drop
|
|
const fileElementRef = useCallback((element: HTMLDivElement | null) => {
|
|
if (!element) return;
|
|
|
|
dragElementRef.current = element;
|
|
|
|
const dragCleanup = draggable({
|
|
element,
|
|
getInitialData: () => ({
|
|
type: 'file',
|
|
fileId: file.id,
|
|
fileName: file.name,
|
|
selectedFiles: [file.id] // Always drag only this file, ignore selection state
|
|
}),
|
|
onDragStart: () => {
|
|
setIsDragging(true);
|
|
},
|
|
onDrop: () => {
|
|
setIsDragging(false);
|
|
}
|
|
});
|
|
|
|
const dropCleanup = dropTargetForElements({
|
|
element,
|
|
getData: () => ({
|
|
type: 'file',
|
|
fileId: file.id
|
|
}),
|
|
canDrop: ({ source }) => {
|
|
const sourceData = source.data;
|
|
return sourceData.type === 'file' && sourceData.fileId !== file.id;
|
|
},
|
|
onDrop: ({ source }) => {
|
|
const sourceData = source.data;
|
|
if (sourceData.type === 'file' && onReorderFiles) {
|
|
const sourceFileId = sourceData.fileId as string;
|
|
const selectedFileIds = sourceData.selectedFiles as string[];
|
|
onReorderFiles(sourceFileId, file.id, selectedFileIds);
|
|
}
|
|
}
|
|
});
|
|
|
|
return () => {
|
|
dragCleanup();
|
|
dropCleanup();
|
|
};
|
|
}, [file.id, file.name, selectionMode, selectedFiles, onReorderFiles]);
|
|
|
|
return (
|
|
<div
|
|
ref={fileElementRef}
|
|
data-file-id={file.id}
|
|
data-testid="file-thumbnail"
|
|
className={`
|
|
${styles.pageContainer}
|
|
!rounded-lg
|
|
cursor-grab
|
|
select-none
|
|
w-[20rem]
|
|
h-[24rem]
|
|
flex flex-col items-center justify-center
|
|
flex-shrink-0
|
|
shadow-sm
|
|
hover:shadow-md
|
|
transition-all
|
|
relative
|
|
${selectionMode
|
|
? 'bg-white hover:bg-gray-50'
|
|
: 'bg-white hover:bg-gray-50'}
|
|
${isDragging ? 'opacity-50 scale-95' : ''}
|
|
`}
|
|
style={{
|
|
opacity: isSupported ? (isDragging ? 0.5 : 1) : 0.5,
|
|
filter: isSupported ? 'none' : 'grayscale(50%)'
|
|
}}
|
|
>
|
|
{selectionMode && (
|
|
<div
|
|
className={styles.checkboxContainer}
|
|
data-testid="file-thumbnail-checkbox"
|
|
style={{
|
|
position: 'absolute',
|
|
top: 8,
|
|
right: 8,
|
|
zIndex: 4,
|
|
backgroundColor: 'white',
|
|
borderRadius: '4px',
|
|
padding: '2px',
|
|
boxShadow: '0 2px 4px rgba(0,0,0,0.1)',
|
|
pointerEvents: 'auto'
|
|
}}
|
|
onMouseDown={(e) => e.stopPropagation()}
|
|
onDragStart={(e) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
}}
|
|
>
|
|
<Checkbox
|
|
checked={selectedFiles.includes(file.id)}
|
|
onChange={(event) => {
|
|
event.stopPropagation();
|
|
if (isSupported) {
|
|
onToggleFile(file.id);
|
|
}
|
|
}}
|
|
onClick={(e) => e.stopPropagation()}
|
|
disabled={!isSupported}
|
|
size="sm"
|
|
/>
|
|
</div>
|
|
)}
|
|
|
|
{/* File content area */}
|
|
<div className="file-container w-[90%] h-[80%] relative">
|
|
{/* Stacked file effect - multiple shadows to simulate pages */}
|
|
<div
|
|
style={{
|
|
width: '100%',
|
|
height: '100%',
|
|
backgroundColor: 'var(--mantine-color-gray-1)',
|
|
borderRadius: 6,
|
|
border: '1px solid var(--mantine-color-gray-3)',
|
|
padding: 4,
|
|
display: 'flex',
|
|
alignItems: 'center',
|
|
justifyContent: 'center',
|
|
position: 'relative',
|
|
boxShadow: '2px 2px 0 rgba(0,0,0,0.1), 4px 4px 0 rgba(0,0,0,0.05)'
|
|
}}
|
|
>
|
|
<img
|
|
src={file.thumbnail}
|
|
alt={file.name}
|
|
draggable={false}
|
|
onError={(e) => {
|
|
// Hide broken image if blob URL was revoked
|
|
const img = e.target as HTMLImageElement;
|
|
img.style.display = 'none';
|
|
}}
|
|
style={{
|
|
maxWidth: '100%',
|
|
maxHeight: '100%',
|
|
objectFit: 'contain',
|
|
borderRadius: 2,
|
|
}}
|
|
/>
|
|
</div>
|
|
|
|
{/* Page count badge - only show for PDFs */}
|
|
{file.pageCount > 0 && (
|
|
<Badge
|
|
size="sm"
|
|
variant="filled"
|
|
color="blue"
|
|
style={{
|
|
position: 'absolute',
|
|
top: 8,
|
|
left: 8,
|
|
zIndex: 3,
|
|
}}
|
|
>
|
|
{file.pageCount} {file.pageCount === 1 ? 'page' : 'pages'}
|
|
</Badge>
|
|
)}
|
|
|
|
{/* Unsupported badge */}
|
|
{!isSupported && (
|
|
<Badge
|
|
size="sm"
|
|
variant="filled"
|
|
color="orange"
|
|
style={{
|
|
position: 'absolute',
|
|
top: 8,
|
|
right: selectionMode ? 48 : 8, // Avoid overlap with checkbox
|
|
zIndex: 3,
|
|
}}
|
|
>
|
|
{t("fileManager.unsupported", "Unsupported")}
|
|
</Badge>
|
|
)}
|
|
|
|
{/* File name overlay */}
|
|
<Text
|
|
className={styles.pageNumber}
|
|
size="xs"
|
|
fw={500}
|
|
c="white"
|
|
style={{
|
|
position: 'absolute',
|
|
bottom: 5,
|
|
left: 5,
|
|
right: 5,
|
|
background: 'rgba(0, 0, 0, 0.8)',
|
|
padding: '4px 6px',
|
|
borderRadius: 4,
|
|
zIndex: 2,
|
|
opacity: 0,
|
|
transition: 'opacity 0.2s ease-in-out',
|
|
textOverflow: 'ellipsis',
|
|
overflow: 'hidden',
|
|
whiteSpace: 'nowrap'
|
|
}}
|
|
>
|
|
{file.name}
|
|
</Text>
|
|
|
|
{/* Hover controls */}
|
|
<div
|
|
className={styles.pageHoverControls}
|
|
style={{
|
|
position: 'absolute',
|
|
top: '50%',
|
|
left: '50%',
|
|
transform: 'translate(-50%, -50%)',
|
|
background: 'rgba(0, 0, 0, 0.8)',
|
|
padding: '8px 12px',
|
|
borderRadius: 20,
|
|
opacity: 0,
|
|
transition: 'opacity 0.2s ease-in-out',
|
|
zIndex: 3,
|
|
display: 'flex',
|
|
gap: '8px',
|
|
alignItems: 'center',
|
|
whiteSpace: 'nowrap'
|
|
}}
|
|
>
|
|
|
|
{actualFile && (
|
|
<Tooltip label={isFilePinned(actualFile) ? "Unpin File" : "Pin File"}>
|
|
<ActionIcon
|
|
size="md"
|
|
variant="subtle"
|
|
c={isFilePinned(actualFile) ? "yellow" : "white"}
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
if (isFilePinned(actualFile)) {
|
|
unpinFile(actualFile);
|
|
onSetStatus(`Unpinned ${file.name}`);
|
|
} else {
|
|
pinFile(actualFile);
|
|
onSetStatus(`Pinned ${file.name}`);
|
|
}
|
|
}}
|
|
>
|
|
{isFilePinned(actualFile) ? (
|
|
<PushPinIcon style={{ fontSize: 20 }} />
|
|
) : (
|
|
<PushPinOutlinedIcon style={{ fontSize: 20 }} />
|
|
)}
|
|
</ActionIcon>
|
|
</Tooltip>
|
|
)}
|
|
|
|
<Tooltip label="Close File">
|
|
<ActionIcon
|
|
size="md"
|
|
variant="subtle"
|
|
c="orange"
|
|
onClick={(e) => {
|
|
e.stopPropagation();
|
|
onDeleteFile(file.id);
|
|
onSetStatus(`Closed ${file.name}`);
|
|
}}
|
|
>
|
|
<CloseIcon style={{ fontSize: 20 }} />
|
|
</ActionIcon>
|
|
</Tooltip>
|
|
</div>
|
|
|
|
<DragIndicatorIcon
|
|
style={{
|
|
position: 'absolute',
|
|
bottom: 4,
|
|
right: 4,
|
|
color: 'rgba(0,0,0,0.3)',
|
|
fontSize: 16,
|
|
zIndex: 1
|
|
}}
|
|
/>
|
|
</div>
|
|
|
|
{/* File info */}
|
|
<div className="w-full px-4 py-2 text-center">
|
|
<Text size="sm" fw={500} truncate>
|
|
{file.name}
|
|
</Text>
|
|
<Text size="xs" c="dimmed">
|
|
{formatFileSize(file.size)}
|
|
</Text>
|
|
</div>
|
|
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default FileThumbnail; |