-
Notifications
You must be signed in to change notification settings - Fork 1.2k
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
fix: Improve ListBox perf caused by excessive re-rendering #8010
base: main
Are you sure you want to change the base?
Conversation
…change every render while the hover/selection state changes.
b5011eb
to
18c31cf
Compare
…memoize ListBoxItemInner
122671b
to
4fd8865
Compare
I think I got this working! I figured out a way to avoid any changes to I suspect there may be some other components we need to do this for, but this is a decent start. Thanks for taking a look. |
1f74809
to
e1f0ddd
Compare
I use a
Select
(ListBox
) to display a list of countries in an address component for my project. There are about ~200 countries, few enough that virtualization isn't necessary. However, there is a noticeable delay when hovering the mouse over countries or changing the selected country using the keyboard.This delay stems from the
ListBox
re-rendering everyListBoxItem
whenever the hovered item changes:94490600-503618712-Recording_2025-03-29_192152.mp4
In the lower right corner, you can see the FPS drop sharply.
I tracked down the source to several un-memoized objects that get re-created every render. As React diffs props by reference, this causes a full render on all
ListBoxItem
s. I addeduseMemo(..)
s were appropriate, and now the performance is much smoother - only two re-renders occur (the newly and previously hovered items):295701505-74604607-Recording_2025-03-29_192319.mp4
✅ Pull Request Checklist:
📝 Test Instructions:
I used the
React Aria Components > Select Many
story without any modifications to demonstrate the issue/fix:I used the React Scan extension to identify the un-memoized objects - I had to force the
Popover
within the Select Many example open withisOpen={true}
so I could select it with React Scan.🧢 Your Project:
RSP