Как мы узнаем, когда значение React ref.current изменилось?
Обычно с помощью реквизита мы можем написать
componentDidUpdate(oldProps) {
if (oldProps.foo !== this.props.foo) {
console.log('foo prop changed')
}
}
чтобы обнаружить изменения реквизита.
Но если мы используем React.createRef()
Как определить, когда ссылка изменилась на новый компонент или элемент DOM? Документы React ничего не упоминают.
Fe,
class Foo extends React.Component {
someRef = React.createRef()
componentDidUpdate(oldProps) {
const refChanged = /* What do we put here? */
if (refChanged) {
console.log('new ref value:', this.someRef.current)
}
}
render() {
// ...
}
}
Должны ли мы сами реализовать что-то старое?
Fe,
class Foo extends React.Component {
someRef = React.createRef()
oldRef = {}
componentDidMount() {
this.oldRef.current = this.someRef.current
}
componentDidUpdate(oldProps) {
const refChanged = this.oldRef.current !== this.someRef.current
if (refChanged) {
console.log('new ref value:', this.someRef.current)
this.oldRef.current = this.someRef.current
}
}
render() {
// ...
}
}
Это то, что мы должны делать? Я бы подумал, что React испек бы какую-то легкую функцию для этого.
1 ответ
Документы React рекомендуют использовать ссылки обратного вызова для обнаруженияref
изменение стоимости.
Пример хуков:
export function Foo() {
const [ref, setRef] = useState(null);
const onRefChange = useCallback(node => {
// ref value changed to node
setRef(node); // e.g. change ref state to trigger re-render
if (node === null) {
// node is null, if DOM node of ref had been unmounted before
} else {
// ref value exists
}
}, []);
return <h1 ref={onRefChange}>Hey</h1>;
}
Пример компонента класса:
export class FooClass extends React.Component {
state = { ref: null, ... };
onRefChange = node => {
// same as Hooks example
this.setState({ ref: node });
};
render() {
return <h1 ref={this.onRefChange}>Hey</h1>;
}
}
Примечание: useRef
не уведомляет о ref
изменения. Также не повезло сReact.createRef()
/ ссылка на объект.
Вот тестовый пример, который удаляет и повторно добавляет узел при запуске onRefChange
Перезвоните:
const Foo = () => {
const [ref, setRef] = useState(null);
const [removed, remove] = useState(false);
useEffect(() => {
setTimeout(() => remove(true), 3000); // drop after 3 sec
setTimeout(() => remove(false), 5000); // ... and mount it again
}, []);
const onRefChange = useCallback(node => {
console.log("ref changed to:", node);
setRef(node); // or change other state to re-render
}, []);
return !removed && <h3 ref={onRefChange}>Hello, world</h3>;
}
ReactDOM.render(<Foo />, document.getElementById("root"));
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/16.10.1/umd/react.production.min.js" integrity="sha256-vMEjoeSlzpWvres5mDlxmSKxx6jAmDNY4zCt712YCI0=" crossorigin="anonymous"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/16.10.1/umd/react-dom.production.min.js" integrity="sha256-QQt6MpTdAD0DiPLhqhzVyPs1flIdstR4/R7x4GqCvZ4=" crossorigin="anonymous"></script>
<script> var {useState, useEffect, useCallback} = React</script>
<div id="root"></div>
componentDidUpdate
вызывается при изменении состояния компонента или реквизита, поэтому он не обязательно вызывается при ref
изменения, так как он может быть видоизменен по вашему усмотрению.
Если вы хотите проверить, изменился ли реф по сравнению с предыдущим рендером, вы можете оставить другой реф по сравнению с реальным.
пример
class App extends React.Component {
prevRef = null;
ref = React.createRef();
state = {
isVisible: true
};
componentDidMount() {
this.prevRef = this.ref.current;
setTimeout(() => {
this.setState({ isVisible: false });
}, 1000);
}
componentDidUpdate() {
if (this.prevRef !== this.ref.current) {
console.log("ref changed!");
}
this.prevRef = this.ref.current;
}
render() {
return this.state.isVisible ? <div ref={this.ref}>Foo</div> : null;
}
}
ReactDOM.render(<App />, document.getElementById("root"));
<script src="https://unpkg.com/react@16/umd/react.development.js"></script>
<script src="https://unpkg.com/react-dom@16/umd/react-dom.development.js"></script>
<div id="root"></div>