You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

uploader.tsx 5.4KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153
  1. 'use client'
  2. import type { FC } from 'react'
  3. import React, { useEffect, useRef, useState } from 'react'
  4. import {
  5. RiDeleteBinLine,
  6. RiNodeTree,
  7. RiUploadCloud2Line,
  8. } from '@remixicon/react'
  9. import { useTranslation } from 'react-i18next'
  10. import { useContext } from 'use-context-selector'
  11. import { formatFileSize } from '@/utils/format'
  12. import cn from '@/utils/classnames'
  13. import { ToastContext } from '@/app/components/base/toast'
  14. import ActionButton from '@/app/components/base/action-button'
  15. export type Props = {
  16. file: File | undefined
  17. updateFile: (file?: File) => void
  18. className?: string
  19. }
  20. const Uploader: FC<Props> = ({
  21. file,
  22. updateFile,
  23. className,
  24. }) => {
  25. const { t } = useTranslation()
  26. const { notify } = useContext(ToastContext)
  27. const [dragging, setDragging] = useState(false)
  28. const dropRef = useRef<HTMLDivElement>(null)
  29. const dragRef = useRef<HTMLDivElement>(null)
  30. const fileUploader = useRef<HTMLInputElement>(null)
  31. const handleDragEnter = (e: DragEvent) => {
  32. e.preventDefault()
  33. e.stopPropagation()
  34. e.target !== dragRef.current && setDragging(true)
  35. }
  36. const handleDragOver = (e: DragEvent) => {
  37. e.preventDefault()
  38. e.stopPropagation()
  39. }
  40. const handleDragLeave = (e: DragEvent) => {
  41. e.preventDefault()
  42. e.stopPropagation()
  43. e.target === dragRef.current && setDragging(false)
  44. }
  45. const handleDrop = (e: DragEvent) => {
  46. e.preventDefault()
  47. e.stopPropagation()
  48. setDragging(false)
  49. if (!e.dataTransfer)
  50. return
  51. const files = [...e.dataTransfer.files]
  52. if (files.length > 1) {
  53. notify({ type: 'error', message: t('datasetCreation.stepOne.uploader.validation.count') })
  54. return
  55. }
  56. updateFile(files[0])
  57. }
  58. const selectHandle = () => {
  59. const originalFile = file
  60. if (fileUploader.current) {
  61. fileUploader.current.value = ''
  62. fileUploader.current.click()
  63. // If no file is selected, restore the original file
  64. fileUploader.current.oncancel = () => updateFile(originalFile)
  65. }
  66. }
  67. const removeFile = () => {
  68. if (fileUploader.current)
  69. fileUploader.current.value = ''
  70. updateFile()
  71. }
  72. const fileChangeHandle = (e: React.ChangeEvent<HTMLInputElement>) => {
  73. const currentFile = e.target.files?.[0]
  74. updateFile(currentFile)
  75. }
  76. useEffect(() => {
  77. const dropArea = dropRef.current
  78. dropArea?.addEventListener('dragenter', handleDragEnter)
  79. dropArea?.addEventListener('dragover', handleDragOver)
  80. dropArea?.addEventListener('dragleave', handleDragLeave)
  81. dropArea?.addEventListener('drop', handleDrop)
  82. return () => {
  83. dropArea?.removeEventListener('dragenter', handleDragEnter)
  84. dropArea?.removeEventListener('dragover', handleDragOver)
  85. dropArea?.removeEventListener('dragleave', handleDragLeave)
  86. dropArea?.removeEventListener('drop', handleDrop)
  87. }
  88. }, [])
  89. return (
  90. <div className={cn('mt-6', className)}>
  91. <input
  92. ref={fileUploader}
  93. style={{ display: 'none' }}
  94. type='file'
  95. id='fileUploader'
  96. accept='.pipeline'
  97. onChange={fileChangeHandle}
  98. />
  99. <div ref={dropRef}>
  100. {!file && (
  101. <div
  102. className={cn(
  103. 'flex h-12 items-center rounded-[10px] border border-dashed border-components-dropzone-border bg-components-dropzone-bg text-sm font-normal',
  104. dragging && 'border-components-dropzone-border-accent bg-components-dropzone-bg-accent',
  105. )}>
  106. <div className='flex w-full items-center justify-center space-x-2'>
  107. <RiUploadCloud2Line className='h-6 w-6 text-text-tertiary' />
  108. <div className='text-text-tertiary'>
  109. {t('app.dslUploader.button')}
  110. <span
  111. className='cursor-pointer pl-1 text-text-accent'
  112. onClick={selectHandle}
  113. >
  114. {t('app.dslUploader.browse')}
  115. </span>
  116. </div>
  117. </div>
  118. {dragging && <div ref={dragRef} className='absolute left-0 top-0 h-full w-full' />}
  119. </div>
  120. )}
  121. {file && (
  122. <div className='group flex items-center rounded-lg border-[0.5px] border-components-panel-border bg-components-panel-on-panel-item-bg shadow-xs hover:bg-components-panel-on-panel-item-bg-hover'>
  123. <div className='flex items-center justify-center p-3'>
  124. <RiNodeTree className='h-6 w-6 shrink-0 text-text-secondary' />
  125. </div>
  126. <div className='flex grow flex-col items-start gap-0.5 py-1 pr-2'>
  127. <span className='font-inter max-w-[calc(100%_-_30px)] overflow-hidden text-ellipsis whitespace-nowrap text-[12px] font-medium leading-4 text-text-secondary'>
  128. {file.name}
  129. </span>
  130. <div className='font-inter flex h-3 items-center gap-1 self-stretch text-[10px] font-medium uppercase leading-3 text-text-tertiary'>
  131. <span>PIPELINE</span>
  132. <span className='text-text-quaternary'>·</span>
  133. <span>{formatFileSize(file.size)}</span>
  134. </div>
  135. </div>
  136. <div className='hidden items-center pr-3 group-hover:flex'>
  137. <ActionButton onClick={removeFile}>
  138. <RiDeleteBinLine className='h-4 w-4 text-text-tertiary' />
  139. </ActionButton>
  140. </div>
  141. </div>
  142. )}
  143. </div>
  144. </div>
  145. )
  146. }
  147. export default React.memo(Uploader)