Files
pdf-tools/src/components/ReorderPanel.tsx
2025-11-26 18:02:58 +01:00

226 lines
6.4 KiB
TypeScript

import React, { useEffect, useState } from 'react';
interface ReorderPanelProps {
pageCount: number;
thumbnails: string[] | null;
isBusy: boolean;
hasPdf: boolean;
rotations: Record<number, number>;
onRotate: (pageIndex: number) => void;
onExportReordered: (order: number[]) => void;
reorderDownloadUrl: string | null;
reorderFilename: string | null;
}
const ReorderPanel: React.FC<ReorderPanelProps> = ({
pageCount,
thumbnails,
isBusy,
hasPdf,
rotations,
onRotate,
onExportReordered,
reorderDownloadUrl,
reorderFilename,
}) => {
const [order, setOrder] = useState<number[]>([]);
const [draggingIndex, setDraggingIndex] = useState<number | null>(null);
useEffect(() => {
if (pageCount > 0) {
setOrder(Array.from({ length: pageCount }, (_, i) => i));
} else {
setOrder([]);
}
}, [pageCount]);
const handleDragStart = (index: number) => (e: React.DragEvent) => {
setDraggingIndex(index);
e.dataTransfer.effectAllowed = 'move';
};
const handleDragOver = (index: number) => (e: React.DragEvent) => {
e.preventDefault();
e.dataTransfer.dropEffect = 'move';
};
const handleDrop = (index: number) => (e: React.DragEvent) => {
e.preventDefault();
if (draggingIndex === null || draggingIndex === index) return;
setOrder((prev) => {
const updated = [...prev];
const [moved] = updated.splice(draggingIndex, 1);
updated.splice(index, 0, moved);
return updated;
});
setDraggingIndex(null);
};
const handleDragEnd = () => {
setDraggingIndex(null);
};
const handleDelete = (visualIndex: number) => () => {
setOrder((prev) => prev.filter((_, idx) => idx !== visualIndex));
};
const handleRotateClick = (pageIndex: number) => () => {
onRotate(pageIndex);
};
const handleExport = () => {
if (!hasPdf || order.length === 0) return;
onExportReordered(order);
};
if (!hasPdf) {
return (
<div className="card">
<h2>Reorder pages</h2>
<p>Load a PDF first to reorder, delete, or rotate its pages.</p>
</div>
);
}
return (
<div className="card">
<h2>Reorder / delete / rotate</h2>
<p>
Drag pages to reorder them. Use rotate and delete controls below each
thumbnail. All changes stay in memory until you export a new PDF.
</p>
<div
style={{
display: 'flex',
flexWrap: 'wrap',
gap: '0.5rem',
marginBottom: '0.75rem',
}}
>
{order.map((pageIndex, visualIndex) => {
const thumb = thumbnails?.[pageIndex];
const isDragging = visualIndex === draggingIndex;
const rotation = rotations[pageIndex] ?? 0;
return (
<div
key={`${pageIndex}-${visualIndex}`}
draggable
onDragStart={handleDragStart(visualIndex)}
onDragOver={handleDragOver(visualIndex)}
onDrop={handleDrop(visualIndex)}
onDragEnd={handleDragEnd}
style={{
width: '130px',
padding: '0.4rem',
borderRadius: '0.5rem',
border: isDragging ? '2px solid #2563eb' : '1px solid #e5e7eb',
background: isDragging ? '#dbeafe' : '#f9fafb',
display: 'flex',
flexDirection: 'column',
alignItems: 'center',
gap: '0.25rem',
cursor: 'grab',
}}
>
{thumb ? (
<img
src={thumb}
alt={`Page ${pageIndex + 1}`}
style={{
maxHeight: '90px',
width: 'auto',
borderRadius: '0.25rem',
border: '1px solid #e5e7eb',
background: 'white',
}}
/>
) : (
<div
style={{
width: '60px',
height: '80px',
borderRadius: '0.25rem',
border: '1px dashed #d1d5db',
background: '#f3f4f6',
}}
/>
)}
<span style={{ fontSize: '0.8rem' }}>Page {pageIndex + 1}</span>
<span style={{ fontSize: '0.7rem', color: '#6b7280' }}>
Pos {visualIndex + 1} · Rot {rotation}°
</span>
<div
style={{
display: 'flex',
gap: '0.25rem',
marginTop: '0.25rem',
}}
>
<button
type="button"
onClick={handleRotateClick(pageIndex)}
style={{
border: 'none',
borderRadius: '999px',
padding: '0.15rem 0.4rem',
fontSize: '0.75rem',
background: '#e5e7eb',
cursor: 'pointer',
}}
>
90°
</button>
<button
type="button"
onClick={handleDelete(visualIndex)}
style={{
border: 'none',
borderRadius: '999px',
padding: '0.15rem 0.4rem',
fontSize: '0.75rem',
background: '#fecaca',
color: '#b91c1c',
cursor: 'pointer',
}}
title="Remove this page from the exported PDF"
>
</button>
</div>
</div>
);
})}
</div>
<div className="button-row">
<button
className="primary"
disabled={!hasPdf || isBusy || order.length === 0}
onClick={handleExport}
>
{isBusy ? 'Exporting…' : 'Export reordered PDF'}
</button>
</div>
{reorderDownloadUrl && reorderFilename && (
<div style={{ marginTop: '0.5rem', fontSize: '0.9rem' }}>
<strong>Reordered result:</strong>{' '}
<a
href={reorderDownloadUrl}
download={reorderFilename}
className="download-link"
>
Download {reorderFilename}
</a>
</div>
)}
</div>
);
};
export default ReorderPanel;