-
Notifications
You must be signed in to change notification settings - Fork 24.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Animated.Value resets if component was re-mounted #28114
Comments
Hi, At first let me explain the problem in details then I'll suggest possible solutions. useNativeDriver = true Explanation in detailsExample without unmounting:export default class App extends React.Component {
x = new Animated.Value(0);
render() {
return <Animated.View style={transform: {translate: this.x}}/>
}
} Example with unmounting:export default class App extends React.Component {
state = {show: true}
x = new Animated.Value(0);
render() {
return (
<>
{show && <Animated.View style={transform: {translate: this.x}}/>}
<Button onPress={() => this.setState(prevState => ({show: !prevState }))}>Toggle</Button>
</>
}
} Example contains simple React component that renders Example with your "fix"export default class App extends React.Component {
state = {show: true}
x = new Animated.Value(0);
render() {
return (
<>
{show && <Animated.View style={{ transform: [{ translateX: this.x }] }}/>}
<Animated.View style={{ transform: [{ translateX: this.x }] }}/>
<Button onPress={() => this.setState(prevState => ({show: !prevState }))}>Toggle</Button>
</>
}
} Here we are rendering Animated.View twice ! Why your "fix" worksYour fix works because our Animated.Value Possible solutions
|
From the way the components are written, it seems the intention was for the JS node to hold the correct value while it is not attached to any native nodes. I definitely agree with your second suggestion, I had a look through reanimated, and calling getValue explicitly is a good fix as well. I had in mind a listener similar to onAnimatedValue , however provided there are no situations where the native value gets lost without going through JS |
cc @JoshuaGross Can you take a look at our proposed solution #28841? We could really use any feedback. Thanks |
Description:
If Animated.Value is used for a natively driven animation and then the component is unmounted, when the component is mounted back, the rendered value resets to the initial state.
React Native version:
0.61.5
Steps To Reproduce
Expected Results
Animated.Value should hold the last value it had before the component was unmounted, and the component should return to the same state on being re-mounted as a result.
Code Example
https://github.com/djonin/react-native-animated-value-bug
This is a duplicate of issues #23712 and #23621, which were incorrectly closed as fixed, due to using a listener in the example, which avoids this defect. The stated fix #24571 only fixed the problem with the listener not correctly re-binding to the new value, however the broader case issue mentioned in #23621 is still there.
This issue is due to the fact the native node, which has the up to date value, is destroyed after a component is dismounted. The JS animated value object doesn't receive any native value updates unless there is a listener attached explicitly by the user. Therefore the next time the component is mounted, it uses the value as it was before the native animation.
Currently the only way to preserve the value is to attach a listener, meaning the value updates will be sent over the bridge often during animations. In addition, it's not obvious to users that attaching a listener is required for correct re-mounting behavior.
Proposed solution
One possible fix would be to make a new event to call back to JS with the value just once when the native node is being detached/destroyed, so that we can preserve whatever the final native value was in our JS object, and for the JS animated value object to always listen to this new native event, even with no listeners attached explicitly. We need some place to store our value, and since the native side is quite correctly being cleaned up after unmount, JS side seems like the best place to do it.
The text was updated successfully, but these errors were encountered: