An area that accepts file drops and uploads.
"use client";
import { FileUpload as FileUploadPrimitives } from "@ark-ui/react";
import { IconTrash } from "@tabler/icons-react";
import React, { forwardRef } from "react";
const Dropzone = forwardRef<
React.ElementRef<typeof FileUploadPrimitives.Root>,
React.ComponentPropsWithoutRef<typeof FileUploadPrimitives.Root>
>(({ children, ...props }, ref) => {
return (
<FileUploadPrimitives.Root
ref={ref}
className="relative aspect-video w-full"
{...props}
>
<FileUploadPrimitives.Dropzone className="border-base-300 mb-4 flex h-full w-full flex-col items-center justify-center gap-y-4 rounded-sm border pb-5">
{children}
</FileUploadPrimitives.Dropzone>
<FileUploadPrimitives.HiddenInput className="" />
<FileUploadPrimitives.ItemGroup className="relative space-y-2">
<FileUploadPrimitives.Context>
{({ acceptedFiles }) =>
acceptedFiles.map((file, index) => (
<FileUploadPrimitives.Item
key={index}
file={file}
className="border-base-300 flex items-center gap-x-4 rounded-sm border p-4"
>
{file.type.startsWith("image") ? (
<FileUploadPrimitives.ItemPreview type="image/*" className="">
<FileUploadPrimitives.ItemPreviewImage
src={URL.createObjectURL(file)}
className="h-12 w-12 object-contain object-center"
/>
</FileUploadPrimitives.ItemPreview>
) : (
<FileUploadPrimitives.ItemPreview
type=".*"
className="border-base-300 bg-base-200 text-base-500 flex h-12 w-12 items-center justify-center rounded border text-sm font-semibold uppercase"
>
{file.name.split(".").pop()}
</FileUploadPrimitives.ItemPreview>
)}
<div className="flex flex-1 flex-col">
<FileUploadPrimitives.ItemName className="text-base-800 text-sm font-medium" />
<FileUploadPrimitives.ItemSizeText className="text-base-500 text-sm" />
</div>
<FileUploadPrimitives.ItemDeleteTrigger className="border-base-300 hover:bg-base-200 rounded border p-1.5 transition-colors">
<IconTrash className="text-base-600 h-4 w-4" />
</FileUploadPrimitives.ItemDeleteTrigger>
</FileUploadPrimitives.Item>
))
}
</FileUploadPrimitives.Context>
</FileUploadPrimitives.ItemGroup>
</FileUploadPrimitives.Root>
);
});
Dropzone.displayName = "Dropzone";
const DropzoneText = forwardRef<
HTMLDivElement,
React.HTMLAttributes<HTMLDivElement>
>(({ children, ...props }, ref) => {
return (
<div ref={ref} className="text-base-600 text-lg" {...props}>
{children}
</div>
);
});
DropzoneText.displayName = "DropzoneText";
const DropzoneTrigger = FileUploadPrimitives.Trigger;
export { Dropzone, DropzoneText, DropzoneTrigger };
import { Button } from "../core/button";
import { Dropzone, DropzoneText, DropzoneTrigger } from "../core/dropzone";
export const DropzoneExample = () => {
return (
<Dropzone maxFiles={5}>
<DropzoneText>Drop your file here</DropzoneText>
<DropzoneTrigger asChild>
<Button size="sm">Select file</Button>
</DropzoneTrigger>
</Dropzone>
);
};