mirror of
https://github.com/infiniflow/ragflow.git
synced 2025-12-08 20:42:30 +08:00
### What problem does this PR solve? Feat: Upload files in the chat box on the agent page #3221 ### Type of change - [x] New Feature (non-breaking change which adds functionality)
1437 lines
37 KiB
TypeScript
1437 lines
37 KiB
TypeScript
// https://www.diceui.com/docs/components/file-upload
|
|
|
|
'use client';
|
|
|
|
import { cn } from '@/lib/utils';
|
|
import { Slot } from '@radix-ui/react-slot';
|
|
import {
|
|
FileArchiveIcon,
|
|
FileAudioIcon,
|
|
FileCodeIcon,
|
|
FileCogIcon,
|
|
FileIcon,
|
|
FileTextIcon,
|
|
FileVideoIcon,
|
|
} from 'lucide-react';
|
|
import * as React from 'react';
|
|
|
|
const ROOT_NAME = 'FileUpload';
|
|
const DROPZONE_NAME = 'FileUploadDropzone';
|
|
const TRIGGER_NAME = 'FileUploadTrigger';
|
|
const LIST_NAME = 'FileUploadList';
|
|
const ITEM_NAME = 'FileUploadItem';
|
|
const ITEM_PREVIEW_NAME = 'FileUploadItemPreview';
|
|
const ITEM_METADATA_NAME = 'FileUploadItemMetadata';
|
|
const ITEM_PROGRESS_NAME = 'FileUploadItemProgress';
|
|
const ITEM_DELETE_NAME = 'FileUploadItemDelete';
|
|
const CLEAR_NAME = 'FileUploadClear';
|
|
|
|
function useLazyRef<T>(fn: () => T) {
|
|
const ref = React.useRef<T | null>(null);
|
|
|
|
if (ref.current === null) {
|
|
ref.current = fn();
|
|
}
|
|
|
|
return ref as React.RefObject<T>;
|
|
}
|
|
|
|
type Direction = 'ltr' | 'rtl';
|
|
|
|
const DirectionContext = React.createContext<Direction | undefined>(undefined);
|
|
|
|
function useDirection(dirProp?: Direction): Direction {
|
|
const contextDir = React.useContext(DirectionContext);
|
|
return dirProp ?? contextDir ?? 'ltr';
|
|
}
|
|
|
|
interface FileState {
|
|
file: File;
|
|
progress: number;
|
|
error?: string;
|
|
status: 'idle' | 'uploading' | 'error' | 'success';
|
|
}
|
|
|
|
interface StoreState {
|
|
files: Map<File, FileState>;
|
|
dragOver: boolean;
|
|
invalid: boolean;
|
|
}
|
|
|
|
type StoreAction =
|
|
| { type: 'ADD_FILES'; files: File[] }
|
|
| { type: 'SET_FILES'; files: File[] }
|
|
| { type: 'SET_PROGRESS'; file: File; progress: number }
|
|
| { type: 'SET_SUCCESS'; file: File }
|
|
| { type: 'SET_ERROR'; file: File; error: string }
|
|
| { type: 'REMOVE_FILE'; file: File }
|
|
| { type: 'SET_DRAG_OVER'; dragOver: boolean }
|
|
| { type: 'SET_INVALID'; invalid: boolean }
|
|
| { type: 'CLEAR' };
|
|
|
|
function createStore(
|
|
listeners: Set<() => void>,
|
|
files: Map<File, FileState>,
|
|
urlCache: WeakMap<File, string>,
|
|
invalid: boolean,
|
|
onValueChange?: (files: File[]) => void,
|
|
) {
|
|
let state: StoreState = {
|
|
files,
|
|
dragOver: false,
|
|
invalid: invalid,
|
|
};
|
|
|
|
function reducer(state: StoreState, action: StoreAction): StoreState {
|
|
switch (action.type) {
|
|
case 'ADD_FILES': {
|
|
for (const file of action.files) {
|
|
files.set(file, {
|
|
file,
|
|
progress: 0,
|
|
status: 'idle',
|
|
});
|
|
}
|
|
|
|
if (onValueChange) {
|
|
const fileList = Array.from(files.values()).map(
|
|
(fileState) => fileState.file,
|
|
);
|
|
onValueChange(fileList);
|
|
}
|
|
return { ...state, files };
|
|
}
|
|
|
|
case 'SET_FILES': {
|
|
const newFileSet = new Set(action.files);
|
|
for (const existingFile of files.keys()) {
|
|
if (!newFileSet.has(existingFile)) {
|
|
files.delete(existingFile);
|
|
}
|
|
}
|
|
|
|
for (const file of action.files) {
|
|
const existingState = files.get(file);
|
|
if (!existingState) {
|
|
files.set(file, {
|
|
file,
|
|
progress: 0,
|
|
status: 'idle',
|
|
});
|
|
}
|
|
}
|
|
return { ...state, files };
|
|
}
|
|
|
|
case 'SET_PROGRESS': {
|
|
const fileState = files.get(action.file);
|
|
if (fileState) {
|
|
files.set(action.file, {
|
|
...fileState,
|
|
progress: action.progress,
|
|
status: 'uploading',
|
|
});
|
|
}
|
|
return { ...state, files };
|
|
}
|
|
|
|
case 'SET_SUCCESS': {
|
|
const fileState = files.get(action.file);
|
|
if (fileState) {
|
|
files.set(action.file, {
|
|
...fileState,
|
|
progress: 100,
|
|
status: 'success',
|
|
});
|
|
}
|
|
return { ...state, files };
|
|
}
|
|
|
|
case 'SET_ERROR': {
|
|
const fileState = files.get(action.file);
|
|
if (fileState) {
|
|
files.set(action.file, {
|
|
...fileState,
|
|
error: action.error,
|
|
status: 'error',
|
|
});
|
|
}
|
|
return { ...state, files };
|
|
}
|
|
|
|
case 'REMOVE_FILE': {
|
|
if (urlCache) {
|
|
const cachedUrl = urlCache.get(action.file);
|
|
if (cachedUrl) {
|
|
URL.revokeObjectURL(cachedUrl);
|
|
urlCache.delete(action.file);
|
|
}
|
|
}
|
|
|
|
files.delete(action.file);
|
|
|
|
if (onValueChange) {
|
|
const fileList = Array.from(files.values()).map(
|
|
(fileState) => fileState.file,
|
|
);
|
|
onValueChange(fileList);
|
|
}
|
|
return { ...state, files };
|
|
}
|
|
|
|
case 'SET_DRAG_OVER': {
|
|
return { ...state, dragOver: action.dragOver };
|
|
}
|
|
|
|
case 'SET_INVALID': {
|
|
return { ...state, invalid: action.invalid };
|
|
}
|
|
|
|
case 'CLEAR': {
|
|
if (urlCache) {
|
|
for (const file of files.keys()) {
|
|
const cachedUrl = urlCache.get(file);
|
|
if (cachedUrl) {
|
|
URL.revokeObjectURL(cachedUrl);
|
|
urlCache.delete(file);
|
|
}
|
|
}
|
|
}
|
|
|
|
files.clear();
|
|
if (onValueChange) {
|
|
onValueChange([]);
|
|
}
|
|
return { ...state, files, invalid: false };
|
|
}
|
|
|
|
default:
|
|
return state;
|
|
}
|
|
}
|
|
|
|
function getState() {
|
|
return state;
|
|
}
|
|
|
|
function dispatch(action: StoreAction) {
|
|
state = reducer(state, action);
|
|
for (const listener of listeners) {
|
|
listener();
|
|
}
|
|
}
|
|
|
|
function subscribe(listener: () => void) {
|
|
listeners.add(listener);
|
|
return () => listeners.delete(listener);
|
|
}
|
|
|
|
return { getState, dispatch, subscribe };
|
|
}
|
|
|
|
const StoreContext = React.createContext<ReturnType<typeof createStore> | null>(
|
|
null,
|
|
);
|
|
|
|
function useStoreContext(consumerName: string) {
|
|
const context = React.useContext(StoreContext);
|
|
if (!context) {
|
|
throw new Error(`\`${consumerName}\` must be used within \`${ROOT_NAME}\``);
|
|
}
|
|
return context;
|
|
}
|
|
|
|
function useStore<T>(selector: (state: StoreState) => T): T {
|
|
const store = useStoreContext(ROOT_NAME);
|
|
|
|
const lastValueRef = useLazyRef<{ value: T; state: StoreState } | null>(
|
|
() => null,
|
|
);
|
|
|
|
const getSnapshot = React.useCallback(() => {
|
|
const state = store.getState();
|
|
const prevValue = lastValueRef.current;
|
|
|
|
if (prevValue && prevValue.state === state) {
|
|
return prevValue.value;
|
|
}
|
|
|
|
const nextValue = selector(state);
|
|
lastValueRef.current = { value: nextValue, state };
|
|
return nextValue;
|
|
}, [store, selector, lastValueRef]);
|
|
|
|
return React.useSyncExternalStore(store.subscribe, getSnapshot, getSnapshot);
|
|
}
|
|
|
|
interface FileUploadContextValue {
|
|
inputId: string;
|
|
dropzoneId: string;
|
|
listId: string;
|
|
labelId: string;
|
|
disabled: boolean;
|
|
dir: Direction;
|
|
inputRef: React.RefObject<HTMLInputElement | null>;
|
|
urlCache: WeakMap<File, string>;
|
|
}
|
|
|
|
const FileUploadContext = React.createContext<FileUploadContextValue | null>(
|
|
null,
|
|
);
|
|
|
|
function useFileUploadContext(consumerName: string) {
|
|
const context = React.useContext(FileUploadContext);
|
|
if (!context) {
|
|
throw new Error(`\`${consumerName}\` must be used within \`${ROOT_NAME}\``);
|
|
}
|
|
return context;
|
|
}
|
|
|
|
interface FileUploadRootProps
|
|
extends Omit<
|
|
React.ComponentPropsWithoutRef<'div'>,
|
|
'defaultValue' | 'onChange'
|
|
> {
|
|
value?: File[];
|
|
defaultValue?: File[];
|
|
onValueChange?: (files: File[]) => void;
|
|
onAccept?: (files: File[]) => void;
|
|
onFileAccept?: (file: File) => void;
|
|
onFileReject?: (file: File, message: string) => void;
|
|
onFileValidate?: (file: File) => string | null | undefined;
|
|
onUpload?: (
|
|
files: File[],
|
|
options: {
|
|
onProgress: (file: File, progress: number) => void;
|
|
onSuccess: (file: File) => void;
|
|
onError: (file: File, error: Error) => void;
|
|
},
|
|
) => Promise<void> | void;
|
|
accept?: string;
|
|
maxFiles?: number;
|
|
maxSize?: number;
|
|
dir?: Direction;
|
|
label?: string;
|
|
name?: string;
|
|
asChild?: boolean;
|
|
disabled?: boolean;
|
|
invalid?: boolean;
|
|
multiple?: boolean;
|
|
required?: boolean;
|
|
}
|
|
|
|
function FileUploadRoot(props: FileUploadRootProps) {
|
|
const {
|
|
value,
|
|
defaultValue,
|
|
onValueChange,
|
|
onAccept,
|
|
onFileAccept,
|
|
onFileReject,
|
|
onFileValidate,
|
|
onUpload,
|
|
accept,
|
|
maxFiles,
|
|
maxSize,
|
|
dir: dirProp,
|
|
label,
|
|
name,
|
|
asChild,
|
|
disabled = false,
|
|
invalid = false,
|
|
multiple = false,
|
|
required = false,
|
|
children,
|
|
className,
|
|
...rootProps
|
|
} = props;
|
|
|
|
const inputId = React.useId();
|
|
const dropzoneId = React.useId();
|
|
const listId = React.useId();
|
|
const labelId = React.useId();
|
|
|
|
const dir = useDirection(dirProp);
|
|
const listeners = useLazyRef(() => new Set<() => void>()).current;
|
|
const files = useLazyRef<Map<File, FileState>>(() => new Map()).current;
|
|
const urlCache = useLazyRef(() => new WeakMap<File, string>()).current;
|
|
const inputRef = React.useRef<HTMLInputElement>(null);
|
|
const isControlled = value !== undefined;
|
|
|
|
const store = React.useMemo(
|
|
() => createStore(listeners, files, urlCache, invalid, onValueChange),
|
|
[listeners, files, invalid, onValueChange, urlCache],
|
|
);
|
|
|
|
const acceptTypes = React.useMemo(
|
|
() => accept?.split(',').map((t) => t.trim()) ?? null,
|
|
[accept],
|
|
);
|
|
|
|
const onProgress = useLazyRef(() => {
|
|
let frame = 0;
|
|
return (file: File, progress: number) => {
|
|
if (frame) return;
|
|
frame = requestAnimationFrame(() => {
|
|
frame = 0;
|
|
store.dispatch({
|
|
type: 'SET_PROGRESS',
|
|
file,
|
|
progress: Math.min(Math.max(0, progress), 100),
|
|
});
|
|
});
|
|
};
|
|
}).current;
|
|
|
|
React.useEffect(() => {
|
|
if (isControlled) {
|
|
store.dispatch({ type: 'SET_FILES', files: value });
|
|
} else if (
|
|
defaultValue &&
|
|
defaultValue.length > 0 &&
|
|
!store.getState().files.size
|
|
) {
|
|
store.dispatch({ type: 'SET_FILES', files: defaultValue });
|
|
}
|
|
}, [value, defaultValue, isControlled, store]);
|
|
|
|
React.useEffect(() => {
|
|
return () => {
|
|
for (const file of files.keys()) {
|
|
const cachedUrl = urlCache.get(file);
|
|
if (cachedUrl) {
|
|
URL.revokeObjectURL(cachedUrl);
|
|
}
|
|
}
|
|
};
|
|
}, [files, urlCache]);
|
|
|
|
const onFilesChange = React.useCallback(
|
|
(originalFiles: File[]) => {
|
|
if (disabled) return;
|
|
|
|
let filesToProcess = [...originalFiles];
|
|
let invalid = false;
|
|
|
|
if (maxFiles) {
|
|
const currentCount = store.getState().files.size;
|
|
const remainingSlotCount = Math.max(0, maxFiles - currentCount);
|
|
|
|
if (remainingSlotCount < filesToProcess.length) {
|
|
const rejectedFiles = filesToProcess.slice(remainingSlotCount);
|
|
invalid = true;
|
|
|
|
filesToProcess = filesToProcess.slice(0, remainingSlotCount);
|
|
|
|
for (const file of rejectedFiles) {
|
|
let rejectionMessage = `Maximum ${maxFiles} files allowed`;
|
|
|
|
if (onFileValidate) {
|
|
const validationMessage = onFileValidate(file);
|
|
if (validationMessage) {
|
|
rejectionMessage = validationMessage;
|
|
}
|
|
}
|
|
|
|
onFileReject?.(file, rejectionMessage);
|
|
}
|
|
}
|
|
}
|
|
|
|
const acceptedFiles: File[] = [];
|
|
const rejectedFiles: { file: File; message: string }[] = [];
|
|
|
|
for (const file of filesToProcess) {
|
|
let rejected = false;
|
|
let rejectionMessage = '';
|
|
|
|
if (onFileValidate) {
|
|
const validationMessage = onFileValidate(file);
|
|
if (validationMessage) {
|
|
rejectionMessage = validationMessage;
|
|
onFileReject?.(file, rejectionMessage);
|
|
rejected = true;
|
|
invalid = true;
|
|
continue;
|
|
}
|
|
}
|
|
|
|
if (acceptTypes) {
|
|
const fileType = file.type;
|
|
const fileExtension = `.${file.name.split('.').pop()}`;
|
|
|
|
if (
|
|
!acceptTypes.some(
|
|
(type) =>
|
|
type === fileType ||
|
|
type === fileExtension ||
|
|
(type.includes('/*') &&
|
|
fileType.startsWith(type.replace('/*', '/'))),
|
|
)
|
|
) {
|
|
rejectionMessage = 'File type not accepted';
|
|
onFileReject?.(file, rejectionMessage);
|
|
rejected = true;
|
|
invalid = true;
|
|
}
|
|
}
|
|
|
|
if (maxSize && file.size > maxSize) {
|
|
rejectionMessage = 'File too large';
|
|
onFileReject?.(file, rejectionMessage);
|
|
rejected = true;
|
|
invalid = true;
|
|
}
|
|
|
|
if (!rejected) {
|
|
acceptedFiles.push(file);
|
|
} else {
|
|
rejectedFiles.push({ file, message: rejectionMessage });
|
|
}
|
|
}
|
|
|
|
if (invalid) {
|
|
store.dispatch({ type: 'SET_INVALID', invalid });
|
|
setTimeout(() => {
|
|
store.dispatch({ type: 'SET_INVALID', invalid: false });
|
|
}, 2000);
|
|
}
|
|
|
|
if (acceptedFiles.length > 0) {
|
|
store.dispatch({ type: 'ADD_FILES', files: acceptedFiles });
|
|
|
|
if (isControlled && onValueChange) {
|
|
const currentFiles = Array.from(store.getState().files.values()).map(
|
|
(f) => f.file,
|
|
);
|
|
onValueChange([...currentFiles]);
|
|
}
|
|
|
|
if (onAccept) {
|
|
onAccept(acceptedFiles);
|
|
}
|
|
|
|
for (const file of acceptedFiles) {
|
|
onFileAccept?.(file);
|
|
}
|
|
|
|
if (onUpload) {
|
|
requestAnimationFrame(() => {
|
|
onFilesUpload(acceptedFiles);
|
|
});
|
|
}
|
|
}
|
|
},
|
|
[
|
|
store,
|
|
isControlled,
|
|
onValueChange,
|
|
onAccept,
|
|
onFileAccept,
|
|
onUpload,
|
|
maxFiles,
|
|
onFileValidate,
|
|
onFileReject,
|
|
acceptTypes,
|
|
maxSize,
|
|
disabled,
|
|
],
|
|
);
|
|
|
|
const onFilesUpload = React.useCallback(
|
|
async (files: File[]) => {
|
|
try {
|
|
for (const file of files) {
|
|
store.dispatch({ type: 'SET_PROGRESS', file, progress: 0 });
|
|
}
|
|
|
|
if (onUpload) {
|
|
await onUpload(files, {
|
|
onProgress,
|
|
onSuccess: (file) => {
|
|
store.dispatch({ type: 'SET_SUCCESS', file });
|
|
},
|
|
onError: (file, error) => {
|
|
store.dispatch({
|
|
type: 'SET_ERROR',
|
|
file,
|
|
error: error.message ?? 'Upload failed',
|
|
});
|
|
},
|
|
});
|
|
} else {
|
|
for (const file of files) {
|
|
store.dispatch({ type: 'SET_SUCCESS', file });
|
|
}
|
|
}
|
|
} catch (error) {
|
|
const errorMessage =
|
|
error instanceof Error ? error.message : 'Upload failed';
|
|
for (const file of files) {
|
|
store.dispatch({
|
|
type: 'SET_ERROR',
|
|
file,
|
|
error: errorMessage,
|
|
});
|
|
}
|
|
}
|
|
},
|
|
[store, onUpload, onProgress],
|
|
);
|
|
|
|
const onInputChange = React.useCallback(
|
|
(event: React.ChangeEvent<HTMLInputElement>) => {
|
|
const files = Array.from(event.target.files ?? []);
|
|
onFilesChange(files);
|
|
event.target.value = '';
|
|
},
|
|
[onFilesChange],
|
|
);
|
|
|
|
const contextValue = React.useMemo<FileUploadContextValue>(
|
|
() => ({
|
|
dropzoneId,
|
|
inputId,
|
|
listId,
|
|
labelId,
|
|
dir,
|
|
disabled,
|
|
inputRef,
|
|
urlCache,
|
|
}),
|
|
[dropzoneId, inputId, listId, labelId, dir, disabled, urlCache],
|
|
);
|
|
|
|
const RootPrimitive = asChild ? Slot : 'div';
|
|
|
|
return (
|
|
<StoreContext.Provider value={store}>
|
|
<FileUploadContext.Provider value={contextValue}>
|
|
<RootPrimitive
|
|
data-disabled={disabled ? '' : undefined}
|
|
data-slot="file-upload"
|
|
dir={dir}
|
|
{...rootProps}
|
|
className={cn('relative flex flex-col gap-2', className)}
|
|
>
|
|
{children}
|
|
<input
|
|
type="file"
|
|
id={inputId}
|
|
aria-labelledby={labelId}
|
|
aria-describedby={dropzoneId}
|
|
ref={inputRef}
|
|
tabIndex={-1}
|
|
accept={accept}
|
|
name={name}
|
|
className="sr-only"
|
|
disabled={disabled}
|
|
multiple={multiple}
|
|
required={required}
|
|
onChange={onInputChange}
|
|
/>
|
|
<span id={labelId} className="sr-only">
|
|
{label ?? 'File upload'}
|
|
</span>
|
|
</RootPrimitive>
|
|
</FileUploadContext.Provider>
|
|
</StoreContext.Provider>
|
|
);
|
|
}
|
|
|
|
interface FileUploadDropzoneProps
|
|
extends React.ComponentPropsWithoutRef<'div'> {
|
|
asChild?: boolean;
|
|
}
|
|
|
|
function FileUploadDropzone(props: FileUploadDropzoneProps) {
|
|
const {
|
|
asChild,
|
|
className,
|
|
onClick: onClickProp,
|
|
onDragOver: onDragOverProp,
|
|
onDragEnter: onDragEnterProp,
|
|
onDragLeave: onDragLeaveProp,
|
|
onDrop: onDropProp,
|
|
onPaste: onPasteProp,
|
|
onKeyDown: onKeyDownProp,
|
|
...dropzoneProps
|
|
} = props;
|
|
|
|
const context = useFileUploadContext(DROPZONE_NAME);
|
|
const store = useStoreContext(DROPZONE_NAME);
|
|
const dragOver = useStore((state) => state.dragOver);
|
|
const invalid = useStore((state) => state.invalid);
|
|
|
|
const onClick = React.useCallback(
|
|
(event: React.MouseEvent<HTMLDivElement>) => {
|
|
onClickProp?.(event);
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
const target = event.target;
|
|
|
|
const isFromTrigger =
|
|
target instanceof HTMLElement &&
|
|
target.closest('[data-slot="file-upload-trigger"]');
|
|
|
|
if (!isFromTrigger) {
|
|
context.inputRef.current?.click();
|
|
}
|
|
},
|
|
[context.inputRef, onClickProp],
|
|
);
|
|
|
|
const onDragOver = React.useCallback(
|
|
(event: React.DragEvent<HTMLDivElement>) => {
|
|
onDragOverProp?.(event);
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
event.preventDefault();
|
|
store.dispatch({ type: 'SET_DRAG_OVER', dragOver: true });
|
|
},
|
|
[store, onDragOverProp],
|
|
);
|
|
|
|
const onDragEnter = React.useCallback(
|
|
(event: React.DragEvent<HTMLDivElement>) => {
|
|
onDragEnterProp?.(event);
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
event.preventDefault();
|
|
store.dispatch({ type: 'SET_DRAG_OVER', dragOver: true });
|
|
},
|
|
[store, onDragEnterProp],
|
|
);
|
|
|
|
const onDragLeave = React.useCallback(
|
|
(event: React.DragEvent<HTMLDivElement>) => {
|
|
onDragLeaveProp?.(event);
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
const relatedTarget = event.relatedTarget;
|
|
if (
|
|
relatedTarget &&
|
|
relatedTarget instanceof Node &&
|
|
event.currentTarget.contains(relatedTarget)
|
|
) {
|
|
return;
|
|
}
|
|
|
|
event.preventDefault();
|
|
store.dispatch({ type: 'SET_DRAG_OVER', dragOver: false });
|
|
},
|
|
[store, onDragLeaveProp],
|
|
);
|
|
|
|
const onDrop = React.useCallback(
|
|
(event: React.DragEvent<HTMLDivElement>) => {
|
|
onDropProp?.(event);
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
event.preventDefault();
|
|
store.dispatch({ type: 'SET_DRAG_OVER', dragOver: false });
|
|
|
|
const files = Array.from(event.dataTransfer.files);
|
|
const inputElement = context.inputRef.current;
|
|
if (!inputElement) return;
|
|
|
|
const dataTransfer = new DataTransfer();
|
|
for (const file of files) {
|
|
dataTransfer.items.add(file);
|
|
}
|
|
|
|
inputElement.files = dataTransfer.files;
|
|
inputElement.dispatchEvent(new Event('change', { bubbles: true }));
|
|
},
|
|
[store, context.inputRef, onDropProp],
|
|
);
|
|
|
|
const onPaste = React.useCallback(
|
|
(event: React.ClipboardEvent<HTMLDivElement>) => {
|
|
onPasteProp?.(event);
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
event.preventDefault();
|
|
store.dispatch({ type: 'SET_DRAG_OVER', dragOver: false });
|
|
|
|
const items = event.clipboardData?.items;
|
|
if (!items) return;
|
|
|
|
const files: File[] = [];
|
|
for (let i = 0; i < items.length; i++) {
|
|
const item = items[i];
|
|
if (item?.kind === 'file') {
|
|
const file = item.getAsFile();
|
|
if (file) {
|
|
files.push(file);
|
|
}
|
|
}
|
|
}
|
|
|
|
if (files.length === 0) return;
|
|
|
|
const inputElement = context.inputRef.current;
|
|
if (!inputElement) return;
|
|
|
|
const dataTransfer = new DataTransfer();
|
|
for (const file of files) {
|
|
dataTransfer.items.add(file);
|
|
}
|
|
|
|
inputElement.files = dataTransfer.files;
|
|
inputElement.dispatchEvent(new Event('change', { bubbles: true }));
|
|
},
|
|
[store, context.inputRef, onPasteProp],
|
|
);
|
|
|
|
const onKeyDown = React.useCallback(
|
|
(event: React.KeyboardEvent<HTMLDivElement>) => {
|
|
onKeyDownProp?.(event);
|
|
|
|
if (
|
|
!event.defaultPrevented &&
|
|
(event.key === 'Enter' || event.key === ' ')
|
|
) {
|
|
event.preventDefault();
|
|
context.inputRef.current?.click();
|
|
}
|
|
},
|
|
[context.inputRef, onKeyDownProp],
|
|
);
|
|
|
|
const DropzonePrimitive = asChild ? Slot : 'div';
|
|
|
|
return (
|
|
<DropzonePrimitive
|
|
role="region"
|
|
id={context.dropzoneId}
|
|
aria-controls={`${context.inputId} ${context.listId}`}
|
|
aria-disabled={context.disabled}
|
|
aria-invalid={invalid}
|
|
data-disabled={context.disabled ? '' : undefined}
|
|
data-dragging={dragOver ? '' : undefined}
|
|
data-invalid={invalid ? '' : undefined}
|
|
data-slot="file-upload-dropzone"
|
|
dir={context.dir}
|
|
tabIndex={context.disabled ? undefined : 0}
|
|
{...dropzoneProps}
|
|
className={cn(
|
|
'relative flex select-none flex-col items-center justify-center gap-2 rounded-lg border-2 border-dashed p-6 outline-none transition-colors hover:bg-accent/30 focus-visible:border-ring/50 data-[disabled]:pointer-events-none data-[dragging]:border-primary/30 data-[invalid]:border-destructive data-[dragging]:bg-accent/30 data-[invalid]:ring-destructive/20',
|
|
className,
|
|
)}
|
|
onClick={onClick}
|
|
onDragEnter={onDragEnter}
|
|
onDragLeave={onDragLeave}
|
|
onDragOver={onDragOver}
|
|
onDrop={onDrop}
|
|
onKeyDown={onKeyDown}
|
|
onPaste={onPaste}
|
|
/>
|
|
);
|
|
}
|
|
|
|
interface FileUploadTriggerProps
|
|
extends React.ComponentPropsWithoutRef<'button'> {
|
|
asChild?: boolean;
|
|
}
|
|
|
|
function FileUploadTrigger(props: FileUploadTriggerProps) {
|
|
const { asChild, onClick: onClickProp, ...triggerProps } = props;
|
|
const context = useFileUploadContext(TRIGGER_NAME);
|
|
|
|
const onClick = React.useCallback(
|
|
(event: React.MouseEvent<HTMLButtonElement>) => {
|
|
onClickProp?.(event);
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
context.inputRef.current?.click();
|
|
},
|
|
[context.inputRef, onClickProp],
|
|
);
|
|
|
|
const TriggerPrimitive = asChild ? Slot : 'button';
|
|
|
|
return (
|
|
<TriggerPrimitive
|
|
type="button"
|
|
aria-controls={context.inputId}
|
|
data-disabled={context.disabled ? '' : undefined}
|
|
data-slot="file-upload-trigger"
|
|
{...triggerProps}
|
|
disabled={context.disabled}
|
|
onClick={onClick}
|
|
/>
|
|
);
|
|
}
|
|
|
|
interface FileUploadListProps extends React.ComponentPropsWithoutRef<'div'> {
|
|
orientation?: 'horizontal' | 'vertical';
|
|
asChild?: boolean;
|
|
forceMount?: boolean;
|
|
}
|
|
|
|
function FileUploadList(props: FileUploadListProps) {
|
|
const {
|
|
className,
|
|
orientation = 'vertical',
|
|
asChild,
|
|
forceMount,
|
|
...listProps
|
|
} = props;
|
|
|
|
const context = useFileUploadContext(LIST_NAME);
|
|
const fileCount = useStore((state) => state.files.size);
|
|
const shouldRender = forceMount || fileCount > 0;
|
|
|
|
if (!shouldRender) return null;
|
|
|
|
const ListPrimitive = asChild ? Slot : 'div';
|
|
|
|
return (
|
|
<ListPrimitive
|
|
role="list"
|
|
id={context.listId}
|
|
aria-orientation={orientation}
|
|
data-orientation={orientation}
|
|
data-slot="file-upload-list"
|
|
data-state={shouldRender ? 'active' : 'inactive'}
|
|
dir={context.dir}
|
|
{...listProps}
|
|
className={cn(
|
|
'data-[state=inactive]:fade-out-0 data-[state=active]:fade-in-0 data-[state=inactive]:slide-out-to-top-2 data-[state=active]:slide-in-from-top-2 flex flex-col gap-2 data-[state=active]:animate-in data-[state=inactive]:animate-out',
|
|
orientation === 'horizontal' && 'flex-row overflow-x-auto p-1.5',
|
|
className,
|
|
)}
|
|
/>
|
|
);
|
|
}
|
|
|
|
interface FileUploadItemContextValue {
|
|
id: string;
|
|
fileState: FileState | undefined;
|
|
nameId: string;
|
|
sizeId: string;
|
|
statusId: string;
|
|
messageId: string;
|
|
}
|
|
|
|
const FileUploadItemContext =
|
|
React.createContext<FileUploadItemContextValue | null>(null);
|
|
|
|
function useFileUploadItemContext(consumerName: string) {
|
|
const context = React.useContext(FileUploadItemContext);
|
|
if (!context) {
|
|
throw new Error(`\`${consumerName}\` must be used within \`${ITEM_NAME}\``);
|
|
}
|
|
return context;
|
|
}
|
|
|
|
interface FileUploadItemProps extends React.ComponentPropsWithoutRef<'div'> {
|
|
value: File;
|
|
asChild?: boolean;
|
|
}
|
|
|
|
function FileUploadItem(props: FileUploadItemProps) {
|
|
const { value, asChild, className, ...itemProps } = props;
|
|
|
|
const id = React.useId();
|
|
const statusId = `${id}-status`;
|
|
const nameId = `${id}-name`;
|
|
const sizeId = `${id}-size`;
|
|
const messageId = `${id}-message`;
|
|
|
|
const context = useFileUploadContext(ITEM_NAME);
|
|
const fileState = useStore((state) => state.files.get(value));
|
|
const fileCount = useStore((state) => state.files.size);
|
|
const fileIndex = useStore((state) => {
|
|
const files = Array.from(state.files.keys());
|
|
return files.indexOf(value) + 1;
|
|
});
|
|
|
|
const itemContext = React.useMemo(
|
|
() => ({
|
|
id,
|
|
fileState,
|
|
nameId,
|
|
sizeId,
|
|
statusId,
|
|
messageId,
|
|
}),
|
|
[id, fileState, statusId, nameId, sizeId, messageId],
|
|
);
|
|
|
|
if (!fileState) return null;
|
|
|
|
const statusText = fileState.error
|
|
? `Error: ${fileState.error}`
|
|
: fileState.status === 'uploading'
|
|
? `Uploading: ${fileState.progress}% complete`
|
|
: fileState.status === 'success'
|
|
? 'Upload complete'
|
|
: 'Ready to upload';
|
|
|
|
const ItemPrimitive = asChild ? Slot : 'div';
|
|
|
|
return (
|
|
<FileUploadItemContext.Provider value={itemContext}>
|
|
<ItemPrimitive
|
|
role="listitem"
|
|
id={id}
|
|
aria-setsize={fileCount}
|
|
aria-posinset={fileIndex}
|
|
aria-describedby={`${nameId} ${sizeId} ${statusId} ${
|
|
fileState.error ? messageId : ''
|
|
}`}
|
|
aria-labelledby={nameId}
|
|
data-slot="file-upload-item"
|
|
dir={context.dir}
|
|
{...itemProps}
|
|
className={cn(
|
|
'relative flex items-center gap-2.5 rounded-md border p-3',
|
|
className,
|
|
)}
|
|
>
|
|
{props.children}
|
|
<span id={statusId} className="sr-only">
|
|
{statusText}
|
|
</span>
|
|
</ItemPrimitive>
|
|
</FileUploadItemContext.Provider>
|
|
);
|
|
}
|
|
|
|
function formatBytes(bytes: number) {
|
|
if (bytes === 0) return '0 B';
|
|
const sizes = ['B', 'KB', 'MB', 'GB', 'TB'];
|
|
const i = Math.floor(Math.log(bytes) / Math.log(1024));
|
|
return `${(bytes / 1024 ** i).toFixed(i ? 1 : 0)} ${sizes[i]}`;
|
|
}
|
|
|
|
function getFileIcon(file: File) {
|
|
const type = file.type;
|
|
const extension = file.name.split('.').pop()?.toLowerCase() ?? '';
|
|
|
|
if (type.startsWith('video/')) {
|
|
return <FileVideoIcon />;
|
|
}
|
|
|
|
if (type.startsWith('audio/')) {
|
|
return <FileAudioIcon />;
|
|
}
|
|
|
|
if (
|
|
type.startsWith('text/') ||
|
|
['txt', 'md', 'rtf', 'pdf'].includes(extension)
|
|
) {
|
|
return <FileTextIcon />;
|
|
}
|
|
|
|
if (
|
|
[
|
|
'html',
|
|
'css',
|
|
'js',
|
|
'jsx',
|
|
'ts',
|
|
'tsx',
|
|
'json',
|
|
'xml',
|
|
'php',
|
|
'py',
|
|
'rb',
|
|
'java',
|
|
'c',
|
|
'cpp',
|
|
'cs',
|
|
].includes(extension)
|
|
) {
|
|
return <FileCodeIcon />;
|
|
}
|
|
|
|
if (['zip', 'rar', '7z', 'tar', 'gz', 'bz2'].includes(extension)) {
|
|
return <FileArchiveIcon />;
|
|
}
|
|
|
|
if (
|
|
['exe', 'msi', 'app', 'apk', 'deb', 'rpm'].includes(extension) ||
|
|
type.startsWith('application/')
|
|
) {
|
|
return <FileCogIcon />;
|
|
}
|
|
|
|
return <FileIcon />;
|
|
}
|
|
|
|
interface FileUploadItemPreviewProps
|
|
extends React.ComponentPropsWithoutRef<'div'> {
|
|
render?: (file: File) => React.ReactNode;
|
|
asChild?: boolean;
|
|
}
|
|
|
|
function FileUploadItemPreview(props: FileUploadItemPreviewProps) {
|
|
const { render, asChild, children, className, ...previewProps } = props;
|
|
|
|
const itemContext = useFileUploadItemContext(ITEM_PREVIEW_NAME);
|
|
const context = useFileUploadContext(ITEM_PREVIEW_NAME);
|
|
|
|
const onPreviewRender = React.useCallback(
|
|
(file: File) => {
|
|
if (render) return render(file);
|
|
|
|
if (itemContext.fileState?.file.type.startsWith('image/')) {
|
|
let url = context.urlCache.get(file);
|
|
if (!url) {
|
|
url = URL.createObjectURL(file);
|
|
context.urlCache.set(file, url);
|
|
}
|
|
|
|
return (
|
|
<img src={url} alt={file.name} className="size-full object-cover" />
|
|
);
|
|
}
|
|
|
|
return getFileIcon(file);
|
|
},
|
|
[render, itemContext.fileState?.file.type, context.urlCache],
|
|
);
|
|
|
|
if (!itemContext.fileState) return null;
|
|
|
|
const ItemPreviewPrimitive = asChild ? Slot : 'div';
|
|
|
|
return (
|
|
<ItemPreviewPrimitive
|
|
aria-labelledby={itemContext.nameId}
|
|
data-slot="file-upload-preview"
|
|
{...previewProps}
|
|
className={cn(
|
|
'relative flex size-10 shrink-0 items-center justify-center overflow-hidden rounded border bg-accent/50 [&>svg]:size-10',
|
|
className,
|
|
)}
|
|
>
|
|
{onPreviewRender(itemContext.fileState.file)}
|
|
{children}
|
|
</ItemPreviewPrimitive>
|
|
);
|
|
}
|
|
|
|
interface FileUploadItemMetadataProps
|
|
extends React.ComponentPropsWithoutRef<'div'> {
|
|
asChild?: boolean;
|
|
size?: 'default' | 'sm';
|
|
}
|
|
|
|
function FileUploadItemMetadata(props: FileUploadItemMetadataProps) {
|
|
const {
|
|
asChild,
|
|
size = 'default',
|
|
children,
|
|
className,
|
|
...metadataProps
|
|
} = props;
|
|
|
|
const context = useFileUploadContext(ITEM_METADATA_NAME);
|
|
const itemContext = useFileUploadItemContext(ITEM_METADATA_NAME);
|
|
|
|
if (!itemContext.fileState) return null;
|
|
|
|
const ItemMetadataPrimitive = asChild ? Slot : 'div';
|
|
|
|
return (
|
|
<ItemMetadataPrimitive
|
|
data-slot="file-upload-metadata"
|
|
dir={context.dir}
|
|
{...metadataProps}
|
|
className={cn('flex min-w-0 flex-1 flex-col', className)}
|
|
>
|
|
{children ?? (
|
|
<>
|
|
<span
|
|
id={itemContext.nameId}
|
|
className={cn(
|
|
'truncate font-medium text-sm',
|
|
size === 'sm' && 'font-normal text-[13px] leading-snug',
|
|
)}
|
|
>
|
|
{itemContext.fileState.file.name}
|
|
</span>
|
|
<span
|
|
id={itemContext.sizeId}
|
|
className={cn(
|
|
'truncate text-muted-foreground text-xs',
|
|
size === 'sm' && 'text-[11px] leading-snug',
|
|
)}
|
|
>
|
|
{formatBytes(itemContext.fileState.file.size)}
|
|
</span>
|
|
{itemContext.fileState.error && (
|
|
<span
|
|
id={itemContext.messageId}
|
|
className="text-destructive text-xs"
|
|
>
|
|
{itemContext.fileState.error}
|
|
</span>
|
|
)}
|
|
</>
|
|
)}
|
|
</ItemMetadataPrimitive>
|
|
);
|
|
}
|
|
interface FileUploadItemProgressProps
|
|
extends React.ComponentPropsWithoutRef<'div'> {
|
|
variant?: 'linear' | 'circular' | 'fill';
|
|
size?: number;
|
|
asChild?: boolean;
|
|
forceMount?: boolean;
|
|
}
|
|
|
|
function FileUploadItemProgress(props: FileUploadItemProgressProps) {
|
|
const {
|
|
variant = 'linear',
|
|
size = 40,
|
|
asChild,
|
|
forceMount,
|
|
className,
|
|
...progressProps
|
|
} = props;
|
|
|
|
const itemContext = useFileUploadItemContext(ITEM_PROGRESS_NAME);
|
|
|
|
if (!itemContext.fileState) return null;
|
|
|
|
const shouldRender = forceMount || itemContext.fileState.progress !== 100;
|
|
|
|
if (!shouldRender) return null;
|
|
|
|
const ItemProgressPrimitive = asChild ? Slot : 'div';
|
|
|
|
switch (variant) {
|
|
case 'circular': {
|
|
const circumference = 2 * Math.PI * ((size - 4) / 2);
|
|
const strokeDashoffset =
|
|
circumference - (itemContext.fileState.progress / 100) * circumference;
|
|
|
|
return (
|
|
<ItemProgressPrimitive
|
|
role="progressbar"
|
|
aria-valuemin={0}
|
|
aria-valuemax={100}
|
|
aria-valuenow={itemContext.fileState.progress}
|
|
aria-valuetext={`${itemContext.fileState.progress}%`}
|
|
aria-labelledby={itemContext.nameId}
|
|
data-slot="file-upload-progress"
|
|
{...progressProps}
|
|
className={cn(
|
|
'-translate-x-1/2 -translate-y-1/2 absolute top-1/2 left-1/2',
|
|
className,
|
|
)}
|
|
>
|
|
<svg
|
|
className="rotate-[-90deg] transform"
|
|
width={size}
|
|
height={size}
|
|
viewBox={`0 0 ${size} ${size}`}
|
|
fill="none"
|
|
stroke="currentColor"
|
|
>
|
|
<circle
|
|
className="text-primary/20"
|
|
strokeWidth="2"
|
|
cx={size / 2}
|
|
cy={size / 2}
|
|
r={(size - 4) / 2}
|
|
/>
|
|
<circle
|
|
className="text-primary transition-[stroke-dashoffset] duration-300 ease-linear"
|
|
strokeWidth="2"
|
|
strokeLinecap="round"
|
|
strokeDasharray={circumference}
|
|
strokeDashoffset={strokeDashoffset}
|
|
cx={size / 2}
|
|
cy={size / 2}
|
|
r={(size - 4) / 2}
|
|
/>
|
|
</svg>
|
|
</ItemProgressPrimitive>
|
|
);
|
|
}
|
|
|
|
case 'fill': {
|
|
const progressPercentage = itemContext.fileState.progress;
|
|
const topInset = 100 - progressPercentage;
|
|
|
|
return (
|
|
<ItemProgressPrimitive
|
|
role="progressbar"
|
|
aria-valuemin={0}
|
|
aria-valuemax={100}
|
|
aria-valuenow={progressPercentage}
|
|
aria-valuetext={`${progressPercentage}%`}
|
|
aria-labelledby={itemContext.nameId}
|
|
data-slot="file-upload-progress"
|
|
{...progressProps}
|
|
className={cn(
|
|
'absolute inset-0 bg-primary/50 transition-[clip-path] duration-300 ease-linear',
|
|
className,
|
|
)}
|
|
style={{
|
|
clipPath: `inset(${topInset}% 0% 0% 0%)`,
|
|
}}
|
|
/>
|
|
);
|
|
}
|
|
|
|
default:
|
|
return (
|
|
<ItemProgressPrimitive
|
|
role="progressbar"
|
|
aria-valuemin={0}
|
|
aria-valuemax={100}
|
|
aria-valuenow={itemContext.fileState.progress}
|
|
aria-valuetext={`${itemContext.fileState.progress}%`}
|
|
aria-labelledby={itemContext.nameId}
|
|
data-slot="file-upload-progress"
|
|
{...progressProps}
|
|
className={cn(
|
|
'relative h-1.5 w-full overflow-hidden rounded-full bg-primary/20',
|
|
className,
|
|
)}
|
|
>
|
|
<div
|
|
className="h-full w-full flex-1 bg-primary transition-transform duration-300 ease-linear"
|
|
style={{
|
|
transform: `translateX(-${100 - itemContext.fileState.progress}%)`,
|
|
}}
|
|
/>
|
|
</ItemProgressPrimitive>
|
|
);
|
|
}
|
|
}
|
|
|
|
interface FileUploadItemDeleteProps
|
|
extends React.ComponentPropsWithoutRef<'button'> {
|
|
asChild?: boolean;
|
|
}
|
|
|
|
function FileUploadItemDelete(props: FileUploadItemDeleteProps) {
|
|
const { asChild, onClick: onClickProp, ...deleteProps } = props;
|
|
|
|
const store = useStoreContext(ITEM_DELETE_NAME);
|
|
const itemContext = useFileUploadItemContext(ITEM_DELETE_NAME);
|
|
|
|
const onClick = React.useCallback(
|
|
(event: React.MouseEvent<HTMLButtonElement>) => {
|
|
onClickProp?.(event);
|
|
|
|
if (!itemContext.fileState || event.defaultPrevented) return;
|
|
|
|
store.dispatch({
|
|
type: 'REMOVE_FILE',
|
|
file: itemContext.fileState.file,
|
|
});
|
|
},
|
|
[store, itemContext.fileState, onClickProp],
|
|
);
|
|
|
|
if (!itemContext.fileState) return null;
|
|
|
|
const ItemDeletePrimitive = asChild ? Slot : 'button';
|
|
|
|
return (
|
|
<ItemDeletePrimitive
|
|
type="button"
|
|
aria-controls={itemContext.id}
|
|
aria-describedby={itemContext.nameId}
|
|
data-slot="file-upload-item-delete"
|
|
{...deleteProps}
|
|
onClick={onClick}
|
|
/>
|
|
);
|
|
}
|
|
|
|
interface FileUploadClearProps
|
|
extends React.ComponentPropsWithoutRef<'button'> {
|
|
forceMount?: boolean;
|
|
asChild?: boolean;
|
|
}
|
|
|
|
function FileUploadClear(props: FileUploadClearProps) {
|
|
const {
|
|
asChild,
|
|
forceMount,
|
|
disabled,
|
|
onClick: onClickProp,
|
|
...clearProps
|
|
} = props;
|
|
|
|
const context = useFileUploadContext(CLEAR_NAME);
|
|
const store = useStoreContext(CLEAR_NAME);
|
|
const fileCount = useStore((state) => state.files.size);
|
|
|
|
const isDisabled = disabled || context.disabled;
|
|
|
|
const onClick = React.useCallback(
|
|
(event: React.MouseEvent<HTMLButtonElement>) => {
|
|
onClickProp?.(event);
|
|
|
|
if (event.defaultPrevented) return;
|
|
|
|
store.dispatch({ type: 'CLEAR' });
|
|
},
|
|
[store, onClickProp],
|
|
);
|
|
|
|
const shouldRender = forceMount || fileCount > 0;
|
|
|
|
if (!shouldRender) return null;
|
|
|
|
const ClearPrimitive = asChild ? Slot : 'button';
|
|
|
|
return (
|
|
<ClearPrimitive
|
|
type="button"
|
|
aria-controls={context.listId}
|
|
data-slot="file-upload-clear"
|
|
data-disabled={isDisabled ? '' : undefined}
|
|
{...clearProps}
|
|
disabled={isDisabled}
|
|
onClick={onClick}
|
|
/>
|
|
);
|
|
}
|
|
|
|
export {
|
|
FileUploadClear as Clear,
|
|
FileUploadDropzone as Dropzone,
|
|
FileUploadRoot as FileUpload,
|
|
FileUploadClear,
|
|
FileUploadDropzone,
|
|
FileUploadItem,
|
|
FileUploadItemDelete,
|
|
FileUploadItemMetadata,
|
|
FileUploadItemPreview,
|
|
FileUploadItemProgress,
|
|
FileUploadList,
|
|
FileUploadTrigger,
|
|
FileUploadItem as Item,
|
|
FileUploadItemDelete as ItemDelete,
|
|
FileUploadItemMetadata as ItemMetadata,
|
|
FileUploadItemPreview as ItemPreview,
|
|
FileUploadItemProgress as ItemProgress,
|
|
FileUploadList as List,
|
|
//
|
|
FileUploadRoot as Root,
|
|
FileUploadTrigger as Trigger,
|
|
//
|
|
useStore as useFileUpload,
|
|
//
|
|
type FileUploadRootProps as FileUploadProps,
|
|
};
|