由于您已经在应用程序中使用了钩子,因此您需要做的唯一更改是使用useRef
而不是createRef
生成列表的引用。
const Button = React.forwardRef((props, ref) => {
return (
<button
onClick={props.toggleList}
ref={ref}
>
button
</button>
);
})
const List = (props) => {
const list = React.useRef();
handleClick = (e) => {
const clickIsOutsideList = !list.current.contains(e.target);
console.log(list, clickIsOutsideList);
if (clickIsOutsideList) {
props.hideList();
}
}
React.useEffect(function addClickHandler() {
document.addEventListener('click', handleClick);
}, []);
return (
<ul ref={list}>
<li>item</li>
<li>item</li>
<li>item</li>
</ul>
);
}
const App = () => {
const [ListHidden, setListHidden] = React.useState(true);
const button = React.useRef();
const toggleList = () => {
setListHidden(!ListHidden);
}
const hideList = () => {
setListHidden(true);
button.current.focus();
}
return (
<div className="App">
<Button
toggleList={toggleList}
ref={button}
/>
{
!ListHidden &&
<List hideList={hideList} />
}
</div>
);
}
ReactDOM.render(<App />, document.getElementById('root'));
工作演示
你需要它的原因是因为在你的函数组件的每次渲染中,如果你使用React.createRef
whileuseRef
实现一个新的 ref,它会在第一次调用时生成一个 ref 并在将来的任何时候返回相同的引用重新渲染。
PS一个经验法则,你可以说useRef
当你想在功能组件中使用引用时应该使用它,而createRef
应该在类组件中使用它。