Вы не можете выбрать более 25 тем Темы должны начинаться с буквы или цифры, могут содержать дефисы(-) и должны содержать не более 35 символов.

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. 'use client'
  2. import type { FC } from 'react'
  3. import React, { useEffect, useRef, useState } from 'react'
  4. import {
  5. RiDeleteBinLine,
  6. RiUploadCloud2Line,
  7. } from '@remixicon/react'
  8. import { useTranslation } from 'react-i18next'
  9. import { useContext } from 'use-context-selector'
  10. import { formatFileSize } from '@/utils/format'
  11. import cn from '@/utils/classnames'
  12. import { Yaml as YamlIcon } from '@/app/components/base/icons/src/public/files'
  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. dropRef.current?.addEventListener('dragenter', handleDragEnter)
  78. dropRef.current?.addEventListener('dragover', handleDragOver)
  79. dropRef.current?.addEventListener('dragleave', handleDragLeave)
  80. dropRef.current?.addEventListener('drop', handleDrop)
  81. return () => {
  82. dropRef.current?.removeEventListener('dragenter', handleDragEnter)
  83. dropRef.current?.removeEventListener('dragover', handleDragOver)
  84. dropRef.current?.removeEventListener('dragleave', handleDragLeave)
  85. dropRef.current?.removeEventListener('drop', handleDrop)
  86. }
  87. }, [])
  88. return (
  89. <div className={cn('mt-6', className)}>
  90. <input
  91. ref={fileUploader}
  92. style={{ display: 'none' }}
  93. type="file"
  94. id="fileUploader"
  95. accept='.yaml,.yml'
  96. onChange={fileChangeHandle}
  97. />
  98. <div ref={dropRef}>
  99. {!file && (
  100. <div className={cn('flex h-12 items-center rounded-[10px] border border-dashed border-components-dropzone-border bg-components-dropzone-bg text-sm font-normal', dragging && 'border-components-dropzone-border-accent bg-components-dropzone-bg-accent')}>
  101. <div className='flex w-full items-center justify-center space-x-2'>
  102. <RiUploadCloud2Line className='h-6 w-6 text-text-tertiary' />
  103. <div className='text-text-tertiary'>
  104. {t('app.dslUploader.button')}
  105. <span className='cursor-pointer pl-1 text-text-accent' onClick={selectHandle}>{t('app.dslUploader.browse')}</span>
  106. </div>
  107. </div>
  108. {dragging && <div ref={dragRef} className='absolute left-0 top-0 h-full w-full' />}
  109. </div>
  110. )}
  111. {file && (
  112. <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:bg-components-panel-on-panel-item-bg-hover')}>
  113. <div className='flex items-center justify-center p-3'>
  114. <YamlIcon className="h-6 w-6 shrink-0" />
  115. </div>
  116. <div className='flex grow flex-col items-start gap-0.5 py-1 pr-2'>
  117. <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>
  118. <div className='font-inter flex h-3 items-center gap-1 self-stretch text-[10px] font-medium uppercase leading-3 text-text-tertiary'>
  119. <span>YAML</span>
  120. <span className='text-text-quaternary'>·</span>
  121. <span>{formatFileSize(file.size)}</span>
  122. </div>
  123. </div>
  124. <div className='hidden items-center pr-3 group-hover:flex'>
  125. <ActionButton onClick={removeFile}>
  126. <RiDeleteBinLine className='h-4 w-4 text-text-tertiary' />
  127. </ActionButton>
  128. </div>
  129. </div>
  130. )}
  131. </div>
  132. </div>
  133. )
  134. }
  135. export default React.memo(Uploader)