2024-03-13 17:07:03 +00:00
|
|
|
import Uppy, { type State, debugLogger } from "@uppy/core"
|
2024-03-12 12:42:17 +00:00
|
|
|
|
|
|
|
import Tus from "@uppy/tus"
|
|
|
|
import toArray from "@uppy/utils/lib/toArray"
|
|
|
|
|
|
|
|
import {
|
2024-03-13 17:07:03 +00:00
|
|
|
type ChangeEvent,
|
2024-03-12 12:42:17 +00:00
|
|
|
useCallback,
|
|
|
|
useEffect,
|
|
|
|
useMemo,
|
|
|
|
useRef,
|
|
|
|
useState
|
|
|
|
} from "react"
|
2024-03-13 17:07:03 +00:00
|
|
|
import DropTarget, { type DropTargetOptions } from "./uppy-dropzone"
|
2024-03-12 12:42:17 +00:00
|
|
|
|
|
|
|
const LISTENING_EVENTS = [
|
|
|
|
"upload",
|
|
|
|
"upload-success",
|
|
|
|
"upload-error",
|
|
|
|
"file-added",
|
|
|
|
"file-removed",
|
|
|
|
"files-added"
|
|
|
|
] as const
|
|
|
|
|
|
|
|
export function useUppy({
|
|
|
|
uploader,
|
|
|
|
endpoint
|
|
|
|
}: {
|
|
|
|
uploader: "tus"
|
|
|
|
endpoint: string
|
|
|
|
}) {
|
|
|
|
const inputRef = useRef<HTMLInputElement>(null)
|
|
|
|
const [targetRef, _setTargetRef] = useState<HTMLElement | null>(null)
|
|
|
|
const uppyInstance = useRef<Uppy>()
|
|
|
|
const setRef = useCallback(
|
|
|
|
(element: HTMLElement | null) => _setTargetRef(element),
|
|
|
|
[]
|
|
|
|
)
|
2024-03-13 17:07:03 +00:00
|
|
|
const [, setUppyState] = useState<State>()
|
2024-03-12 14:32:00 +00:00
|
|
|
const [state, setState] = useState<
|
|
|
|
"completed" | "idle" | "initializing" | "error" | "uploading"
|
|
|
|
>("initializing")
|
2024-03-12 12:42:17 +00:00
|
|
|
|
|
|
|
const [inputProps, setInputProps] = useState<
|
|
|
|
| {
|
|
|
|
ref: typeof inputRef
|
|
|
|
type: "file"
|
|
|
|
onChange: (event: ChangeEvent<HTMLInputElement>) => void
|
|
|
|
}
|
|
|
|
| object
|
|
|
|
>({})
|
|
|
|
const getRootProps = useMemo(
|
|
|
|
() => () => {
|
|
|
|
return {
|
|
|
|
ref: setRef,
|
|
|
|
onClick: () => {
|
|
|
|
if (inputRef.current) {
|
|
|
|
//@ts-expect-error -- dumb html
|
|
|
|
inputRef.current.value = null
|
|
|
|
inputRef.current.click()
|
|
|
|
console.log("clicked", { input: inputRef.current })
|
|
|
|
}
|
|
|
|
},
|
|
|
|
role: "presentation"
|
|
|
|
}
|
|
|
|
},
|
|
|
|
[setRef]
|
|
|
|
)
|
|
|
|
const removeFile = useCallback(
|
|
|
|
(id: string) => {
|
|
|
|
uppyInstance.current?.removeFile(id)
|
|
|
|
},
|
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
|
|
[targetRef, uppyInstance]
|
|
|
|
)
|
|
|
|
const cancelAll = useCallback(
|
2024-03-12 14:32:00 +00:00
|
|
|
() => uppyInstance.current?.cancelAll({ reason: "user" }),
|
2024-03-12 12:42:17 +00:00
|
|
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
|
|
|
[targetRef, uppyInstance]
|
|
|
|
)
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
if (!targetRef) return
|
|
|
|
|
|
|
|
const uppy = new Uppy({ logger: debugLogger }).use(DropTarget, {
|
|
|
|
target: targetRef
|
|
|
|
} as DropTargetOptions)
|
|
|
|
|
|
|
|
uppyInstance.current = uppy
|
|
|
|
setInputProps({
|
|
|
|
ref: inputRef,
|
|
|
|
type: "file",
|
|
|
|
onChange: (event) => {
|
|
|
|
const files = toArray(event.target.files)
|
|
|
|
if (files.length > 0) {
|
|
|
|
uppyInstance.current?.log("[DragDrop] Files selected through input")
|
|
|
|
uppyInstance.current?.addFiles(files)
|
|
|
|
}
|
|
|
|
|
|
|
|
// We clear the input after a file is selected, because otherwise
|
|
|
|
// change event is not fired in Chrome and Safari when a file
|
|
|
|
// with the same name is selected.
|
|
|
|
// ___Why not use value="" on <input/> instead?
|
|
|
|
// Because if we use that method of clearing the input,
|
|
|
|
// Chrome will not trigger change if we drop the same file twice (Issue #768).
|
|
|
|
// @ts-expect-error TS freaks out, but this is fine
|
|
|
|
// eslint-disable-next-line no-param-reassign
|
|
|
|
event.target.value = null
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
switch (uploader) {
|
|
|
|
case "tus":
|
|
|
|
uppy.use(Tus, { endpoint: endpoint, limit: 6 })
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
}
|
|
|
|
|
|
|
|
uppy.on("complete", (result) => {
|
|
|
|
if (result.failed.length === 0) {
|
|
|
|
console.log("Upload successful üòÄ")
|
2024-03-12 14:32:00 +00:00
|
|
|
setState("completed")
|
2024-03-12 12:42:17 +00:00
|
|
|
} else {
|
|
|
|
console.warn("Upload failed üòû")
|
2024-03-12 14:32:00 +00:00
|
|
|
setState("error")
|
2024-03-12 12:42:17 +00:00
|
|
|
}
|
|
|
|
console.log("successful files:", result.successful)
|
|
|
|
console.log("failed files:", result.failed)
|
|
|
|
})
|
|
|
|
|
2024-03-12 14:32:00 +00:00
|
|
|
const setStateCb = (event: (typeof LISTENING_EVENTS)[number]) => {
|
|
|
|
switch (event) {
|
|
|
|
case "upload":
|
|
|
|
setState("uploading")
|
|
|
|
break
|
|
|
|
case "upload-error":
|
|
|
|
setState("error")
|
|
|
|
break
|
|
|
|
default:
|
|
|
|
break
|
|
|
|
}
|
|
|
|
setUppyState(uppy.getState())
|
2024-03-12 12:42:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (const event of LISTENING_EVENTS) {
|
2024-03-12 14:32:00 +00:00
|
|
|
uppy.on(event, function cb() {
|
|
|
|
setStateCb(event)
|
|
|
|
})
|
2024-03-12 12:42:17 +00:00
|
|
|
}
|
2024-03-12 14:32:00 +00:00
|
|
|
setState("idle")
|
2024-03-12 14:36:44 +00:00
|
|
|
}, [targetRef, endpoint, uploader])
|
2024-03-12 12:42:17 +00:00
|
|
|
|
2024-03-12 14:36:44 +00:00
|
|
|
useEffect(() => {
|
2024-03-12 12:42:17 +00:00
|
|
|
return () => {
|
2024-03-12 14:36:44 +00:00
|
|
|
uppyInstance.current?.cancelAll({ reason: "unmount" })
|
|
|
|
uppyInstance.current?.logout()
|
|
|
|
uppyInstance.current?.close()
|
|
|
|
uppyInstance.current = undefined
|
2024-03-12 12:42:17 +00:00
|
|
|
}
|
2024-03-12 14:36:44 +00:00
|
|
|
}, [])
|
2024-03-12 12:42:17 +00:00
|
|
|
return {
|
2024-03-12 14:32:00 +00:00
|
|
|
getFiles: () => uppyInstance.current?.getFiles() ?? [],
|
2024-03-12 12:42:17 +00:00
|
|
|
error: uppyInstance.current?.getState,
|
|
|
|
state,
|
|
|
|
upload: () =>
|
|
|
|
uppyInstance.current?.upload() ??
|
|
|
|
new Error("[useUppy] Uppy has not initialized yet."),
|
|
|
|
getInputProps: () => inputProps,
|
|
|
|
getRootProps,
|
|
|
|
removeFile,
|
2024-03-12 14:32:00 +00:00
|
|
|
cancelAll
|
2024-03-12 12:42:17 +00:00
|
|
|
}
|
|
|
|
}
|