Files
memos/web/src/components/MemoEditor/ActionButton/UploadResourceButton.tsx

86 lines
2.3 KiB
TypeScript

import { IconButton } from "@mui/joy";
import { PaperclipIcon } from "lucide-react";
import { useContext, useRef, useState } from "react";
import toast from "react-hot-toast";
import { useResourceStore } from "@/store/v1";
import { Resource } from "@/types/proto/api/v1/resource_service";
import { MemoEditorContext } from "../types";
interface State {
uploadingFlag: boolean;
}
const UploadResourceButton = () => {
const context = useContext(MemoEditorContext);
const resourceStore = useResourceStore();
const [state, setState] = useState<State>({
uploadingFlag: false,
});
const fileInputRef = useRef<HTMLInputElement>(null);
const handleFileInputChange = async () => {
if (!fileInputRef.current || !fileInputRef.current.files || fileInputRef.current.files.length === 0) {
return;
}
if (state.uploadingFlag) {
return;
}
setState((state) => {
return {
...state,
uploadingFlag: true,
};
});
const createdResourceList: Resource[] = [];
try {
if (!fileInputRef.current || !fileInputRef.current.files) {
return;
}
for (const file of fileInputRef.current.files) {
const { name: filename, size, type } = file;
const buffer = new Uint8Array(await file.arrayBuffer());
const resource = await resourceStore.createResource({
resource: Resource.fromPartial({
filename,
size,
type,
content: buffer,
}),
});
createdResourceList.push(resource);
}
} catch (error: any) {
console.error(error);
toast.error(error.details);
}
context.setResourceList([...context.resourceList, ...createdResourceList]);
setState((state) => {
return {
...state,
uploadingFlag: false,
};
});
};
return (
<IconButton size="sm" disabled={state.uploadingFlag}>
<PaperclipIcon className="w-5 h-5 mx-auto" />
<input
className="absolute inset-0 w-full h-full opacity-0 cursor-pointer"
ref={fileInputRef}
disabled={state.uploadingFlag}
onChange={handleFileInputChange}
type="file"
id="files"
multiple={true}
accept="*"
/>
</IconButton>
);
};
export default UploadResourceButton;