movie-web/src2/components/SelectBox.js

83 lines
2.4 KiB
JavaScript
Raw Normal View History

2021-10-25 23:41:42 +02:00
import { useRef, useState, useEffect } from "react"
import "./SelectBox.css"
function Option({ option, ...props }) {
2021-10-25 23:41:42 +02:00
return (
<div className="option" {...props}>
2021-10-25 23:41:42 +02:00
<input
type="radio"
className="radio"
id={option.id} />
<label htmlFor={option.id}>
<div className="item">{option.name}</div>
</label>
</div>
)
}
export function SelectBox({ options, selectedItem, setSelectedItem }) {
2021-10-25 23:41:42 +02:00
if (!Array.isArray(options)) {
throw new Error("Items must be an array!")
2021-10-25 23:41:42 +02:00
}
const [active, setActive] = useState(false)
const containerRef = useRef();
const handleClick = e => {
if (containerRef.current.contains(e.target)) {
// inside click
return;
}
// outside click
closeDropdown()
};
const closeDropdown = () => {
setActive(false)
}
useEffect(() => {
// add when mounted
document.addEventListener("mousedown", handleClick);
// return function to be called when unmounted
return () => {
document.removeEventListener("mousedown", handleClick);
};
2021-10-26 14:29:04 +02:00
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
2021-10-25 23:41:42 +02:00
const onOptionClick = (e, option, i) => {
e.stopPropagation()
2021-10-25 23:41:42 +02:00
setSelectedItem(i)
closeDropdown()
}
const handleSelectedKeyPress = event => {
if (event.code === 'Enter' || event.code === 'Space'){
setActive(a => !a);
}
}
const handleOptionKeyPress = (option, i) => event => {
if (event.code === 'Enter' || event.code === 'Space'){
onOptionClick(event, option, i);
}
}
2021-10-25 23:41:42 +02:00
return (
<div className="select-box" ref={containerRef} onClick={() => setActive(a => !a)} >
<div className="selected" tabIndex={0} onKeyPress={handleSelectedKeyPress}>
2021-10-25 23:41:42 +02:00
{options ? (
<Option option={options[selectedItem]} />
) : null}
</div>
<div className={"options-container" + (active ? " active" : "")}>
{options.map((opt, i) => (
<Option option={opt} key={i} onClick={(e) => onOptionClick(e, opt, i)} tabIndex={active ? 0 : undefined} onKeyPress={active ? handleOptionKeyPress(opt, i) : undefined} />
))}
</div>
2021-10-25 23:41:42 +02:00
</div>
)
2021-10-26 14:25:34 +02:00
}