2022-05-28 19:52:48 +02:00
|
|
|
import { h, Fragment } from 'preact';
|
|
|
|
import { useCallback, useEffect, useMemo, useRef, useState } from 'preact/hooks';
|
2022-05-31 17:38:40 +02:00
|
|
|
import { noAwait } from './util/promises';
|
2022-05-28 19:52:48 +02:00
|
|
|
|
|
|
|
export interface FilePickerAcceptType {
|
|
|
|
description?: string | undefined;
|
|
|
|
accept: Record<string, string | string[]>;
|
|
|
|
}
|
|
|
|
|
|
|
|
const ACCEPT_EVERYTHING_TYPE: FilePickerAcceptType = {
|
|
|
|
description: 'Any file',
|
|
|
|
accept: { '*/*': [] },
|
|
|
|
};
|
|
|
|
|
|
|
|
export interface FileChooserProps {
|
|
|
|
disabled?: boolean;
|
2022-06-03 19:12:30 +02:00
|
|
|
onChange: (handles: Array<FileSystemFileHandle>) => void;
|
2022-05-28 19:52:48 +02:00
|
|
|
accept?: FilePickerAcceptType[];
|
|
|
|
control?: typeof controlDefault;
|
|
|
|
}
|
|
|
|
|
2022-05-31 17:41:24 -07:00
|
|
|
const hasPicker = !!window.showOpenFilePicker;
|
2022-05-28 19:52:48 +02:00
|
|
|
const controlDefault = hasPicker ? 'picker' : 'input';
|
|
|
|
|
|
|
|
interface InputFileChooserProps {
|
|
|
|
disabled?: boolean;
|
|
|
|
onChange?: (files: FileList) => void;
|
|
|
|
accept?: FilePickerAcceptType[];
|
|
|
|
}
|
|
|
|
|
|
|
|
interface ExtraProps {
|
|
|
|
accept?: string;
|
|
|
|
}
|
|
|
|
|
|
|
|
const InputFileChooser = ({
|
|
|
|
disabled = false,
|
2022-05-31 17:38:40 +02:00
|
|
|
onChange = () => { /* do nothing */ },
|
2022-05-28 19:52:48 +02:00
|
|
|
accept = [],
|
|
|
|
}: InputFileChooserProps) => {
|
|
|
|
const inputRef = useRef<HTMLInputElement>(null);
|
|
|
|
const filesRef = useRef<FileList>();
|
|
|
|
|
|
|
|
const onChangeInternal = useCallback(() => {
|
|
|
|
if (inputRef.current?.files) {
|
|
|
|
const newFiles = inputRef.current?.files;
|
|
|
|
if (filesRef.current !== newFiles) {
|
|
|
|
filesRef.current = newFiles;
|
|
|
|
onChange(newFiles);
|
|
|
|
}
|
|
|
|
}
|
2022-05-29 13:48:51 -07:00
|
|
|
}, [onChange]);
|
2022-05-28 19:52:48 +02:00
|
|
|
|
|
|
|
const extraProps = useMemo<ExtraProps>(() => {
|
|
|
|
// Accept all of the given MIME types and extensions. An argument
|
|
|
|
// could be made to throw out all of the MIME types and just keep
|
|
|
|
// the extensions, but this seemed to follow the intent in the
|
|
|
|
// spec:
|
|
|
|
//
|
|
|
|
// https://html.spec.whatwg.org/multipage/input.html#file-upload-state-(type=file)
|
|
|
|
//
|
|
|
|
// which allows pretty generous interpretations.
|
|
|
|
//
|
|
|
|
const newAccept = [];
|
|
|
|
for (const type of accept) {
|
|
|
|
for (const [typeString, suffixes] of Object.entries(type.accept)) {
|
|
|
|
newAccept.push(typeString);
|
|
|
|
if (Array.isArray(suffixes)) {
|
|
|
|
newAccept.push(...suffixes);
|
|
|
|
} else {
|
|
|
|
newAccept.push(suffixes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const extraProps: { accept?: string } = {};
|
|
|
|
if (newAccept.length > 0) {
|
|
|
|
extraProps['accept'] = newAccept.join(',');
|
|
|
|
}
|
|
|
|
|
|
|
|
return extraProps;
|
|
|
|
}, [accept]);
|
|
|
|
|
|
|
|
return (
|
|
|
|
<input type="file" role='button' aria-label='Open file'
|
|
|
|
ref={inputRef}
|
|
|
|
onChange={onChangeInternal}
|
|
|
|
disabled={disabled}
|
|
|
|
{...extraProps} />
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
interface FilePickerChooserProps {
|
|
|
|
disabled?: boolean;
|
|
|
|
onChange?: (files: FileSystemFileHandle[]) => void;
|
|
|
|
accept?: FilePickerAcceptType[];
|
|
|
|
}
|
|
|
|
|
|
|
|
const FilePickerChooser = ({
|
|
|
|
disabled = false,
|
2022-05-31 17:38:40 +02:00
|
|
|
onChange = () => { /* do nothing */ },
|
2022-05-28 19:52:48 +02:00
|
|
|
accept = [ACCEPT_EVERYTHING_TYPE]
|
|
|
|
}: FilePickerChooserProps) => {
|
|
|
|
const [busy, setBusy] = useState<boolean>(false);
|
|
|
|
const [selectedFilename, setSelectedFilename] = useState<string>();
|
2022-06-04 11:06:38 -07:00
|
|
|
const [fileHandles, setFileHandles] = useState<FileSystemFileHandle[]>();
|
2022-05-28 19:52:48 +02:00
|
|
|
|
|
|
|
const onClickInternal = useCallback(async () => {
|
|
|
|
if (busy) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setBusy(true);
|
|
|
|
try {
|
|
|
|
const pickedFiles = await window.showOpenFilePicker({
|
|
|
|
multiple: false,
|
|
|
|
excludeAcceptAllOption: true,
|
|
|
|
types: accept,
|
|
|
|
});
|
2022-06-04 11:06:38 -07:00
|
|
|
if (fileHandles !== pickedFiles) {
|
|
|
|
setFileHandles(pickedFiles);
|
2022-05-28 19:52:48 +02:00
|
|
|
onChange(pickedFiles);
|
|
|
|
}
|
|
|
|
} catch (e: unknown) {
|
|
|
|
console.error(e);
|
|
|
|
} finally {
|
|
|
|
setBusy(false);
|
|
|
|
}
|
2022-06-04 11:06:38 -07:00
|
|
|
}, [accept, busy, fileHandles, onChange]);
|
2022-05-28 19:52:48 +02:00
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
setSelectedFilename(
|
2022-06-04 11:06:38 -07:00
|
|
|
fileHandles?.length
|
|
|
|
? fileHandles[0].name
|
2022-05-28 19:52:48 +02:00
|
|
|
: 'No file selected');
|
2022-06-04 11:06:38 -07:00
|
|
|
}, [fileHandles]);
|
2022-05-28 19:52:48 +02:00
|
|
|
|
|
|
|
return (
|
|
|
|
<>
|
2022-05-31 17:38:40 +02:00
|
|
|
<button onClick={noAwait(onClickInternal)} disabled={disabled || busy}>
|
2022-05-28 19:52:48 +02:00
|
|
|
Choose File
|
|
|
|
</button>
|
|
|
|
|
|
|
|
<span role="label">{selectedFilename}</span>
|
|
|
|
</>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* File chooser component displayed as a button followed by the name of the
|
|
|
|
* chosen file (if any). When clicked, the button opens a native file chooser.
|
|
|
|
* If the browser supports the `window.showOpenFilePicker` function, this
|
|
|
|
* component uses it to open the file chooser. Otherwise, the component uses
|
|
|
|
* a regular file input element.
|
|
|
|
*
|
|
|
|
* Using `window.showOpenFilePicker` has the advantage of allowing read/write
|
|
|
|
* access to the file, whereas the regular input element only gives read
|
|
|
|
* access.
|
|
|
|
*/
|
|
|
|
export const FileChooser = ({
|
|
|
|
onChange,
|
|
|
|
control = controlDefault,
|
|
|
|
...rest
|
|
|
|
}: FileChooserProps) => {
|
|
|
|
|
|
|
|
const onChangeForInput = useCallback((files: FileList) => {
|
2022-06-03 19:12:30 +02:00
|
|
|
const handles: FileSystemFileHandle[] = [];
|
2022-05-28 19:52:48 +02:00
|
|
|
for (let i = 0; i < files.length; i++) {
|
|
|
|
const file = files.item(i);
|
|
|
|
if (file === null) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
handles.push({
|
|
|
|
kind: 'file',
|
|
|
|
name: file.name,
|
|
|
|
getFile: () => Promise.resolve(file),
|
|
|
|
createWritable: (_options) => Promise.reject('File not writable.'),
|
2022-06-03 19:12:30 +02:00
|
|
|
queryPermission: (descriptor) => Promise.resolve(descriptor === 'read' ? 'granted' : 'denied'),
|
|
|
|
requestPermission: (descriptor) => Promise.resolve(descriptor === 'read' ? 'granted' : 'denied'),
|
|
|
|
isSameEntry: (_unused) => Promise.resolve(false),
|
|
|
|
isDirectory: false,
|
|
|
|
isFile: true,
|
2022-05-28 19:52:48 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
onChange(handles);
|
2022-05-29 13:48:51 -07:00
|
|
|
}, [onChange]);
|
2022-05-28 19:52:48 +02:00
|
|
|
|
|
|
|
const onChangeForPicker = useCallback((fileHandles: FileSystemFileHandle[]) => {
|
2022-06-03 19:12:30 +02:00
|
|
|
onChange(fileHandles);
|
2022-05-29 13:48:51 -07:00
|
|
|
}, [onChange]);
|
2022-05-28 19:52:48 +02:00
|
|
|
|
|
|
|
return control === 'picker'
|
|
|
|
? (
|
|
|
|
<FilePickerChooser onChange={onChangeForPicker} {...rest} />
|
|
|
|
)
|
|
|
|
: (
|
|
|
|
<InputFileChooser onChange={onChangeForInput} {...rest} />
|
|
|
|
);
|
|
|
|
};
|