|
@@ -1,19 +1,21 @@
|
|
'use client'
|
|
'use client'
|
|
-import React, { useCallback, useEffect, useRef, useState } from 'react'
|
|
|
|
|
|
+import React, { useEffect, useRef, useState } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { useTranslation } from 'react-i18next'
|
|
import { useContext } from 'use-context-selector'
|
|
import { useContext } from 'use-context-selector'
|
|
import cn from 'classnames'
|
|
import cn from 'classnames'
|
|
import s from './index.module.css'
|
|
import s from './index.module.css'
|
|
import type { File as FileEntity } from '@/models/datasets'
|
|
import type { File as FileEntity } from '@/models/datasets'
|
|
import { ToastContext } from '@/app/components/base/toast'
|
|
import { ToastContext } from '@/app/components/base/toast'
|
|
-import Button from '@/app/components/base/button'
|
|
|
|
|
|
|
|
import { upload } from '@/service/base'
|
|
import { upload } from '@/service/base'
|
|
|
|
|
|
type IFileUploaderProps = {
|
|
type IFileUploaderProps = {
|
|
- file?: FileEntity
|
|
|
|
|
|
+ fileList: any[]
|
|
titleClassName?: string
|
|
titleClassName?: string
|
|
- onFileUpdate: (file?: FileEntity) => void
|
|
|
|
|
|
+ prepareFileList: (files: any[]) => void
|
|
|
|
+ onFileUpdate: (fileItem: any, progress: number, list: any[]) => void
|
|
|
|
+ onFileListUpdate?: (files: any) => void
|
|
|
|
+ onPreview: (file: FileEntity) => void
|
|
}
|
|
}
|
|
|
|
|
|
const ACCEPTS = [
|
|
const ACCEPTS = [
|
|
@@ -28,19 +30,25 @@ const ACCEPTS = [
|
|
'.csv',
|
|
'.csv',
|
|
]
|
|
]
|
|
|
|
|
|
-const MAX_SIZE = 15 * 1024 * 1024
|
|
|
|
|
|
+const MAX_SIZE = 10 * 1024 * 1024
|
|
|
|
+const BATCH_COUNT = 5
|
|
|
|
|
|
-const FileUploader = ({ file, onFileUpdate, titleClassName }: IFileUploaderProps) => {
|
|
|
|
|
|
+const FileUploader = ({
|
|
|
|
+ fileList,
|
|
|
|
+ titleClassName,
|
|
|
|
+ prepareFileList,
|
|
|
|
+ onFileUpdate,
|
|
|
|
+ onFileListUpdate,
|
|
|
|
+ onPreview,
|
|
|
|
+}: IFileUploaderProps) => {
|
|
const { t } = useTranslation()
|
|
const { t } = useTranslation()
|
|
const { notify } = useContext(ToastContext)
|
|
const { notify } = useContext(ToastContext)
|
|
const [dragging, setDragging] = useState(false)
|
|
const [dragging, setDragging] = useState(false)
|
|
const dropRef = useRef<HTMLDivElement>(null)
|
|
const dropRef = useRef<HTMLDivElement>(null)
|
|
const dragRef = useRef<HTMLDivElement>(null)
|
|
const dragRef = useRef<HTMLDivElement>(null)
|
|
const fileUploader = useRef<HTMLInputElement>(null)
|
|
const fileUploader = useRef<HTMLInputElement>(null)
|
|
- const uploadPromise = useRef<any>(null)
|
|
|
|
- const [currentFile, setCurrentFile] = useState<File>()
|
|
|
|
- const [uploading, setUploading] = useState(false)
|
|
|
|
- const [percent, setPercent] = useState(0)
|
|
|
|
|
|
+
|
|
|
|
+ const fileListRef = useRef<any>([])
|
|
|
|
|
|
// utils
|
|
// utils
|
|
const getFileType = (currentFile: File) => {
|
|
const getFileType = (currentFile: File) => {
|
|
@@ -50,10 +58,7 @@ const FileUploader = ({ file, onFileUpdate, titleClassName }: IFileUploaderProps
|
|
const arr = currentFile.name.split('.')
|
|
const arr = currentFile.name.split('.')
|
|
return arr[arr.length - 1]
|
|
return arr[arr.length - 1]
|
|
}
|
|
}
|
|
- const getFileName = (name: string) => {
|
|
|
|
- const arr = name.split('.')
|
|
|
|
- return arr.slice(0, -1).join()
|
|
|
|
- }
|
|
|
|
|
|
+
|
|
const getFileSize = (size: number) => {
|
|
const getFileSize = (size: number) => {
|
|
if (size / 1024 < 10)
|
|
if (size / 1024 < 10)
|
|
return `${(size / 1024).toFixed(2)}KB`
|
|
return `${(size / 1024).toFixed(2)}KB`
|
|
@@ -74,52 +79,76 @@ const FileUploader = ({ file, onFileUpdate, titleClassName }: IFileUploaderProps
|
|
|
|
|
|
return isValidType && isValidSize
|
|
return isValidType && isValidSize
|
|
}
|
|
}
|
|
- const onProgress = useCallback((e: ProgressEvent) => {
|
|
|
|
- if (e.lengthComputable) {
|
|
|
|
- const percent = Math.floor(e.loaded / e.total * 100)
|
|
|
|
- setPercent(percent)
|
|
|
|
- }
|
|
|
|
- }, [setPercent])
|
|
|
|
- const abort = () => {
|
|
|
|
- const currentXHR = uploadPromise.current
|
|
|
|
- currentXHR.abort()
|
|
|
|
- }
|
|
|
|
- const fileUpload = async (file?: File) => {
|
|
|
|
- if (!file)
|
|
|
|
- return
|
|
|
|
-
|
|
|
|
- if (!isValid(file))
|
|
|
|
- return
|
|
|
|
|
|
|
|
- setCurrentFile(file)
|
|
|
|
- setUploading(true)
|
|
|
|
|
|
+ const fileUpload = async (fileItem: any) => {
|
|
const formData = new FormData()
|
|
const formData = new FormData()
|
|
- formData.append('file', file)
|
|
|
|
- // store for abort
|
|
|
|
- const currentXHR = new XMLHttpRequest()
|
|
|
|
- uploadPromise.current = currentXHR
|
|
|
|
- try {
|
|
|
|
- const result = await upload({
|
|
|
|
- xhr: currentXHR,
|
|
|
|
- data: formData,
|
|
|
|
- onprogress: onProgress,
|
|
|
|
- }) as FileEntity
|
|
|
|
- onFileUpdate(result)
|
|
|
|
- setUploading(false)
|
|
|
|
|
|
+ formData.append('file', fileItem.file)
|
|
|
|
+ const onProgress = (e: ProgressEvent) => {
|
|
|
|
+ if (e.lengthComputable) {
|
|
|
|
+ const percent = Math.floor(e.loaded / e.total * 100)
|
|
|
|
+ onFileUpdate(fileItem, percent, fileListRef.current)
|
|
|
|
+ }
|
|
}
|
|
}
|
|
- catch (xhr: any) {
|
|
|
|
- setUploading(false)
|
|
|
|
- // abort handle
|
|
|
|
- if (xhr.readyState === 0 && xhr.status === 0) {
|
|
|
|
- if (fileUploader.current)
|
|
|
|
- fileUploader.current.value = ''
|
|
|
|
|
|
|
|
- setCurrentFile(undefined)
|
|
|
|
- return
|
|
|
|
- }
|
|
|
|
- notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.failed') })
|
|
|
|
|
|
+ return upload({
|
|
|
|
+ xhr: new XMLHttpRequest(),
|
|
|
|
+ data: formData,
|
|
|
|
+ onprogress: onProgress,
|
|
|
|
+ })
|
|
|
|
+ .then((res: FileEntity) => {
|
|
|
|
+ const fileListCopy = fileListRef.current
|
|
|
|
+
|
|
|
|
+ const completeFile = {
|
|
|
|
+ fileID: fileItem.fileID,
|
|
|
|
+ file: res,
|
|
|
|
+ }
|
|
|
|
+ const index = fileListCopy.findIndex((item: any) => item.fileID === fileItem.fileID)
|
|
|
|
+ fileListCopy[index] = completeFile
|
|
|
|
+ onFileUpdate(completeFile, 100, fileListCopy)
|
|
|
|
+ return Promise.resolve({ ...completeFile })
|
|
|
|
+ })
|
|
|
|
+ .catch(() => {
|
|
|
|
+ notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.failed') })
|
|
|
|
+ onFileUpdate(fileItem, -2, fileListCopy)
|
|
|
|
+ return Promise.resolve({ ...fileItem })
|
|
|
|
+ })
|
|
|
|
+ .finally()
|
|
|
|
+ }
|
|
|
|
+ const uploadBatchFiles = (bFiles: any) => {
|
|
|
|
+ bFiles.forEach((bf: any) => (bf.progress = 0))
|
|
|
|
+ return Promise.all(bFiles.map((bFile: any) => fileUpload(bFile)))
|
|
|
|
+ }
|
|
|
|
+ const uploadMultipleFiles = async (files: any) => {
|
|
|
|
+ const length = files.length
|
|
|
|
+ let start = 0
|
|
|
|
+ let end = 0
|
|
|
|
+
|
|
|
|
+ while (start < length) {
|
|
|
|
+ if (start + BATCH_COUNT > length)
|
|
|
|
+ end = length
|
|
|
|
+ else
|
|
|
|
+ end = start + BATCH_COUNT
|
|
|
|
+ const bFiles = files.slice(start, end)
|
|
|
|
+ await uploadBatchFiles(bFiles)
|
|
|
|
+ start = end
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
+ const initialUpload = (files: any) => {
|
|
|
|
+ if (!files.length)
|
|
|
|
+ return false
|
|
|
|
+ const preparedFiles = files.map((file: any, index: number) => {
|
|
|
|
+ const fileItem = {
|
|
|
|
+ fileID: `file${index}-${Date.now()}`,
|
|
|
|
+ file,
|
|
|
|
+ progress: -1,
|
|
|
|
+ }
|
|
|
|
+ return fileItem
|
|
|
|
+ })
|
|
|
|
+ const newFiles = [...fileListRef.current, ...preparedFiles]
|
|
|
|
+ prepareFileList(newFiles)
|
|
|
|
+ fileListRef.current = newFiles
|
|
|
|
+ uploadMultipleFiles(preparedFiles)
|
|
|
|
+ }
|
|
const handleDragEnter = (e: DragEvent) => {
|
|
const handleDragEnter = (e: DragEvent) => {
|
|
e.preventDefault()
|
|
e.preventDefault()
|
|
e.stopPropagation()
|
|
e.stopPropagation()
|
|
@@ -134,6 +163,7 @@ const FileUploader = ({ file, onFileUpdate, titleClassName }: IFileUploaderProps
|
|
e.stopPropagation()
|
|
e.stopPropagation()
|
|
e.target === dragRef.current && setDragging(false)
|
|
e.target === dragRef.current && setDragging(false)
|
|
}
|
|
}
|
|
|
|
+
|
|
const handleDrop = (e: DragEvent) => {
|
|
const handleDrop = (e: DragEvent) => {
|
|
e.preventDefault()
|
|
e.preventDefault()
|
|
e.stopPropagation()
|
|
e.stopPropagation()
|
|
@@ -142,29 +172,26 @@ const FileUploader = ({ file, onFileUpdate, titleClassName }: IFileUploaderProps
|
|
return
|
|
return
|
|
|
|
|
|
const files = [...e.dataTransfer.files]
|
|
const files = [...e.dataTransfer.files]
|
|
- if (files.length > 1) {
|
|
|
|
- notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.validation.count') })
|
|
|
|
- return
|
|
|
|
- }
|
|
|
|
- onFileUpdate()
|
|
|
|
- fileUpload(files[0])
|
|
|
|
|
|
+ const validFiles = files.filter(file => isValid(file))
|
|
|
|
+ // fileUpload(files[0])
|
|
|
|
+ initialUpload(validFiles)
|
|
}
|
|
}
|
|
|
|
|
|
const selectHandle = () => {
|
|
const selectHandle = () => {
|
|
if (fileUploader.current)
|
|
if (fileUploader.current)
|
|
fileUploader.current.click()
|
|
fileUploader.current.click()
|
|
}
|
|
}
|
|
- const removeFile = () => {
|
|
|
|
|
|
+
|
|
|
|
+ const removeFile = (fileID: string) => {
|
|
if (fileUploader.current)
|
|
if (fileUploader.current)
|
|
fileUploader.current.value = ''
|
|
fileUploader.current.value = ''
|
|
|
|
|
|
- setCurrentFile(undefined)
|
|
|
|
- onFileUpdate()
|
|
|
|
|
|
+ fileListRef.current = fileListRef.current.filter((item: any) => item.fileID !== fileID)
|
|
|
|
+ onFileListUpdate?.([...fileListRef.current])
|
|
}
|
|
}
|
|
const fileChangeHandle = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
const fileChangeHandle = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
- const currentFile = e.target.files?.[0]
|
|
|
|
- onFileUpdate()
|
|
|
|
- fileUpload(currentFile)
|
|
|
|
|
|
+ const files = [...(e.target.files ?? [])].filter(file => isValid(file))
|
|
|
|
+ initialUpload(files)
|
|
}
|
|
}
|
|
|
|
|
|
useEffect(() => {
|
|
useEffect(() => {
|
|
@@ -184,83 +211,83 @@ const FileUploader = ({ file, onFileUpdate, titleClassName }: IFileUploaderProps
|
|
<div className={s.fileUploader}>
|
|
<div className={s.fileUploader}>
|
|
<input
|
|
<input
|
|
ref={fileUploader}
|
|
ref={fileUploader}
|
|
|
|
+ id="fileUploader"
|
|
style={{ display: 'none' }}
|
|
style={{ display: 'none' }}
|
|
type="file"
|
|
type="file"
|
|
- id="fileUploader"
|
|
|
|
|
|
+ multiple
|
|
accept={ACCEPTS.join(',')}
|
|
accept={ACCEPTS.join(',')}
|
|
onChange={fileChangeHandle}
|
|
onChange={fileChangeHandle}
|
|
/>
|
|
/>
|
|
<div className={cn(s.title, titleClassName)}>{t('datasetCreation.stepOne.uploader.title')}</div>
|
|
<div className={cn(s.title, titleClassName)}>{t('datasetCreation.stepOne.uploader.title')}</div>
|
|
- <div ref={dropRef}>
|
|
|
|
- {!currentFile && !file && (
|
|
|
|
- <div className={cn(s.uploader, dragging && s.dragging)}>
|
|
|
|
- <span>{t('datasetCreation.stepOne.uploader.button')}</span>
|
|
|
|
- <label className={s.browse} onClick={selectHandle}>{t('datasetCreation.stepOne.uploader.browse')}</label>
|
|
|
|
- {dragging && <div ref={dragRef} className={s.draggingCover}/>}
|
|
|
|
- </div>
|
|
|
|
- )}
|
|
|
|
|
|
+ <div ref={dropRef} className={cn(s.uploader, dragging && s.dragging)}>
|
|
|
|
+ <div className='flex justify-center items-center h-6 mb-2'>
|
|
|
|
+ <span className={s.uploadIcon}/>
|
|
|
|
+ <span>{t('datasetCreation.stepOne.uploader.button')}</span>
|
|
|
|
+ <label className={s.browse} onClick={selectHandle}>{t('datasetCreation.stepOne.uploader.browse')}</label>
|
|
|
|
+ </div>
|
|
|
|
+ <div className={s.tip}>{t('datasetCreation.stepOne.uploader.tip')}</div>
|
|
|
|
+ {dragging && <div ref={dragRef} className={s.draggingCover}/>}
|
|
</div>
|
|
</div>
|
|
- {currentFile && (
|
|
|
|
- <div className={cn(s.file, uploading && s.uploading)}>
|
|
|
|
- {uploading && (
|
|
|
|
- <div className={s.progressbar} style={{ width: `${percent}%` }}/>
|
|
|
|
- )}
|
|
|
|
- <div className={cn(s.fileIcon, s[getFileType(currentFile)])}/>
|
|
|
|
- <div className={s.fileInfo}>
|
|
|
|
- <div className={s.filename}>
|
|
|
|
- <span className={s.name}>{getFileName(currentFile.name)}</span>
|
|
|
|
- <span className={s.extension}>{`.${getFileType(currentFile)}`}</span>
|
|
|
|
|
|
+ <div className={s.fileList}>
|
|
|
|
+ {fileList.map((fileItem, index) => (
|
|
|
|
+ <div
|
|
|
|
+ key={`${fileItem.fileID}-${index}`}
|
|
|
|
+ onClick={() => fileItem.file?.id && onPreview(fileItem.file)}
|
|
|
|
+ className={cn(
|
|
|
|
+ s.file,
|
|
|
|
+ fileItem.progress < 100 && s.uploading,
|
|
|
|
+ // s.active,
|
|
|
|
+ )}
|
|
|
|
+ >
|
|
|
|
+ {fileItem.progress < 100 && (
|
|
|
|
+ <div className={s.progressbar} style={{ width: `${fileItem.progress}%` }}/>
|
|
|
|
+ )}
|
|
|
|
+ <div className={s.fileInfo}>
|
|
|
|
+ <div className={cn(s.fileIcon, s[getFileType(fileItem.file)])}/>
|
|
|
|
+ <div className={s.filename}>{fileItem.file.name}</div>
|
|
|
|
+ <div className={s.size}>{getFileSize(fileItem.file.size)}</div>
|
|
</div>
|
|
</div>
|
|
- <div className={s.fileExtraInfo}>
|
|
|
|
- <span className={s.size}>{getFileSize(currentFile.size)}</span>
|
|
|
|
- <span className={s.error}></span>
|
|
|
|
|
|
+ <div className={s.actionWrapper}>
|
|
|
|
+ {(fileItem.progress < 100 && fileItem.progress >= 0) && (
|
|
|
|
+ <div className={s.percent}>{`${fileItem.progress}%`}</div>
|
|
|
|
+ )}
|
|
|
|
+ {fileItem.progress === 100 && (
|
|
|
|
+ <div className={s.remove} onClick={(e) => {
|
|
|
|
+ e.stopPropagation()
|
|
|
|
+ removeFile(fileItem.fileID)
|
|
|
|
+ }}/>
|
|
|
|
+ )}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
- <div className={s.actionWrapper}>
|
|
|
|
- {uploading && (
|
|
|
|
- <>
|
|
|
|
- <div className={s.percent}>{`${percent}%`}</div>
|
|
|
|
- <div className={s.divider}/>
|
|
|
|
- <div className={s.buttonWrapper}>
|
|
|
|
- <Button className={cn(s.button, 'ml-2 !h-8 bg-white')} onClick={abort}>{t('datasetCreation.stepOne.uploader.cancel')}</Button>
|
|
|
|
- </div>
|
|
|
|
- </>
|
|
|
|
|
|
+ ))}
|
|
|
|
+ {/* {currentFile && (
|
|
|
|
+ <div
|
|
|
|
+ // onClick={() => onPreview(currentFile)}
|
|
|
|
+ className={cn(
|
|
|
|
+ s.file,
|
|
|
|
+ uploading && s.uploading,
|
|
|
|
+ // s.active,
|
|
)}
|
|
)}
|
|
- {!uploading && (
|
|
|
|
- <>
|
|
|
|
- <div className={s.buttonWrapper}>
|
|
|
|
- <Button className={cn(s.button, 'ml-2 !h-8 bg-white')} onClick={selectHandle}>{t('datasetCreation.stepOne.uploader.change')}</Button>
|
|
|
|
- <div className={s.divider}/>
|
|
|
|
- <div className={s.remove} onClick={removeFile}/>
|
|
|
|
- </div>
|
|
|
|
- </>
|
|
|
|
|
|
+ >
|
|
|
|
+ {uploading && (
|
|
|
|
+ <div className={s.progressbar} style={{ width: `${percent}%` }}/>
|
|
)}
|
|
)}
|
|
- </div>
|
|
|
|
- </div>
|
|
|
|
- )}
|
|
|
|
- {!currentFile && file && (
|
|
|
|
- <div className={cn(s.file)}>
|
|
|
|
- <div className={cn(s.fileIcon, s[file.extension])}/>
|
|
|
|
- <div className={s.fileInfo}>
|
|
|
|
- <div className={s.filename}>
|
|
|
|
- <span className={s.name}>{getFileName(file.name)}</span>
|
|
|
|
- <span className={s.extension}>{`.${file.extension}`}</span>
|
|
|
|
- </div>
|
|
|
|
- <div className={s.fileExtraInfo}>
|
|
|
|
- <span className={s.size}>{getFileSize(file.size)}</span>
|
|
|
|
- <span className={s.error}></span>
|
|
|
|
|
|
+ <div className={s.fileInfo}>
|
|
|
|
+ <div className={cn(s.fileIcon, s[getFileType(currentFile)])}/>
|
|
|
|
+ <div className={s.filename}>{currentFile.name}</div>
|
|
|
|
+ <div className={s.size}>{getFileSize(currentFile.size)}</div>
|
|
</div>
|
|
</div>
|
|
- </div>
|
|
|
|
- <div className={s.actionWrapper}>
|
|
|
|
- <div className={s.buttonWrapper}>
|
|
|
|
- <Button className={cn(s.button, 'ml-2 !h-8 bg-white')} onClick={selectHandle}>{t('datasetCreation.stepOne.uploader.change')}</Button>
|
|
|
|
- <div className={s.divider}/>
|
|
|
|
- <div className={s.remove} onClick={removeFile}/>
|
|
|
|
|
|
+ <div className={s.actionWrapper}>
|
|
|
|
+ {uploading && (
|
|
|
|
+ <div className={s.percent}>{`${percent}%`}</div>
|
|
|
|
+ )}
|
|
|
|
+ {!uploading && (
|
|
|
|
+ <div className={s.remove} onClick={() => removeFile(index)}/>
|
|
|
|
+ )}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
- </div>
|
|
|
|
- )}
|
|
|
|
- <div className={s.tip}>{t('datasetCreation.stepOne.uploader.tip')}</div>
|
|
|
|
|
|
+ )} */}
|
|
|
|
+ </div>
|
|
</div>
|
|
</div>
|
|
)
|
|
)
|
|
}
|
|
}
|