DragDropUpload.tsx 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136
  1. import React, { useCallback, useState } from 'react';
  2. import { Upload as UploadIcon, FileText, Image as ImageIcon, Folder, FileUp, ShieldCheck } from 'lucide-react';
  3. import { useLanguage } from '../contexts/LanguageContext';
  4. import { motion, AnimatePresence } from 'framer-motion';
  5. interface DragDropUploadProps {
  6. onFilesSelected: (files: FileList) => void;
  7. isAdmin: boolean;
  8. globalMode?: boolean;
  9. }
  10. export const DragDropUpload: React.FC<DragDropUploadProps> = ({ onFilesSelected, isAdmin, globalMode = false }) => {
  11. const { t } = useLanguage();
  12. const [isDragging, setIsDragging] = useState(false);
  13. const handleDragEnter = useCallback((e: React.DragEvent) => {
  14. e.preventDefault();
  15. e.stopPropagation();
  16. if (e.dataTransfer.items && e.dataTransfer.items.length > 0) {
  17. setIsDragging(true);
  18. }
  19. }, []);
  20. const handleDragLeave = useCallback((e: React.DragEvent) => {
  21. e.preventDefault();
  22. e.stopPropagation();
  23. setIsDragging(false);
  24. }, []);
  25. const handleDragOver = useCallback((e: React.DragEvent) => {
  26. e.preventDefault();
  27. e.stopPropagation();
  28. e.dataTransfer.dropEffect = 'copy';
  29. }, []);
  30. const handleDrop = useCallback((e: React.DragEvent) => {
  31. e.preventDefault();
  32. e.stopPropagation();
  33. setIsDragging(false);
  34. if (e.dataTransfer.files && e.dataTransfer.files.length > 0) {
  35. onFilesSelected(e.dataTransfer.files);
  36. e.dataTransfer.clearData();
  37. }
  38. }, [onFilesSelected]);
  39. const handleFileInput = (e: React.ChangeEvent<HTMLInputElement>) => {
  40. if (e.target.files && e.target.files.length > 0) {
  41. onFilesSelected(e.target.files);
  42. e.target.value = '';
  43. }
  44. };
  45. if (!isAdmin) return null;
  46. return (
  47. <motion.div
  48. initial={{ opacity: 0, scale: 0.98 }}
  49. animate={{ opacity: 1, scale: 1 }}
  50. className={`relative w-full overflow-hidden rounded-2xl border-2 border-dashed transition-all duration-300 ${isDragging
  51. ? 'border-blue-500 bg-blue-50/50 shadow-[0_0_25px_rgba(59,130,246,0.1)]'
  52. : 'border-slate-200 bg-slate-50/50 hover:border-slate-300 hover:bg-slate-50'
  53. }`}
  54. onDragEnter={handleDragEnter}
  55. onDragOver={handleDragOver}
  56. onDragLeave={handleDragLeave}
  57. onDrop={handleDrop}
  58. onClick={() => document.getElementById('file-upload-input')?.click()}
  59. >
  60. <div className="flex flex-col items-center justify-center py-12 px-6 text-center cursor-pointer">
  61. <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'}`}>
  62. <FileUp size={32} />
  63. </div>
  64. <div className="space-y-1 mb-8">
  65. <h3 className="text-lg font-bold text-slate-900 tracking-tight">
  66. {t('dragDropUploadTitle')}
  67. </h3>
  68. <p className="text-sm text-slate-500 font-medium">
  69. {t('dragDropUploadDesc')}
  70. </p>
  71. </div>
  72. <div className="flex flex-wrap items-center justify-center gap-4 mb-8">
  73. <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">
  74. <ShieldCheck size={14} className="text-emerald-500" />
  75. <span>{t('secureIngestion')}</span>
  76. </div>
  77. <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">
  78. <FileText size={14} className="text-blue-500" />
  79. <span>{t('documentsAndText')}</span>
  80. </div>
  81. <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">
  82. <ImageIcon size={14} className="text-purple-500" />
  83. <span>{t('imagesAndVision')}</span>
  84. </div>
  85. </div>
  86. <button
  87. type="button"
  88. 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"
  89. >
  90. <Folder size={18} />
  91. {t('browseFiles')}
  92. </button>
  93. <input
  94. type="file"
  95. multiple
  96. onChange={handleFileInput}
  97. className="hidden"
  98. id="file-upload-input"
  99. accept=".pdf,.doc,.docx,.xls,.xlsx,.ppt,.pptx,.txt,.md,.html,.csv,.rtf,.odt,.ods,.odp,.json,.js,.jsx,.ts,.tsx,.css,.xml,image/*"
  100. />
  101. </div>
  102. <AnimatePresence>
  103. {isDragging && (
  104. <motion.div
  105. initial={{ opacity: 0 }}
  106. animate={{ opacity: 1 }}
  107. exit={{ opacity: 0 }}
  108. 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"
  109. >
  110. <div className="bg-white p-6 rounded-3xl shadow-2xl flex flex-col items-center gap-3">
  111. <div className="w-12 h-12 bg-blue-600 text-white rounded-2xl flex items-center justify-center animate-bounce">
  112. <FileUp size={24} />
  113. </div>
  114. <span className="text-blue-600 font-bold">{t('dropToIngest')}</span>
  115. </div>
  116. </motion.div>
  117. )}
  118. </AnimatePresence>
  119. </motion.div>
  120. );
  121. };