Você não pode selecionar mais de 25 tópicos Os tópicos devem começar com uma letra ou um número, podem incluir traços ('-') e podem ter até 35 caracteres.

uploader.tsx 5.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149
  1. 'use client'
  2. import type { FC } from 'react'
  3. import React, { useEffect, useRef, useState } from 'react'
  4. import {
  5. RiDeleteBinLine,
  6. } from '@remixicon/react'
  7. import { useTranslation } from 'react-i18next'
  8. import { useContext } from 'use-context-selector'
  9. import { formatFileSize } from '@/utils/format'
  10. import cn from '@/utils/classnames'
  11. import { Yaml as YamlIcon } from '@/app/components/base/icons/src/public/files'
  12. import { ToastContext } from '@/app/components/base/toast'
  13. import { UploadCloud01 } from '@/app/components/base/icons/src/vender/line/general'
  14. import Button from '@/app/components/base/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='.yaml,.yml'
  97. onChange={fileChangeHandle}
  98. />
  99. <div ref={dropRef}>
  100. {!file && (
  101. <div className={cn('flex h-12 items-center rounded-xl border border-dashed border-gray-200 bg-gray-50 text-sm font-normal', dragging && 'border border-[#B2CCFF] bg-[#F5F8FF]')}>
  102. <div className='flex w-full items-center justify-center space-x-2'>
  103. <UploadCloud01 className='mr-2 h-6 w-6' />
  104. <div className='text-text-tertiary'>
  105. {t('datasetCreation.stepOne.uploader.button')}
  106. <span
  107. className='cursor-pointer pl-1 text-text-accent'
  108. onClick={selectHandle}
  109. >
  110. {t('datasetDocuments.list.batchModal.browse')}
  111. </span>
  112. </div>
  113. </div>
  114. {dragging && <div ref={dragRef} className='absolute left-0 top-0 h-full w-full' />}
  115. </div>
  116. )}
  117. {file && (
  118. <div className={cn('group flex items-center rounded-lg border-[0.5px] border-components-panel-border bg-components-panel-on-panel-item-bg shadow-xs', 'hover:border-[#B2CCFF] hover:bg-[#F5F8FF]')}>
  119. <div className='flex items-center justify-center p-3'>
  120. <YamlIcon className='h-6 w-6 shrink-0' />
  121. </div>
  122. <div className='flex grow flex-col items-start gap-0.5 py-1 pr-2'>
  123. <span className='font-inter max-w-[calc(100%_-_30px)] overflow-hidden text-ellipsis whitespace-nowrap text-[12px] font-medium leading-4 text-text-secondary'>{file.name}</span>
  124. <div className='font-inter flex h-3 items-center gap-1 self-stretch text-[10px] font-medium uppercase leading-3 text-text-tertiary'>
  125. <span>YAML</span>
  126. <span className='text-text-quaternary'>·</span>
  127. <span>{formatFileSize(file.size)}</span>
  128. </div>
  129. </div>
  130. <div className='hidden items-center group-hover:flex'>
  131. <Button onClick={selectHandle}>{t('datasetCreation.stepOne.uploader.change')}</Button>
  132. <div className='mx-2 h-4 w-px bg-gray-200' />
  133. <div className='cursor-pointer p-2' onClick={removeFile}>
  134. <RiDeleteBinLine className='h-4 w-4 text-text-tertiary' />
  135. </div>
  136. </div>
  137. </div>
  138. )}
  139. </div>
  140. </div>
  141. )
  142. }
  143. export default React.memo(Uploader)