0a9588abb7
- Add pagination support to findAll (page, limit query params) - Add findByTemplateId method to service - Add GET /by-template/:templateId endpoint to controller - Service already includes CRUD for QuestionBank and QuestionBankItem
136 lines
5.3 KiB
TypeScript
136 lines
5.3 KiB
TypeScript
import React, { useCallback, useState } from 'react';
|
|
import { Upload as UploadIcon, FileText, Image as ImageIcon, Folder, FileUp, ShieldCheck } from 'lucide-react';
|
|
import { useLanguage } from '../contexts/LanguageContext';
|
|
import { motion, AnimatePresence } from 'framer-motion';
|
|
|
|
interface DragDropUploadProps {
|
|
onFilesSelected: (files: FileList) => void;
|
|
isAdmin: boolean;
|
|
globalMode?: boolean;
|
|
}
|
|
|
|
export const DragDropUpload: React.FC<DragDropUploadProps> = ({ onFilesSelected, isAdmin, globalMode = false }) => {
|
|
const { t } = useLanguage();
|
|
const [isDragging, setIsDragging] = useState(false);
|
|
|
|
const handleDragEnter = useCallback((e: React.DragEvent) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
if (e.dataTransfer.items && e.dataTransfer.items.length > 0) {
|
|
setIsDragging(true);
|
|
}
|
|
}, []);
|
|
|
|
const handleDragLeave = useCallback((e: React.DragEvent) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
setIsDragging(false);
|
|
}, []);
|
|
|
|
const handleDragOver = useCallback((e: React.DragEvent) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
e.dataTransfer.dropEffect = 'copy';
|
|
}, []);
|
|
|
|
const handleDrop = useCallback((e: React.DragEvent) => {
|
|
e.preventDefault();
|
|
e.stopPropagation();
|
|
setIsDragging(false);
|
|
|
|
if (e.dataTransfer.files && e.dataTransfer.files.length > 0) {
|
|
onFilesSelected(e.dataTransfer.files);
|
|
e.dataTransfer.clearData();
|
|
}
|
|
}, [onFilesSelected]);
|
|
|
|
const handleFileInput = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
if (e.target.files && e.target.files.length > 0) {
|
|
onFilesSelected(e.target.files);
|
|
e.target.value = '';
|
|
}
|
|
};
|
|
|
|
if (!isAdmin) return null;
|
|
|
|
return (
|
|
<motion.div
|
|
initial={{ opacity: 0, scale: 0.98 }}
|
|
animate={{ opacity: 1, scale: 1 }}
|
|
className={`relative w-full overflow-hidden rounded-2xl border-2 border-dashed transition-all duration-300 ${isDragging
|
|
? 'border-blue-500 bg-blue-50/50 shadow-[0_0_25px_rgba(59,130,246,0.1)]'
|
|
: 'border-slate-200 bg-slate-50/50 hover:border-slate-300 hover:bg-slate-50'
|
|
}`}
|
|
onDragEnter={handleDragEnter}
|
|
onDragOver={handleDragOver}
|
|
onDragLeave={handleDragLeave}
|
|
onDrop={handleDrop}
|
|
onClick={() => document.getElementById('file-upload-input')?.click()}
|
|
>
|
|
<div className="flex flex-col items-center justify-center py-12 px-6 text-center cursor-pointer">
|
|
<div className={`p-4 rounded-2xl mb-6 transition-all duration-300 ${isDragging ? 'bg-blue-600 text-white scale-110' : 'bg-white text-blue-600 shadow-sm border border-slate-100'}`}>
|
|
<FileUp size={32} />
|
|
</div>
|
|
|
|
<div className="space-y-1 mb-8">
|
|
<h3 className="text-lg font-bold text-slate-900 tracking-tight">
|
|
{t('dragDropUploadTitle')}
|
|
</h3>
|
|
<p className="text-sm text-slate-500 font-medium">
|
|
{t('dragDropUploadDesc')}
|
|
</p>
|
|
</div>
|
|
|
|
<div className="flex flex-wrap items-center justify-center gap-4 mb-8">
|
|
<div className="flex items-center gap-2 px-3 py-1.5 bg-white border border-slate-100 rounded-full text-[11px] font-bold text-slate-500 uppercase tracking-wider shadow-sm">
|
|
<ShieldCheck size={14} className="text-emerald-500" />
|
|
<span>{t('secureIngestion')}</span>
|
|
</div>
|
|
<div className="flex items-center gap-2 px-3 py-1.5 bg-white border border-slate-100 rounded-full text-[11px] font-bold text-slate-500 uppercase tracking-wider shadow-sm">
|
|
<FileText size={14} className="text-blue-500" />
|
|
<span>{t('documentsAndText')}</span>
|
|
</div>
|
|
<div className="flex items-center gap-2 px-3 py-1.5 bg-white border border-slate-100 rounded-full text-[11px] font-bold text-slate-500 uppercase tracking-wider shadow-sm">
|
|
<ImageIcon size={14} className="text-purple-500" />
|
|
<span>{t('imagesAndVision')}</span>
|
|
</div>
|
|
</div>
|
|
|
|
<button
|
|
type="button"
|
|
className="px-6 py-2.5 bg-blue-600 hover:bg-blue-700 text-white rounded-xl shadow-md shadow-blue-100 transition-all font-semibold text-sm active:scale-95 flex items-center gap-2"
|
|
>
|
|
<Folder size={18} />
|
|
{t('browseFiles')}
|
|
</button>
|
|
|
|
<input
|
|
type="file"
|
|
multiple
|
|
onChange={handleFileInput}
|
|
className="hidden"
|
|
id="file-upload-input"
|
|
accept=".pdf,.doc,.docx,.xls,.xlsx,.ppt,.pptx,.txt,.md,.html,.csv,.rtf,.odt,.ods,.odp,.json,.js,.jsx,.ts,.tsx,.css,.xml,image/*"
|
|
/>
|
|
</div>
|
|
|
|
<AnimatePresence>
|
|
{isDragging && (
|
|
<motion.div
|
|
initial={{ opacity: 0 }}
|
|
animate={{ opacity: 1 }}
|
|
exit={{ opacity: 0 }}
|
|
className="absolute inset-0 bg-blue-600/5 backdrop-blur-[2px] pointer-events-none flex items-center justify-center border-2 border-blue-500 rounded-2xl"
|
|
>
|
|
<div className="bg-white p-6 rounded-3xl shadow-2xl flex flex-col items-center gap-3">
|
|
<div className="w-12 h-12 bg-blue-600 text-white rounded-2xl flex items-center justify-center animate-bounce">
|
|
<FileUp size={24} />
|
|
</div>
|
|
<span className="text-blue-600 font-bold">{t('dropToIngest')}</span>
|
|
</div>
|
|
</motion.div>
|
|
)}
|
|
</AnimatePresence>
|
|
</motion.div>
|
|
);
|
|
}; |