169 lines
5.7 KiB
TypeScript
169 lines
5.7 KiB
TypeScript
"use client";
|
|
|
|
import { useState } from "react";
|
|
import { X, Trash2, Download, Loader2 } from "lucide-react";
|
|
import { toast } from "sonner";
|
|
|
|
interface DocumentPreviewModalProps {
|
|
isOpen: boolean;
|
|
onClose: () => void;
|
|
document: {
|
|
key: string;
|
|
name: string;
|
|
type: string;
|
|
size: number;
|
|
downloadUrl: string;
|
|
} | null;
|
|
onDelete: (fileKey: string) => Promise<void>;
|
|
}
|
|
|
|
export function DocumentPreviewModal({
|
|
isOpen,
|
|
onClose,
|
|
document,
|
|
onDelete
|
|
}: DocumentPreviewModalProps) {
|
|
const [showDeleteConfirm, setShowDeleteConfirm] = useState(false);
|
|
const [isDeleting, setIsDeleting] = useState(false);
|
|
|
|
if (!isOpen || !document) return null;
|
|
|
|
const isImage = document.name.match(/\.(jpg|jpeg|png)$/i);
|
|
const isPdf = document.name.match(/\.pdf$/i);
|
|
|
|
const handleDelete = async () => {
|
|
setIsDeleting(true);
|
|
try {
|
|
await onDelete(document.key);
|
|
toast.success("Document supprimé avec succès");
|
|
onClose();
|
|
} catch (error) {
|
|
toast.error("Erreur lors de la suppression du document");
|
|
console.error(error);
|
|
} finally {
|
|
setIsDeleting(false);
|
|
setShowDeleteConfirm(false);
|
|
}
|
|
};
|
|
|
|
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 Math.round(bytes / Math.pow(k, i) * 100) / 100 + ' ' + sizes[i];
|
|
};
|
|
|
|
return (
|
|
<div className="fixed inset-0 z-50 flex items-center justify-center bg-black/70 p-4">
|
|
<div className="bg-white rounded-2xl shadow-2xl max-w-4xl w-full max-h-[90vh] flex flex-col overflow-hidden">
|
|
{/* Header */}
|
|
<div className="flex items-center justify-between p-4 border-b bg-gray-50">
|
|
<div className="flex-1 min-w-0">
|
|
<h2 className="text-lg font-semibold truncate">{document.name}</h2>
|
|
<div className="flex items-center gap-3 text-sm text-gray-500 mt-1">
|
|
<span className="px-2 py-0.5 bg-blue-100 text-blue-700 rounded text-xs font-medium">
|
|
{document.type}
|
|
</span>
|
|
<span>{formatFileSize(document.size)}</span>
|
|
</div>
|
|
</div>
|
|
<button
|
|
onClick={onClose}
|
|
className="ml-4 p-2 hover:bg-gray-200 rounded-lg transition-colors"
|
|
aria-label="Fermer"
|
|
>
|
|
<X className="h-5 w-5" />
|
|
</button>
|
|
</div>
|
|
|
|
{/* Content */}
|
|
<div className="flex-1 overflow-auto bg-gray-100 p-4">
|
|
{isImage && (
|
|
<div className="flex items-center justify-center h-full">
|
|
<img
|
|
src={document.downloadUrl}
|
|
alt={document.name}
|
|
className="max-w-full max-h-full object-contain rounded-lg shadow-lg"
|
|
/>
|
|
</div>
|
|
)}
|
|
{isPdf && (
|
|
<iframe
|
|
src={document.downloadUrl}
|
|
className="w-full h-full min-h-[600px] rounded-lg shadow-lg bg-white"
|
|
title={document.name}
|
|
/>
|
|
)}
|
|
{!isImage && !isPdf && (
|
|
<div className="flex items-center justify-center h-full">
|
|
<div className="text-center">
|
|
<p className="text-gray-500 mb-4">
|
|
Prévisualisation non disponible pour ce type de fichier
|
|
</p>
|
|
<a
|
|
href={document.downloadUrl}
|
|
download={document.name}
|
|
className="inline-flex items-center gap-2 px-4 py-2 bg-blue-600 text-white rounded-lg hover:bg-blue-700"
|
|
>
|
|
<Download className="h-4 w-4" />
|
|
Télécharger le fichier
|
|
</a>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* Footer with actions */}
|
|
<div className="flex items-center justify-between p-4 border-t bg-gray-50">
|
|
<a
|
|
href={document.downloadUrl}
|
|
download={document.name}
|
|
className="inline-flex items-center gap-2 px-4 py-2 text-sm font-medium text-blue-600 hover:bg-blue-50 rounded-lg transition-colors"
|
|
>
|
|
<Download className="h-4 w-4" />
|
|
Télécharger
|
|
</a>
|
|
|
|
{!showDeleteConfirm ? (
|
|
<button
|
|
onClick={() => setShowDeleteConfirm(true)}
|
|
className="inline-flex items-center gap-2 px-4 py-2 text-sm font-medium text-red-600 hover:bg-red-50 rounded-lg transition-colors"
|
|
>
|
|
<Trash2 className="h-4 w-4" />
|
|
Supprimer
|
|
</button>
|
|
) : (
|
|
<div className="flex items-center gap-2">
|
|
<span className="text-sm text-gray-600 mr-2">Confirmer la suppression ?</span>
|
|
<button
|
|
onClick={() => setShowDeleteConfirm(false)}
|
|
className="px-3 py-1.5 text-sm text-gray-600 hover:bg-gray-200 rounded-lg"
|
|
disabled={isDeleting}
|
|
>
|
|
Annuler
|
|
</button>
|
|
<button
|
|
onClick={handleDelete}
|
|
disabled={isDeleting}
|
|
className="inline-flex items-center gap-2 px-3 py-1.5 text-sm bg-red-600 text-white hover:bg-red-700 rounded-lg disabled:opacity-50 disabled:cursor-not-allowed"
|
|
>
|
|
{isDeleting ? (
|
|
<>
|
|
<Loader2 className="h-4 w-4 animate-spin" />
|
|
Suppression...
|
|
</>
|
|
) : (
|
|
<>
|
|
<Trash2 className="h-4 w-4" />
|
|
Confirmer
|
|
</>
|
|
)}
|
|
</button>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|