mirror of
https://github.com/langgenius/dify.git
synced 2026-05-13 08:57:28 +08:00
128 lines
5.2 KiB
TypeScript
128 lines
5.2 KiB
TypeScript
'use client'
|
|
import type { FC } from 'react'
|
|
import { RiDeleteBinLine, RiNodeTree, RiUploadCloud2Line } from '@remixicon/react'
|
|
import * as React from 'react'
|
|
import { useEffect, useRef, useState } from 'react'
|
|
import { useTranslation } from 'react-i18next'
|
|
import ActionButton from '@/app/components/base/action-button'
|
|
import { toast } from '@/app/components/base/ui/toast'
|
|
import { cn } from '@/utils/classnames'
|
|
import { formatFileSize } from '@/utils/format'
|
|
|
|
export type Props = {
|
|
file: File | undefined
|
|
updateFile: (file?: File) => void
|
|
className?: string
|
|
}
|
|
const Uploader: FC<Props> = ({ file, updateFile, className }) => {
|
|
const { t } = useTranslation()
|
|
const [dragging, setDragging] = useState(false)
|
|
const dropRef = useRef<HTMLDivElement>(null)
|
|
const dragRef = useRef<HTMLDivElement>(null)
|
|
const fileUploader = useRef<HTMLInputElement>(null)
|
|
const handleDragEnter = (e: DragEvent) => {
|
|
e.preventDefault()
|
|
e.stopPropagation()
|
|
if (e.target !== dragRef.current)
|
|
setDragging(true)
|
|
}
|
|
const handleDragOver = (e: DragEvent) => {
|
|
e.preventDefault()
|
|
e.stopPropagation()
|
|
}
|
|
const handleDragLeave = (e: DragEvent) => {
|
|
e.preventDefault()
|
|
e.stopPropagation()
|
|
if (e.target === dragRef.current)
|
|
setDragging(false)
|
|
}
|
|
const handleDrop = (e: DragEvent) => {
|
|
e.preventDefault()
|
|
e.stopPropagation()
|
|
setDragging(false)
|
|
if (!e.dataTransfer)
|
|
return
|
|
const files = Array.from(e.dataTransfer.files)
|
|
if (files.length > 1) {
|
|
toast.error(t('stepOne.uploader.validation.count', { ns: 'datasetCreation' }))
|
|
return
|
|
}
|
|
updateFile(files[0])
|
|
}
|
|
const selectHandle = () => {
|
|
const originalFile = file
|
|
if (fileUploader.current) {
|
|
fileUploader.current.value = ''
|
|
fileUploader.current.click()
|
|
// If no file is selected, restore the original file
|
|
fileUploader.current.oncancel = () => updateFile(originalFile)
|
|
}
|
|
}
|
|
const removeFile = () => {
|
|
if (fileUploader.current)
|
|
fileUploader.current.value = ''
|
|
updateFile()
|
|
}
|
|
const fileChangeHandle = (e: React.ChangeEvent<HTMLInputElement>) => {
|
|
const currentFile = e.target.files?.[0]
|
|
updateFile(currentFile)
|
|
}
|
|
useEffect(() => {
|
|
const dropArea = dropRef.current
|
|
dropArea?.addEventListener('dragenter', handleDragEnter)
|
|
dropArea?.addEventListener('dragover', handleDragOver)
|
|
dropArea?.addEventListener('dragleave', handleDragLeave)
|
|
dropArea?.addEventListener('drop', handleDrop)
|
|
return () => {
|
|
dropArea?.removeEventListener('dragenter', handleDragEnter)
|
|
dropArea?.removeEventListener('dragover', handleDragOver)
|
|
dropArea?.removeEventListener('dragleave', handleDragLeave)
|
|
dropArea?.removeEventListener('drop', handleDrop)
|
|
}
|
|
}, [])
|
|
return (
|
|
<div className={cn('mt-6', className)}>
|
|
<input ref={fileUploader} style={{ display: 'none' }} type="file" id="fileUploader" accept=".pipeline" onChange={fileChangeHandle} />
|
|
<div ref={dropRef}>
|
|
{!file && (
|
|
<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')}>
|
|
<div className="flex w-full items-center justify-center space-x-2">
|
|
<RiUploadCloud2Line className="h-6 w-6 text-text-tertiary" />
|
|
<div className="text-text-tertiary">
|
|
{t('dslUploader.button', { ns: 'app' })}
|
|
<span className="cursor-pointer pl-1 text-text-accent" onClick={selectHandle}>
|
|
{t('dslUploader.browse', { ns: 'app' })}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
{dragging && <div ref={dragRef} className="absolute left-0 top-0 h-full w-full" />}
|
|
</div>
|
|
)}
|
|
{file && (
|
|
<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">
|
|
<div className="flex items-center justify-center p-3">
|
|
<RiNodeTree className="h-6 w-6 shrink-0 text-text-secondary" />
|
|
</div>
|
|
<div className="flex grow flex-col items-start gap-0.5 py-1 pr-2">
|
|
<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>
|
|
<div className="font-inter flex h-3 items-center gap-1 self-stretch text-[10px] font-medium uppercase leading-3 text-text-tertiary">
|
|
<span>PIPELINE</span>
|
|
<span className="text-text-quaternary">·</span>
|
|
<span>{formatFileSize(file.size)}</span>
|
|
</div>
|
|
</div>
|
|
<div className="hidden items-center pr-3 group-hover:flex">
|
|
<ActionButton onClick={removeFile}>
|
|
<RiDeleteBinLine className="h-4 w-4 text-text-tertiary" />
|
|
</ActionButton>
|
|
</div>
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
export default React.memo(Uploader)
|