160 lines
4.1 KiB
TypeScript
160 lines
4.1 KiB
TypeScript
|
import Uppy, { State, debugLogger } from "@uppy/core"
|
||
|
|
||
|
import Tus from "@uppy/tus"
|
||
|
import toArray from "@uppy/utils/lib/toArray"
|
||
|
|
||
|
import {
|
||
|
ChangeEvent,
|
||
|
useCallback,
|
||
|
useEffect,
|
||
|
useMemo,
|
||
|
useRef,
|
||
|
useState
|
||
|
} from "react"
|
||
|
import DropTarget, { DropTargetOptions } from "./uppy-dropzone"
|
||
|
|
||
|
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),
|
||
|
[]
|
||
|
)
|
||
|
const [state, setState] = useState<State>()
|
||
|
|
||
|
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(
|
||
|
() => uppyInstance.current?.cancelAll(),
|
||
|
// 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 üòÄ")
|
||
|
} else {
|
||
|
console.warn("Upload failed üòû")
|
||
|
}
|
||
|
console.log("successful files:", result.successful)
|
||
|
console.log("failed files:", result.failed)
|
||
|
})
|
||
|
|
||
|
const setStateCb = () => {
|
||
|
setState(uppy.getState())
|
||
|
}
|
||
|
|
||
|
for (const event of LISTENING_EVENTS) {
|
||
|
uppy.on(event, setStateCb)
|
||
|
}
|
||
|
|
||
|
return () => {
|
||
|
for (const event of ["complete", ...LISTENING_EVENTS]) {
|
||
|
uppyInstance.current?.off(
|
||
|
event as "complete" & keyof typeof LISTENING_EVENTS,
|
||
|
//@ts-expect-error -- huh? typescript wtf
|
||
|
setStateCb
|
||
|
)
|
||
|
}
|
||
|
uppyInstance.current?.close()
|
||
|
uppyInstance.current = undefined
|
||
|
}
|
||
|
}, [targetRef, endpoint, uploader])
|
||
|
|
||
|
return {
|
||
|
files: uppyInstance.current?.getFiles() ?? [],
|
||
|
error: uppyInstance.current?.getState,
|
||
|
state,
|
||
|
upload: () =>
|
||
|
uppyInstance.current?.upload() ??
|
||
|
new Error("[useUppy] Uppy has not initialized yet."),
|
||
|
getInputProps: () => inputProps,
|
||
|
getRootProps,
|
||
|
removeFile,
|
||
|
cancelAll,
|
||
|
}
|
||
|
}
|