demarches-normaliennes/app/javascript/components/ComboSearch.jsx

165 lines
4.3 KiB
React
Raw Normal View History

2020-10-07 17:41:20 +02:00
import React, { useState, useMemo, useCallback, useRef } from 'react';
import { useDebounce } from 'use-debounce';
2020-10-07 17:41:20 +02:00
import { useQuery } from 'react-query';
import PropTypes from 'prop-types';
import {
Combobox,
ComboboxInput,
ComboboxPopover,
ComboboxList,
ComboboxOption
} from '@reach/combobox';
import '@reach/combobox/styles.css';
import { fire } from '@utils';
2020-10-07 17:41:20 +02:00
2021-06-09 17:13:26 +02:00
import { useDeferredSubmit } from './shared/hooks';
2020-10-07 17:41:20 +02:00
function defaultTransformResults(_, results) {
return results;
}
function ComboSearch({
placeholder,
required,
hiddenFieldId,
onChange,
scope,
minimumInputLength,
transformResult,
allowInputValues = false,
transformResults = defaultTransformResults,
className
2020-10-07 17:41:20 +02:00
}) {
const label = scope;
const hiddenValueField = useMemo(
2020-10-07 17:41:20 +02:00
() => document.querySelector(`input[data-uuid="${hiddenFieldId}"]`),
[hiddenFieldId]
);
const hiddenIdField = useMemo(
() =>
document.querySelector(
`input[data-uuid="${hiddenFieldId}"] + input[data-reference]`
),
[hiddenFieldId]
);
const initialValue = hiddenValueField && hiddenValueField.value;
2020-10-07 17:41:20 +02:00
const [searchTerm, setSearchTerm] = useState('');
const [debouncedSearchTerm] = useDebounce(searchTerm, 300);
2020-10-07 17:41:20 +02:00
const [value, setValue] = useState(initialValue);
const resultsMap = useRef({});
2021-06-09 17:09:07 +02:00
const setExternalValue = useCallback(
(value) => {
if (hiddenValueField) {
hiddenValueField.setAttribute('value', value);
fire(hiddenValueField, 'autosave:trigger');
}
},
[hiddenValueField]
);
const setExternalId = useCallback(
(key) => {
if (hiddenIdField) {
hiddenIdField.setAttribute('value', key);
}
},
[hiddenIdField]
);
const setExternalValueAndId = useCallback((value) => {
const [key, result] = resultsMap.current[value];
setExternalId(key);
setExternalValue(value);
2020-10-07 17:41:20 +02:00
if (onChange) {
onChange(value, result);
}
2021-06-09 17:09:07 +02:00
}, []);
2021-06-09 17:13:26 +02:00
const awaitFormSubmit = useDeferredSubmit(hiddenValueField);
2020-10-07 17:41:20 +02:00
const handleOnChange = useCallback(
({ target: { value } }) => {
setValue(value);
if (value.length >= minimumInputLength) {
setSearchTerm(value.trim());
if (allowInputValues) {
setExternalId('');
2020-10-07 17:41:20 +02:00
setExternalValue(value);
}
}
},
[minimumInputLength]
);
const handleOnSelect = useCallback((value) => {
setExternalValueAndId(value);
2020-10-07 17:41:20 +02:00
setValue(value);
2021-06-09 17:13:26 +02:00
setSearchTerm('');
awaitFormSubmit.done();
2021-06-09 17:09:07 +02:00
}, []);
2020-10-07 17:41:20 +02:00
const { isSuccess, data } = useQuery([scope, debouncedSearchTerm], {
enabled: !!debouncedSearchTerm,
notifyOnStatusChange: false,
refetchOnMount: false
});
const results = isSuccess ? transformResults(debouncedSearchTerm, data) : [];
2021-06-09 17:13:26 +02:00
const onBlur = useCallback(() => {
if (!allowInputValues && isSuccess && results[0]) {
const [, value] = transformResult(results[0]);
awaitFormSubmit(() => {
handleOnSelect(value);
});
}
}, [data]);
2020-10-07 17:41:20 +02:00
return (
<Combobox aria-label={label} onSelect={handleOnSelect}>
<ComboboxInput
className={className}
2020-10-07 17:41:20 +02:00
placeholder={placeholder}
onChange={handleOnChange}
2021-06-09 17:13:26 +02:00
onBlur={onBlur}
2020-10-07 17:41:20 +02:00
value={value}
required={required}
/>
{isSuccess && (
<ComboboxPopover className="shadow-popup">
{results.length > 0 ? (
<ComboboxList>
{results.map((result) => {
const [key, str] = transformResult(result);
resultsMap.current[str] = [key, result];
2020-10-07 17:41:20 +02:00
return (
<ComboboxOption
key={key}
value={str}
data-option-value={str}
/>
);
})}
</ComboboxList>
) : (
<span style={{ display: 'block', margin: 8 }}>
Aucun résultat trouvé
</span>
)}
</ComboboxPopover>
)}
</Combobox>
);
}
ComboSearch.propTypes = {
placeholder: PropTypes.string,
required: PropTypes.bool,
hiddenFieldId: PropTypes.string,
scope: PropTypes.string,
minimumInputLength: PropTypes.number,
transformResult: PropTypes.func,
transformResults: PropTypes.func,
allowInputValues: PropTypes.bool,
onChange: PropTypes.func,
className: PropTypes.string
2020-10-07 17:41:20 +02:00
};
export default ComboSearch;