| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | import type { ClipboardEvent } from 'react' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   useCallback, | 
					
						
							|  |  |  |   useState, | 
					
						
							|  |  |  | } from 'react' | 
					
						
							|  |  |  | import { useParams } from 'next/navigation' | 
					
						
							|  |  |  | import produce from 'immer' | 
					
						
							|  |  |  | import { v4 as uuid4 } from 'uuid' | 
					
						
							|  |  |  | import { useTranslation } from 'react-i18next' | 
					
						
							|  |  |  | import type { FileEntity } from './types' | 
					
						
							|  |  |  | import { useFileStore } from './store' | 
					
						
							|  |  |  | import { | 
					
						
							|  |  |  |   fileUpload, | 
					
						
							|  |  |  |   getSupportFileType, | 
					
						
							|  |  |  |   isAllowedFileExtension, | 
					
						
							|  |  |  | } from './utils' | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  | import { | 
					
						
							|  |  |  |   AUDIO_SIZE_LIMIT, | 
					
						
							|  |  |  |   FILE_SIZE_LIMIT, | 
					
						
							|  |  |  |   IMG_SIZE_LIMIT, | 
					
						
							| 
									
										
										
										
											2024-11-04 15:55:34 +08:00
										 |  |  |   MAX_FILE_UPLOAD_LIMIT, | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  |   VIDEO_SIZE_LIMIT, | 
					
						
							|  |  |  | } from '@/app/components/base/file-uploader/constants' | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | import { useToastContext } from '@/app/components/base/toast' | 
					
						
							|  |  |  | import { TransferMethod } from '@/types/app' | 
					
						
							|  |  |  | import { SupportUploadFileTypes } from '@/app/components/workflow/types' | 
					
						
							|  |  |  | import type { FileUpload } from '@/app/components/base/features/types' | 
					
						
							|  |  |  | import { formatFileSize } from '@/utils/format' | 
					
						
							| 
									
										
										
										
											2024-11-01 15:45:27 +08:00
										 |  |  | import { uploadRemoteFileInfo } from '@/service/common' | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  | import type { FileUploadConfigResponse } from '@/models/common' | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export const useFileSizeLimit = (fileUploadConfig?: FileUploadConfigResponse) => { | 
					
						
							|  |  |  |   const imgSizeLimit = Number(fileUploadConfig?.image_file_size_limit) * 1024 * 1024 || IMG_SIZE_LIMIT | 
					
						
							|  |  |  |   const docSizeLimit = Number(fileUploadConfig?.file_size_limit) * 1024 * 1024 || FILE_SIZE_LIMIT | 
					
						
							|  |  |  |   const audioSizeLimit = Number(fileUploadConfig?.audio_file_size_limit) * 1024 * 1024 || AUDIO_SIZE_LIMIT | 
					
						
							|  |  |  |   const videoSizeLimit = Number(fileUploadConfig?.video_file_size_limit) * 1024 * 1024 || VIDEO_SIZE_LIMIT | 
					
						
							| 
									
										
										
										
											2024-11-04 15:55:34 +08:00
										 |  |  |   const maxFileUploadLimit = Number(fileUploadConfig?.workflow_file_upload_limit) || MAX_FILE_UPLOAD_LIMIT | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     imgSizeLimit, | 
					
						
							|  |  |  |     docSizeLimit, | 
					
						
							|  |  |  |     audioSizeLimit, | 
					
						
							|  |  |  |     videoSizeLimit, | 
					
						
							| 
									
										
										
										
											2024-11-04 15:55:34 +08:00
										 |  |  |     maxFileUploadLimit, | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  | export const useFile = (fileConfig: FileUpload) => { | 
					
						
							|  |  |  |   const { t } = useTranslation() | 
					
						
							|  |  |  |   const { notify } = useToastContext() | 
					
						
							|  |  |  |   const fileStore = useFileStore() | 
					
						
							|  |  |  |   const params = useParams() | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  |   const { imgSizeLimit, docSizeLimit, audioSizeLimit, videoSizeLimit } = useFileSizeLimit(fileConfig.fileUploadConfig) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-01 15:45:27 +08:00
										 |  |  |   const checkSizeLimit = useCallback((fileType: string, fileSize: number) => { | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  |     switch (fileType) { | 
					
						
							|  |  |  |       case SupportUploadFileTypes.image: { | 
					
						
							|  |  |  |         if (fileSize > imgSizeLimit) { | 
					
						
							|  |  |  |           notify({ | 
					
						
							|  |  |  |             type: 'error', | 
					
						
							|  |  |  |             message: t('common.fileUploader.uploadFromComputerLimit', { | 
					
						
							|  |  |  |               type: SupportUploadFileTypes.image, | 
					
						
							|  |  |  |               size: formatFileSize(imgSizeLimit), | 
					
						
							|  |  |  |             }), | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |           return false | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case SupportUploadFileTypes.document: { | 
					
						
							|  |  |  |         if (fileSize > docSizeLimit) { | 
					
						
							|  |  |  |           notify({ | 
					
						
							|  |  |  |             type: 'error', | 
					
						
							|  |  |  |             message: t('common.fileUploader.uploadFromComputerLimit', { | 
					
						
							|  |  |  |               type: SupportUploadFileTypes.document, | 
					
						
							|  |  |  |               size: formatFileSize(docSizeLimit), | 
					
						
							|  |  |  |             }), | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |           return false | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case SupportUploadFileTypes.audio: { | 
					
						
							|  |  |  |         if (fileSize > audioSizeLimit) { | 
					
						
							|  |  |  |           notify({ | 
					
						
							|  |  |  |             type: 'error', | 
					
						
							|  |  |  |             message: t('common.fileUploader.uploadFromComputerLimit', { | 
					
						
							|  |  |  |               type: SupportUploadFileTypes.audio, | 
					
						
							|  |  |  |               size: formatFileSize(audioSizeLimit), | 
					
						
							|  |  |  |             }), | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |           return false | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case SupportUploadFileTypes.video: { | 
					
						
							|  |  |  |         if (fileSize > videoSizeLimit) { | 
					
						
							|  |  |  |           notify({ | 
					
						
							|  |  |  |             type: 'error', | 
					
						
							|  |  |  |             message: t('common.fileUploader.uploadFromComputerLimit', { | 
					
						
							|  |  |  |               type: SupportUploadFileTypes.video, | 
					
						
							|  |  |  |               size: formatFileSize(videoSizeLimit), | 
					
						
							|  |  |  |             }), | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |           return false | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       case SupportUploadFileTypes.custom: { | 
					
						
							|  |  |  |         if (fileSize > docSizeLimit) { | 
					
						
							|  |  |  |           notify({ | 
					
						
							|  |  |  |             type: 'error', | 
					
						
							|  |  |  |             message: t('common.fileUploader.uploadFromComputerLimit', { | 
					
						
							|  |  |  |               type: SupportUploadFileTypes.document, | 
					
						
							|  |  |  |               size: formatFileSize(docSizeLimit), | 
					
						
							|  |  |  |             }), | 
					
						
							|  |  |  |           }) | 
					
						
							|  |  |  |           return false | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       default: { | 
					
						
							|  |  |  |         return true | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-11-01 15:45:27 +08:00
										 |  |  |   }, [audioSizeLimit, docSizeLimit, imgSizeLimit, notify, t, videoSizeLimit]) | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleAddFile = useCallback((newFile: FileEntity) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       files, | 
					
						
							|  |  |  |       setFiles, | 
					
						
							|  |  |  |     } = fileStore.getState() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const newFiles = produce(files, (draft) => { | 
					
						
							|  |  |  |       draft.push(newFile) | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     setFiles(newFiles) | 
					
						
							|  |  |  |   }, [fileStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleUpdateFile = useCallback((newFile: FileEntity) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       files, | 
					
						
							|  |  |  |       setFiles, | 
					
						
							|  |  |  |     } = fileStore.getState() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const newFiles = produce(files, (draft) => { | 
					
						
							|  |  |  |       const index = draft.findIndex(file => file.id === newFile.id) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (index > -1) | 
					
						
							|  |  |  |         draft[index] = newFile | 
					
						
							|  |  |  |     }) | 
					
						
							|  |  |  |     setFiles(newFiles) | 
					
						
							|  |  |  |   }, [fileStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleRemoveFile = useCallback((fileId: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       files, | 
					
						
							|  |  |  |       setFiles, | 
					
						
							|  |  |  |     } = fileStore.getState() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const newFiles = files.filter(file => file.id !== fileId) | 
					
						
							|  |  |  |     setFiles(newFiles) | 
					
						
							|  |  |  |   }, [fileStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleReUploadFile = useCallback((fileId: string) => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       files, | 
					
						
							|  |  |  |       setFiles, | 
					
						
							|  |  |  |     } = fileStore.getState() | 
					
						
							|  |  |  |     const index = files.findIndex(file => file.id === fileId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (index > -1) { | 
					
						
							|  |  |  |       const uploadingFile = files[index] | 
					
						
							|  |  |  |       const newFiles = produce(files, (draft) => { | 
					
						
							|  |  |  |         draft[index].progress = 0 | 
					
						
							|  |  |  |       }) | 
					
						
							|  |  |  |       setFiles(newFiles) | 
					
						
							|  |  |  |       fileUpload({ | 
					
						
							|  |  |  |         file: uploadingFile.originalFile!, | 
					
						
							|  |  |  |         onProgressCallback: (progress) => { | 
					
						
							|  |  |  |           handleUpdateFile({ ...uploadingFile, progress }) | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         onSuccessCallback: (res) => { | 
					
						
							|  |  |  |           handleUpdateFile({ ...uploadingFile, uploadedId: res.id, progress: 100 }) | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |         onErrorCallback: () => { | 
					
						
							|  |  |  |           notify({ type: 'error', message: t('common.fileUploader.uploadFromComputerUploadError') }) | 
					
						
							|  |  |  |           handleUpdateFile({ ...uploadingFile, progress: -1 }) | 
					
						
							|  |  |  |         }, | 
					
						
							|  |  |  |       }, !!params.token) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [fileStore, notify, t, handleUpdateFile, params]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-01 15:45:27 +08:00
										 |  |  |   const startProgressTimer = useCallback((fileId: string) => { | 
					
						
							|  |  |  |     const timer = setInterval(() => { | 
					
						
							|  |  |  |       const files = fileStore.getState().files | 
					
						
							|  |  |  |       const file = files.find(file => file.id === fileId) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (file && file.progress < 80 && file.progress >= 0) | 
					
						
							|  |  |  |         handleUpdateFile({ ...file, progress: file.progress + 20 }) | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         clearTimeout(timer) | 
					
						
							|  |  |  |     }, 200) | 
					
						
							|  |  |  |   }, [fileStore, handleUpdateFile]) | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |   const handleLoadFileFromLink = useCallback((url: string) => { | 
					
						
							|  |  |  |     const allowedFileTypes = fileConfig.allowed_file_types | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const uploadingFile = { | 
					
						
							|  |  |  |       id: uuid4(), | 
					
						
							|  |  |  |       name: url, | 
					
						
							|  |  |  |       type: '', | 
					
						
							|  |  |  |       size: 0, | 
					
						
							|  |  |  |       progress: 0, | 
					
						
							| 
									
										
										
										
											2024-11-29 17:26:31 +08:00
										 |  |  |       transferMethod: TransferMethod.local_file, | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       supportFileType: '', | 
					
						
							|  |  |  |       url, | 
					
						
							| 
									
										
										
										
											2024-11-01 15:45:27 +08:00
										 |  |  |       isRemote: true, | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |     } | 
					
						
							|  |  |  |     handleAddFile(uploadingFile) | 
					
						
							| 
									
										
										
										
											2024-11-01 15:45:27 +08:00
										 |  |  |     startProgressTimer(uploadingFile.id) | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-11-02 17:03:14 +08:00
										 |  |  |     uploadRemoteFileInfo(url, !!params.token).then((res) => { | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       const newFile = { | 
					
						
							|  |  |  |         ...uploadingFile, | 
					
						
							| 
									
										
										
										
											2024-11-01 15:45:27 +08:00
										 |  |  |         type: res.mime_type, | 
					
						
							|  |  |  |         size: res.size, | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |         progress: 100, | 
					
						
							| 
									
										
										
										
											2024-11-01 15:45:27 +08:00
										 |  |  |         supportFileType: getSupportFileType(res.name, res.mime_type, allowedFileTypes?.includes(SupportUploadFileTypes.custom)), | 
					
						
							|  |  |  |         uploadedId: res.id, | 
					
						
							|  |  |  |         url: res.url, | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (!isAllowedFileExtension(res.name, res.mime_type, fileConfig.allowed_file_types || [], fileConfig.allowed_file_extensions || [])) { | 
					
						
							|  |  |  |         notify({ type: 'error', message: t('common.fileUploader.fileExtensionNotSupport') }) | 
					
						
							|  |  |  |         handleRemoveFile(uploadingFile.id) | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  |       if (!checkSizeLimit(newFile.supportFileType, newFile.size)) | 
					
						
							|  |  |  |         handleRemoveFile(uploadingFile.id) | 
					
						
							|  |  |  |       else | 
					
						
							|  |  |  |         handleUpdateFile(newFile) | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |     }).catch(() => { | 
					
						
							|  |  |  |       notify({ type: 'error', message: t('common.fileUploader.pasteFileLinkInvalid') }) | 
					
						
							|  |  |  |       handleRemoveFile(uploadingFile.id) | 
					
						
							|  |  |  |     }) | 
					
						
							| 
									
										
										
										
											2024-11-18 15:57:32 +08:00
										 |  |  |   }, [checkSizeLimit, handleAddFile, handleUpdateFile, notify, t, handleRemoveFile, fileConfig?.allowed_file_types, fileConfig.allowed_file_extensions, startProgressTimer, params.token]) | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleLoadFileFromLinkSuccess = useCallback(() => { }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleLoadFileFromLinkError = useCallback(() => { }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleClearFiles = useCallback(() => { | 
					
						
							|  |  |  |     const { | 
					
						
							|  |  |  |       setFiles, | 
					
						
							|  |  |  |     } = fileStore.getState() | 
					
						
							|  |  |  |     setFiles([]) | 
					
						
							|  |  |  |   }, [fileStore]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleLocalFileUpload = useCallback((file: File) => { | 
					
						
							|  |  |  |     if (!isAllowedFileExtension(file.name, file.type, fileConfig.allowed_file_types || [], fileConfig.allowed_file_extensions || [])) { | 
					
						
							|  |  |  |       notify({ type: 'error', message: t('common.fileUploader.fileExtensionNotSupport') }) | 
					
						
							|  |  |  |       return | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  |     const allowedFileTypes = fileConfig.allowed_file_types | 
					
						
							|  |  |  |     const fileType = getSupportFileType(file.name, file.type, allowedFileTypes?.includes(SupportUploadFileTypes.custom)) | 
					
						
							|  |  |  |     if (!checkSizeLimit(fileType, file.size)) | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |       return | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  |     const reader = new FileReader() | 
					
						
							|  |  |  |     const isImage = file.type.startsWith('image') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     reader.addEventListener( | 
					
						
							|  |  |  |       'load', | 
					
						
							|  |  |  |       () => { | 
					
						
							|  |  |  |         const uploadingFile = { | 
					
						
							|  |  |  |           id: uuid4(), | 
					
						
							|  |  |  |           name: file.name, | 
					
						
							|  |  |  |           type: file.type, | 
					
						
							|  |  |  |           size: file.size, | 
					
						
							|  |  |  |           progress: 0, | 
					
						
							|  |  |  |           transferMethod: TransferMethod.local_file, | 
					
						
							|  |  |  |           supportFileType: getSupportFileType(file.name, file.type, allowedFileTypes?.includes(SupportUploadFileTypes.custom)), | 
					
						
							|  |  |  |           originalFile: file, | 
					
						
							|  |  |  |           base64Url: isImage ? reader.result as string : '', | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         handleAddFile(uploadingFile) | 
					
						
							|  |  |  |         fileUpload({ | 
					
						
							|  |  |  |           file: uploadingFile.originalFile, | 
					
						
							|  |  |  |           onProgressCallback: (progress) => { | 
					
						
							|  |  |  |             handleUpdateFile({ ...uploadingFile, progress }) | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           onSuccessCallback: (res) => { | 
					
						
							|  |  |  |             handleUpdateFile({ ...uploadingFile, uploadedId: res.id, progress: 100 }) | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |           onErrorCallback: () => { | 
					
						
							|  |  |  |             notify({ type: 'error', message: t('common.fileUploader.uploadFromComputerUploadError') }) | 
					
						
							|  |  |  |             handleUpdateFile({ ...uploadingFile, progress: -1 }) | 
					
						
							|  |  |  |           }, | 
					
						
							|  |  |  |         }, !!params.token) | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       false, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     reader.addEventListener( | 
					
						
							|  |  |  |       'error', | 
					
						
							|  |  |  |       () => { | 
					
						
							|  |  |  |         notify({ type: 'error', message: t('common.fileUploader.uploadFromComputerReadError') }) | 
					
						
							|  |  |  |       }, | 
					
						
							|  |  |  |       false, | 
					
						
							|  |  |  |     ) | 
					
						
							|  |  |  |     reader.readAsDataURL(file) | 
					
						
							| 
									
										
										
										
											2024-10-24 22:55:17 +08:00
										 |  |  |   }, [checkSizeLimit, notify, t, handleAddFile, handleUpdateFile, params.token, fileConfig?.allowed_file_types, fileConfig?.allowed_file_extensions]) | 
					
						
							| 
									
										
										
										
											2024-10-21 10:32:37 +08:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const handleClipboardPasteFile = useCallback((e: ClipboardEvent<HTMLTextAreaElement>) => { | 
					
						
							|  |  |  |     const file = e.clipboardData?.files[0] | 
					
						
							|  |  |  |     if (file) { | 
					
						
							|  |  |  |       e.preventDefault() | 
					
						
							|  |  |  |       handleLocalFileUpload(file) | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [handleLocalFileUpload]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const [isDragActive, setIsDragActive] = useState(false) | 
					
						
							|  |  |  |   const handleDragFileEnter = useCallback((e: React.DragEvent<HTMLElement>) => { | 
					
						
							|  |  |  |     e.preventDefault() | 
					
						
							|  |  |  |     e.stopPropagation() | 
					
						
							|  |  |  |     setIsDragActive(true) | 
					
						
							|  |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleDragFileOver = useCallback((e: React.DragEvent<HTMLElement>) => { | 
					
						
							|  |  |  |     e.preventDefault() | 
					
						
							|  |  |  |     e.stopPropagation() | 
					
						
							|  |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleDragFileLeave = useCallback((e: React.DragEvent<HTMLElement>) => { | 
					
						
							|  |  |  |     e.preventDefault() | 
					
						
							|  |  |  |     e.stopPropagation() | 
					
						
							|  |  |  |     setIsDragActive(false) | 
					
						
							|  |  |  |   }, []) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   const handleDropFile = useCallback((e: React.DragEvent<HTMLElement>) => { | 
					
						
							|  |  |  |     e.preventDefault() | 
					
						
							|  |  |  |     e.stopPropagation() | 
					
						
							|  |  |  |     setIsDragActive(false) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const file = e.dataTransfer.files[0] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (file) | 
					
						
							|  |  |  |       handleLocalFileUpload(file) | 
					
						
							|  |  |  |   }, [handleLocalFileUpload]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   return { | 
					
						
							|  |  |  |     handleAddFile, | 
					
						
							|  |  |  |     handleUpdateFile, | 
					
						
							|  |  |  |     handleRemoveFile, | 
					
						
							|  |  |  |     handleReUploadFile, | 
					
						
							|  |  |  |     handleLoadFileFromLink, | 
					
						
							|  |  |  |     handleLoadFileFromLinkSuccess, | 
					
						
							|  |  |  |     handleLoadFileFromLinkError, | 
					
						
							|  |  |  |     handleClearFiles, | 
					
						
							|  |  |  |     handleLocalFileUpload, | 
					
						
							|  |  |  |     handleClipboardPasteFile, | 
					
						
							|  |  |  |     isDragActive, | 
					
						
							|  |  |  |     handleDragFileEnter, | 
					
						
							|  |  |  |     handleDragFileOver, | 
					
						
							|  |  |  |     handleDragFileLeave, | 
					
						
							|  |  |  |     handleDropFile, | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |