Skip to content
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

Wrap custom events dispatching with flushSync #1292

Closed
wants to merge 6 commits into from

Conversation

Andarist
Copy link
Contributor

@Andarist Andarist commented Mar 31, 2022

fixes #1287

Ok, so this is what I got from a quick debugging session:

  1. it's because in React 18 the contextmenuchange isn't dispatched at all in this case and thus you can't prevent it and fire a custom logic for it
  2. this in turn is caused by the fact that you set pointer-events: none on the body, here
  3. you need to set this to handle some requirements and that is fine, the problem is that in this scenario resetPointerEvents is not called "soon enough" to unblock the mentioned contextmenuchange event
  4. In React 17 it actually happens "soon enough" so the whole thing works

I've attached onOpenChange to the ContextMenu.Root and just logged the received value together with some other stuff.

This is the order of the things happening under the hood:
React 17: resetPointerEvents() -> onOpenChange(false)
React 18: onOpenChange(false) -> resetPointerEvents()

As we can see... this somewhat matches the info that I've gathered when debugging. The order is flipped and this leads to a special sort of race condition.

Based on my intuition I've just deduced that this might behave slightly differently in terms of "flushing" the updates as there were some changes to that in React 18. I was looking for a rogue setState, to wrap it with flushSync but then I've realized that during debugging I've seen some CustomEvent stuff and since React is using window.event to determine the update priority... it made sense to just wrap all of those custom events with flushSync. You might want to be more granular, but I think that it's safer to just merge this and then reexamine if some of your custom events should actually not be wrapped with this.

@benoitgrelard
Copy link
Contributor

Hi @Andarist,

Thanks for this. I am not aware of flushSync and have been researching it a bit.
How much do you know about it?
Looking at this here this statement scares me a bit:

You also have an option to flush the entire tree if you know what you’re doing. The API is called ReactDOM.flushSync(fn). I don’t think we have documented it yet, but we definitely will do so at some point during the 16.x release cycle. Note that it actually forces complete re-rendering for updates that happen inside of the call, so you should use it very sparingly. This way it doesn’t break the guarantee of internal consistency between props, state, and refs.

I will look into the original issue a bit more too.

@Andarist
Copy link
Contributor Author

Andarist commented Apr 6, 2022

You can check out the Twitter thread where I've linked to this PR: https://twitter.com/AndaristRake/status/1509537858423566344 This might have some additional interesting info in it. At no point, any React team member has suggested there that this is the wrong solution.

I understand why you might be worried about this here but I would roughly classify this as equivalent to an inferred default priority for all discrete events. It just happens that you are using custom events and thus the builtin detection in React can't classify this automatically as triggered by the user. Note that there might be some subtle differences as this one really flushes the whole update synchronously whereas maybe for discrete events things can be batched in a better way. I think both would flush in the same synchronous frame - but perhaps with flushSync you can end up rerendering multiple times during that synchronous frame (one update would come from within flushSync and another one would have to come from some other place, so the question is also - can another update be scheduled in the same synchronous frame outside of the flushSync "scope"?).

Either way - I think it's worth landing this ASAP and researching the performance implications separately as this should fix bugs for people that are hard to diagnose and I don't think it should come with a severe performance cost, maybe just with a minor one (if at all).

@jjenzz might also have some context for this function as I remember she was asking about it in the React repo quite some time ago

@benoitgrelard
Copy link
Contributor

Thanks for the added details and the link to the tweet.
As you said, it seems the React team agrees that's the way to go.

It might make sense to apply the same change to other places we use custom events (a quick search shows 3 files: DismissableLayer.tsx, Toast.tsx and Tooltip.tsx).

One thing I am unclear about, is whether this only applies to custom events created using the CustomEvent constructor, or also to custom events created using the Event constructor. I'm asking because we use those a fair few also.

I appreciate your help on this 🙏

Copy link
Contributor

@benoitgrelard benoitgrelard left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We will also need to add react-dom as a peer dependency of any package that now uses it and also to any of the packages that depend on one of those.

packages/react/dismissable-layer/src/DismissableLayer.tsx Outdated Show resolved Hide resolved
packages/react/dismissable-layer/src/DismissableLayer.tsx Outdated Show resolved Hide resolved
@Andarist
Copy link
Contributor Author

Andarist commented Apr 7, 2022

It might make sense to apply the same change to other places we use custom events (a quick search shows 3 files: DismissableLayer.tsx, Toast.tsx and Tooltip.tsx).

Wrapped all of them.

One thing I am unclear about, is whether this only applies to custom events created using the CustomEvent constructor, or also to custom events created using the Event constructor. I'm asking because we use those a fair few also.

I believe that this would apply to Event constructor as well. It's just that React is internally determining the priority of the update based on window.event.type and thus if you use custom event names/types your updates might not be treated as coming from "discrete" events.

We will also need to add react-dom as a peer dependency of any package that now uses it and also to any of the packages that depend on one of those.

I've added the peer dep to dismissable-layer, toast and tooltip already had it. I've also checked all dismissable-layer dependants and all of them had a peer dep on react-dom.

onOpen();
}
onOpenChange?.(open);
ReactDOM.flushSync(() => {
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

In here, I've decided to wrap the whole onChange handler to "batch" all updates scheduled by within this callback

@Andarist
Copy link
Contributor Author

Andarist commented Apr 8, 2022

I guess I forgot to update the lock file - just fixed that

@benoitgrelard
Copy link
Contributor

Can you run yarn version check --interactive locally and patch them. May need to run a second time to decline the main workspace after too.

🙏

@Andarist
Copy link
Contributor Author

Andarist commented Apr 8, 2022

@benoitgrelard I've just done this - please verify if this is how you want to bump packages based on this change

@benoitgrelard
Copy link
Contributor

It's all good apart from the main workspace which should be declined instead of patch 👍

@benoitgrelard
Copy link
Contributor

I believe that this would apply to Event constructor as well. It's just that React is internally determining the priority of the update based on window.event.type and thus if you use custom event names/types your updates might not be treated as coming from "discrete" events.

Do you think we should do those too then?

@andy-hook
Copy link
Contributor

Closing in favour of #1378, thanks again for your help on this 🙏

@andy-hook andy-hook closed this May 10, 2022
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

[ContextMenu] Native context menu incorrectly appears after second click
3 participants