tomodachi-share/frontend/src/components/submit-form/switch-file-upload.tsx
trafficlunar 817bda4993 feat: .ltd files
no automatic instructions
2026-04-26 22:49:29 +01:00

83 lines
2.3 KiB
TypeScript

import { useCallback, useState } from "react";
import { type FileWithPath } from "react-dropzone";
import { Icon } from "@iconify/react";
import Dropzone from "../dropzone";
import Camera from "./camera";
import ImageEditorPortrait from "./image-editor";
interface Props {
text: string;
type?: "file" | "image";
forceCrop?: boolean;
file?: string | File | undefined;
setFile?: (value: File | undefined) => void;
image?: string | undefined;
setImage?: (value: string | undefined) => void;
}
export default function SwitchFileUpload({ text, type = "image", forceCrop, file, setFile, image, setImage }: Props) {
const [isCameraOpen, setIsCameraOpen] = useState(false);
const [isCropOpen, setIsCropOpen] = useState(false);
const handleDrop = useCallback(
(acceptedFiles: FileWithPath[]) => {
const file = acceptedFiles[0];
if (type === "file") {
setFile!(file);
} else {
const reader = new FileReader();
reader.onload = (event) => {
setImage!(event.target!.result as string);
if (forceCrop) setIsCropOpen(true);
};
reader.readAsDataURL(file);
}
},
[setFile, setImage],
);
return (
<div className="max-w-md w-full flex flex-col items-center gap-2">
<Dropzone type={type} onDrop={handleDrop} options={{ maxFiles: 1 }}>
<p className="text-center text-sm">
{!file && !image ? (
<>
Drag and drop {text}
<br />
or click to open
</>
) : (
"Uploaded!"
)}
</p>
</Dropzone>
{type === "image" && (
<>
<span>or</span>
<div className="flex gap-2 max-sm:flex-col">
<button type="button" aria-label="Use your camera" onClick={() => setIsCameraOpen(true)} className="pill button gap-2">
<Icon icon="mdi:camera" fontSize={20} />
Use your camera
</button>
<button type="button" aria-label="Crop image" onClick={() => setIsCropOpen(true)} className="pill button gap-2">
<Icon icon="mdi:image-edit" fontSize={20} />
Edit Image
</button>
</div>
<Camera
isOpen={isCameraOpen}
setIsOpen={setIsCameraOpen}
setImage={setImage}
onCapture={() => {
if (forceCrop) setIsCropOpen(true);
}}
/>
<ImageEditorPortrait isOpen={isCropOpen} setIsOpen={setIsCropOpen} image={image} setImage={setImage!} />
</>
)}
</div>
);
}