index.tsx 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. 'use client'
  2. import React, { useEffect, useRef, useState } from 'react'
  3. import { useTranslation } from 'react-i18next'
  4. import { useContext } from 'use-context-selector'
  5. import cn from 'classnames'
  6. import s from './index.module.css'
  7. import type { File as FileEntity } from '@/models/datasets'
  8. import { ToastContext } from '@/app/components/base/toast'
  9. import { upload } from '@/service/base'
  10. type IFileUploaderProps = {
  11. fileList: any[]
  12. titleClassName?: string
  13. prepareFileList: (files: any[]) => void
  14. onFileUpdate: (fileItem: any, progress: number, list: any[]) => void
  15. onFileListUpdate?: (files: any) => void
  16. onPreview: (file: FileEntity) => void
  17. }
  18. const ACCEPTS = [
  19. '.pdf',
  20. '.html',
  21. '.htm',
  22. '.md',
  23. '.markdown',
  24. '.txt',
  25. // '.xls',
  26. '.xlsx',
  27. '.csv',
  28. ]
  29. const MAX_SIZE = 15 * 1024 * 1024
  30. const BATCH_COUNT = 5
  31. const FileUploader = ({
  32. fileList,
  33. titleClassName,
  34. prepareFileList,
  35. onFileUpdate,
  36. onFileListUpdate,
  37. onPreview,
  38. }: IFileUploaderProps) => {
  39. const { t } = useTranslation()
  40. const { notify } = useContext(ToastContext)
  41. const [dragging, setDragging] = useState(false)
  42. const dropRef = useRef<HTMLDivElement>(null)
  43. const dragRef = useRef<HTMLDivElement>(null)
  44. const fileUploader = useRef<HTMLInputElement>(null)
  45. const fileListRef = useRef<any>([])
  46. // utils
  47. const getFileType = (currentFile: File) => {
  48. if (!currentFile)
  49. return ''
  50. const arr = currentFile.name.split('.')
  51. return arr[arr.length - 1]
  52. }
  53. const getFileSize = (size: number) => {
  54. if (size / 1024 < 10)
  55. return `${(size / 1024).toFixed(2)}KB`
  56. return `${(size / 1024 / 1024).toFixed(2)}MB`
  57. }
  58. const isValid = (file: File) => {
  59. const { size } = file
  60. const ext = `.${getFileType(file)}`
  61. const isValidType = ACCEPTS.includes(ext)
  62. if (!isValidType)
  63. notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.validation.typeError') })
  64. const isValidSize = size <= MAX_SIZE
  65. if (!isValidSize)
  66. notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.validation.size') })
  67. return isValidType && isValidSize
  68. }
  69. const fileUpload = async (fileItem: any) => {
  70. const formData = new FormData()
  71. formData.append('file', fileItem.file)
  72. const onProgress = (e: ProgressEvent) => {
  73. if (e.lengthComputable) {
  74. const percent = Math.floor(e.loaded / e.total * 100)
  75. onFileUpdate(fileItem, percent, fileListRef.current)
  76. }
  77. }
  78. return upload({
  79. xhr: new XMLHttpRequest(),
  80. data: formData,
  81. onprogress: onProgress,
  82. })
  83. .then((res: FileEntity) => {
  84. const fileListCopy = fileListRef.current
  85. const completeFile = {
  86. fileID: fileItem.fileID,
  87. file: res,
  88. }
  89. const index = fileListCopy.findIndex((item: any) => item.fileID === fileItem.fileID)
  90. fileListCopy[index] = completeFile
  91. onFileUpdate(completeFile, 100, fileListCopy)
  92. return Promise.resolve({ ...completeFile })
  93. })
  94. .catch(() => {
  95. notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.failed') })
  96. onFileUpdate(fileItem, -2, fileListCopy)
  97. return Promise.resolve({ ...fileItem })
  98. })
  99. .finally()
  100. }
  101. const uploadBatchFiles = (bFiles: any) => {
  102. bFiles.forEach((bf: any) => (bf.progress = 0))
  103. return Promise.all(bFiles.map((bFile: any) => fileUpload(bFile)))
  104. }
  105. const uploadMultipleFiles = async (files: any) => {
  106. const length = files.length
  107. let start = 0
  108. let end = 0
  109. while (start < length) {
  110. if (start + BATCH_COUNT > length)
  111. end = length
  112. else
  113. end = start + BATCH_COUNT
  114. const bFiles = files.slice(start, end)
  115. await uploadBatchFiles(bFiles)
  116. start = end
  117. }
  118. }
  119. const initialUpload = (files: any) => {
  120. if (!files.length)
  121. return false
  122. const preparedFiles = files.map((file: any, index: number) => {
  123. const fileItem = {
  124. fileID: `file${index}-${Date.now()}`,
  125. file,
  126. progress: -1,
  127. }
  128. return fileItem
  129. })
  130. const newFiles = [...fileListRef.current, ...preparedFiles]
  131. prepareFileList(newFiles)
  132. fileListRef.current = newFiles
  133. uploadMultipleFiles(preparedFiles)
  134. }
  135. const handleDragEnter = (e: DragEvent) => {
  136. e.preventDefault()
  137. e.stopPropagation()
  138. e.target !== dragRef.current && setDragging(true)
  139. }
  140. const handleDragOver = (e: DragEvent) => {
  141. e.preventDefault()
  142. e.stopPropagation()
  143. }
  144. const handleDragLeave = (e: DragEvent) => {
  145. e.preventDefault()
  146. e.stopPropagation()
  147. e.target === dragRef.current && setDragging(false)
  148. }
  149. const handleDrop = (e: DragEvent) => {
  150. e.preventDefault()
  151. e.stopPropagation()
  152. setDragging(false)
  153. if (!e.dataTransfer)
  154. return
  155. const files = [...e.dataTransfer.files]
  156. const validFiles = files.filter(file => isValid(file))
  157. // fileUpload(files[0])
  158. initialUpload(validFiles)
  159. }
  160. const selectHandle = () => {
  161. if (fileUploader.current)
  162. fileUploader.current.click()
  163. }
  164. const removeFile = (fileID: string) => {
  165. if (fileUploader.current)
  166. fileUploader.current.value = ''
  167. fileListRef.current = fileListRef.current.filter((item: any) => item.fileID !== fileID)
  168. onFileListUpdate?.([...fileListRef.current])
  169. }
  170. const fileChangeHandle = (e: React.ChangeEvent<HTMLInputElement>) => {
  171. const files = [...(e.target.files ?? [])].filter(file => isValid(file))
  172. initialUpload(files)
  173. }
  174. useEffect(() => {
  175. dropRef.current?.addEventListener('dragenter', handleDragEnter)
  176. dropRef.current?.addEventListener('dragover', handleDragOver)
  177. dropRef.current?.addEventListener('dragleave', handleDragLeave)
  178. dropRef.current?.addEventListener('drop', handleDrop)
  179. return () => {
  180. dropRef.current?.removeEventListener('dragenter', handleDragEnter)
  181. dropRef.current?.removeEventListener('dragover', handleDragOver)
  182. dropRef.current?.removeEventListener('dragleave', handleDragLeave)
  183. dropRef.current?.removeEventListener('drop', handleDrop)
  184. }
  185. }, [])
  186. return (
  187. <div className={s.fileUploader}>
  188. <input
  189. ref={fileUploader}
  190. id="fileUploader"
  191. style={{ display: 'none' }}
  192. type="file"
  193. multiple
  194. accept={ACCEPTS.join(',')}
  195. onChange={fileChangeHandle}
  196. />
  197. <div className={cn(s.title, titleClassName)}>{t('datasetCreation.stepOne.uploader.title')}</div>
  198. <div ref={dropRef} className={cn(s.uploader, dragging && s.dragging)}>
  199. <div className='flex justify-center items-center h-6 mb-2'>
  200. <span className={s.uploadIcon}/>
  201. <span>{t('datasetCreation.stepOne.uploader.button')}</span>
  202. <label className={s.browse} onClick={selectHandle}>{t('datasetCreation.stepOne.uploader.browse')}</label>
  203. </div>
  204. <div className={s.tip}>{t('datasetCreation.stepOne.uploader.tip')}</div>
  205. {dragging && <div ref={dragRef} className={s.draggingCover}/>}
  206. </div>
  207. <div className={s.fileList}>
  208. {fileList.map((fileItem, index) => (
  209. <div
  210. key={`${fileItem.fileID}-${index}`}
  211. onClick={() => fileItem.file?.id && onPreview(fileItem.file)}
  212. className={cn(
  213. s.file,
  214. fileItem.progress < 100 && s.uploading,
  215. // s.active,
  216. )}
  217. >
  218. {fileItem.progress < 100 && (
  219. <div className={s.progressbar} style={{ width: `${fileItem.progress}%` }}/>
  220. )}
  221. <div className={s.fileInfo}>
  222. <div className={cn(s.fileIcon, s[getFileType(fileItem.file)])}/>
  223. <div className={s.filename}>{fileItem.file.name}</div>
  224. <div className={s.size}>{getFileSize(fileItem.file.size)}</div>
  225. </div>
  226. <div className={s.actionWrapper}>
  227. {(fileItem.progress < 100 && fileItem.progress >= 0) && (
  228. <div className={s.percent}>{`${fileItem.progress}%`}</div>
  229. )}
  230. {fileItem.progress === 100 && (
  231. <div className={s.remove} onClick={(e) => {
  232. e.stopPropagation()
  233. removeFile(fileItem.fileID)
  234. }}/>
  235. )}
  236. </div>
  237. </div>
  238. ))}
  239. {/* {currentFile && (
  240. <div
  241. // onClick={() => onPreview(currentFile)}
  242. className={cn(
  243. s.file,
  244. uploading && s.uploading,
  245. // s.active,
  246. )}
  247. >
  248. {uploading && (
  249. <div className={s.progressbar} style={{ width: `${percent}%` }}/>
  250. )}
  251. <div className={s.fileInfo}>
  252. <div className={cn(s.fileIcon, s[getFileType(currentFile)])}/>
  253. <div className={s.filename}>{currentFile.name}</div>
  254. <div className={s.size}>{getFileSize(currentFile.size)}</div>
  255. </div>
  256. <div className={s.actionWrapper}>
  257. {uploading && (
  258. <div className={s.percent}>{`${percent}%`}</div>
  259. )}
  260. {!uploading && (
  261. <div className={s.remove} onClick={() => removeFile(index)}/>
  262. )}
  263. </div>
  264. </div>
  265. )} */}
  266. </div>
  267. </div>
  268. )
  269. }
  270. export default FileUploader