-
Notifications
You must be signed in to change notification settings - Fork 35
Conversation
useLayoutEffect(() => { | ||
ref.current = cb; | ||
}, [cb]); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hm, I might be missing something, but do we need to run this as an effect rather than just straight-up ref.current = cb;
? The ref.current
in the function passed to useCallback
should refer to the value-at-time-of-invocation anyway, so I don't think it should lead to staleness.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Good point, by default I put all the side effects on the commit phase or layout phase. But maybe in this case is not necessary and it can be runned on the render phase as it's only a reasigment and don't interfers on the render.
Maybe changing the ref on render phase can be considered an unexpected side effect?
https://reactjs.org/docs/strict-mode.html#detecting-unexpected-side-effects
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
oh and this pattern is already used on the useGetter
function below.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi,
Doing this kind of side effect looks unsafe to me with concurrent mode. React might decide to start rendering something based on predictions, to warm up the next view, and that transition may end up to never happen. Dan Abramov mentioned that recently on Twitter.
I think it's safer and good enough to use an effect for this usecase
and yes there's the "useGetter" already ;)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hi,
Your solution is something I had in mind to solve this problem.
Maybe with better TS support we could handle it this way, but I'm not very convinced this is a good solution.
Unfortunately, I don't have a better solution so far :/ exposing a "useNavigationRef" could work but feels weird.
@satya164 any opinion on this PR?
return navigation; | ||
} | ||
|
||
function useStableCallback(cb: (...args: any) => any) { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
in terms of typing this is not good enough, all wrapped actions will be any[] => any
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The public api will have the same types as now, the (...args: any) => any
will remain private. It's bad, but not so bad.
src/Hooks.ts
Outdated
isFirstRouteInParent: useStableCallback(navigation.isFirstRouteInParent), | ||
|
||
// drawer navigator, actions | ||
openDrawer: useStableCallback(navigation.openDrawer), |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
before, on a stack navigation, openDrawer would be undefined. Now it's a callback that will crash with "undefined is not a function", not exactly the same ;)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
You are right! On v4 this also change since StackNavigator is in another package. I'll take a look on this.
I think this is not the responsibility of react-native-hooks since this changes the semantics of how the Also, we should not hardcode these methods on the navigation object. As you pointed out, methods like A better solution is to send a PR to core to make these methods stable :) |
There's an issue already closed on core repo for this: This should be here because its too dificult to make that change on core right now and also because v5 it's coming and this package will be deprecated. |
Doing more research I found there's too many posible bugs with this implementation. Maybe it's just simpler let this as it is and add a warning so users can just add this helper to theirs code base: function useStableCallback(cb: (...args: any) => any) {
const ref = useRef(cb);
useLayoutEffect(() => {
ref.current = cb;
}, [cb]);
return useCallback((...args) => ref.current(...args), [ref]);
}; |
I don't think it's too difficult. The action creators are created in one place it should be possible to use a similar approach by storing these in a cache object instead ref. It's more work but it's also correct and handles way more cases than you can handle here without complicating code.
That doesn't mean we should make it inconsistent with the version of React Navigation we use it with. Anyway, as I already mentioned, there are many issues with this approach.
|
Good points, thanks @satya164 I'm closing the PR then
Agree on this, and also a reason I don't think we should fix it here but in core instead. Temporarily, I think the best option would be to recommend users to fix this in userland, with a workaround like
What do you think? |
This solves: #40