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

Solidify addTransitionType Semantics #32797

Merged
merged 5 commits into from
Apr 1, 2025

Conversation

sebmarkbage
Copy link
Collaborator

@sebmarkbage sebmarkbage commented Apr 1, 2025

Stacked on #32793.

This is meant to model the intended semantics of addTransitionType better. The previous hack just consumed all transition types when any root committed so it could steal them from other roots. Really each root should get its own set. Really each transition lane should get its own set.

We can't implement the full ideal semantics yet because 1) we currently entangle transition lanes 2) we lack AsyncContext on the client so for async actions we can't associate a addTransitionType call to a specific startTransition.

This starts by modeling Transition Types to be stored on the Transition instance. Conceptually they belong to the Transition instance of that startTransition they belong to. That instance is otherwise mostly just used for Transition Tracing but it makes sense that those would be able to be passed the Transition Types for that specific instance.

Nested startTransition need to get entangled. So that this addTransitionType can be associated with the setState:

startTransition(() => {
  startTransition(() => {
    addTransitionType(...)
  });
  setState(...);
});

Ideally we'd probably just use the same Transition instance itself since these are conceptually all part of one entangled one. But transition tracing uses multiple names and start times. Unclear what we want to do with that. So I kept separate instances but shared types set.

Next I collect the types added during a startTransition to any root scheduled with a Transition. This should really be collected one set per Transition lane in a LaneMap. In fact, the information would already be there if Transition Tracing was always enabled because it tracks all Transition instances per lane. For now I just keep track of one set for all Transition lanes. Maybe we should only add it if a setState was done on this root in this particular startTransition call rather having already scheduled any Transition earlier.

While async transitions are entangled, we don't know if there will be a startTransition+setState on a new root in the future. Therefore, we collect all transition types while this is happening and if a new root gets startTransition+setState they get added to that root.

startTransition(async () => {
  addTransitionType(...)
  await ...;
  setState(...);
});

@react-sizebot
Copy link

react-sizebot commented Apr 1, 2025

Comparing: deca965...25af4b8

Critical size changes

Includes critical production bundles, as well as any change greater than 2%:

Name +/- Base Current +/- gzip Base gzip Current gzip
oss-stable/react-dom/cjs/react-dom.production.js = 6.68 kB 6.68 kB +0.05% 1.83 kB 1.83 kB
oss-stable/react-dom/cjs/react-dom-client.production.js +0.05% 515.28 kB 515.56 kB +0.03% 91.81 kB 91.84 kB
oss-experimental/react-dom/cjs/react-dom.production.js = 6.69 kB 6.69 kB +0.05% 1.83 kB 1.83 kB
oss-experimental/react-dom/cjs/react-dom-client.production.js +0.29% 614.48 kB 616.24 kB +0.30% 108.71 kB 109.03 kB
facebook-www/ReactDOM-prod.classic.js +0.31% 648.21 kB 650.22 kB +0.34% 114.54 kB 114.92 kB
facebook-www/ReactDOM-prod.modern.js +0.31% 638.49 kB 640.50 kB +0.34% 112.95 kB 113.34 kB

Significant size changes

Includes any change greater than 0.2%:

Expand to show
Name +/- Base Current +/- gzip Base gzip Current gzip
oss-experimental/react/cjs/react.react-server.development.js +1.69% 35.98 kB 36.59 kB +1.48% 8.57 kB 8.70 kB
facebook-www/React-prod.modern.js +1.65% 20.69 kB 21.04 kB +1.51% 5.29 kB 5.37 kB
facebook-www/React-prod.classic.js +1.65% 20.70 kB 21.04 kB +1.51% 5.29 kB 5.37 kB
facebook-www/React-profiling.modern.js +1.62% 21.13 kB 21.47 kB +1.47% 5.37 kB 5.45 kB
facebook-www/React-profiling.classic.js +1.62% 21.13 kB 21.47 kB +1.47% 5.37 kB 5.45 kB
oss-experimental/react/cjs/react.react-server.production.js +1.30% 18.52 kB 18.76 kB +0.85% 4.92 kB 4.97 kB
facebook-www/React-dev.modern.js +1.09% 55.57 kB 56.17 kB +1.79% 12.10 kB 12.31 kB
facebook-www/React-dev.classic.js +1.09% 55.57 kB 56.17 kB +1.79% 12.10 kB 12.32 kB
oss-stable-semver/react/cjs/react.development.js +1.09% 45.06 kB 45.55 kB +0.96% 10.27 kB 10.37 kB
oss-stable/react/cjs/react.development.js +1.08% 45.09 kB 45.58 kB +0.96% 10.30 kB 10.40 kB
facebook-react-native/react/cjs/React-dev.js +0.96% 51.01 kB 51.50 kB +0.90% 11.35 kB 11.45 kB
oss-stable-semver/react/cjs/react.production.js +0.79% 16.88 kB 17.02 kB +0.52% 4.38 kB 4.40 kB
oss-stable/react/cjs/react.production.js +0.79% 16.91 kB 17.04 kB +0.52% 4.41 kB 4.43 kB
facebook-react-native/react/cjs/React-prod.js +0.70% 18.93 kB 19.06 kB +0.49% 4.93 kB 4.95 kB
facebook-react-native/react/cjs/React-profiling.js +0.69% 19.36 kB 19.50 kB +0.48% 5.01 kB 5.03 kB
facebook-www/ReactART-prod.modern.js +0.53% 370.43 kB 372.41 kB +0.59% 62.41 kB 62.78 kB
facebook-www/ReactART-prod.classic.js +0.52% 380.42 kB 382.39 kB +0.60% 64.03 kB 64.41 kB
oss-experimental/react-art/cjs/react-art.production.js +0.51% 340.23 kB 341.95 kB +0.52% 57.77 kB 58.07 kB
facebook-www/ReactReconciler-prod.modern.js +0.43% 484.40 kB 486.46 kB +0.51% 77.80 kB 78.20 kB
facebook-www/ReactReconciler-prod.classic.js +0.42% 494.71 kB 496.77 kB +0.50% 79.43 kB 79.83 kB
facebook-www/ReactART-dev.modern.js +0.41% 690.94 kB 693.77 kB +0.46% 108.71 kB 109.21 kB
facebook-www/ReactART-dev.classic.js +0.40% 700.44 kB 703.27 kB +0.48% 110.53 kB 111.06 kB
oss-experimental/react-art/cjs/react-art.development.js +0.39% 638.37 kB 640.89 kB +0.41% 101.91 kB 102.33 kB
oss-experimental/react-reconciler/cjs/react-reconciler.production.js +0.39% 461.80 kB 463.58 kB +0.47% 74.10 kB 74.45 kB
facebook-www/ReactReconciler-dev.modern.js +0.38% 799.36 kB 802.36 kB +0.38% 125.56 kB 126.04 kB
oss-experimental/react-reconciler/cjs/react-reconciler.profiling.js +0.37% 514.50 kB 516.42 kB +0.41% 82.01 kB 82.34 kB
facebook-www/ReactReconciler-dev.classic.js +0.37% 808.57 kB 811.57 kB +0.39% 127.22 kB 127.72 kB
oss-experimental/react-reconciler/cjs/react-reconciler.development.js +0.35% 760.13 kB 762.82 kB +0.37% 119.99 kB 120.44 kB
facebook-www/ReactDOM-prod.modern.js +0.31% 638.49 kB 640.50 kB +0.34% 112.95 kB 113.34 kB
facebook-www/ReactDOM-prod.classic.js +0.31% 648.21 kB 650.22 kB +0.34% 114.54 kB 114.92 kB
facebook-www/ReactDOMTesting-prod.modern.js +0.31% 652.89 kB 654.90 kB +0.34% 116.56 kB 116.95 kB
facebook-www/ReactDOMTesting-prod.classic.js +0.30% 662.61 kB 664.62 kB +0.34% 118.18 kB 118.59 kB
facebook-www/ReactDOM-profiling.modern.js +0.30% 702.83 kB 704.94 kB +0.31% 122.23 kB 122.60 kB
facebook-www/ReactDOM-profiling.classic.js +0.30% 710.87 kB 712.98 kB +0.31% 123.57 kB 123.96 kB
oss-experimental/react-dom/cjs/react-dom-client.production.js +0.29% 614.48 kB 616.24 kB +0.30% 108.71 kB 109.03 kB
oss-experimental/react-dom/cjs/react-dom-unstable_testing.production.js +0.28% 628.89 kB 630.65 kB +0.29% 112.26 kB 112.58 kB
oss-experimental/react-dom/cjs/react-dom-profiling.profiling.js +0.28% 670.26 kB 672.13 kB +0.30% 117.53 kB 117.88 kB
test_utils/ReactAllWarnings.js +0.26% 64.40 kB 64.57 kB +0.33% 16.13 kB 16.19 kB
facebook-www/ReactDOM-dev.modern.js +0.26% 1,160.98 kB 1,163.98 kB +0.27% 193.26 kB 193.79 kB
facebook-www/ReactDOM-dev.classic.js +0.26% 1,170.12 kB 1,173.12 kB +0.28% 194.97 kB 195.51 kB
facebook-www/ReactDOMTesting-dev.modern.js +0.25% 1,177.51 kB 1,180.51 kB +0.27% 197.03 kB 197.56 kB
facebook-www/ReactDOMTesting-dev.classic.js +0.25% 1,186.65 kB 1,189.65 kB +0.27% 198.80 kB 199.34 kB
oss-experimental/react-dom/cjs/react-dom-client.development.js +0.24% 1,103.31 kB 1,106.01 kB +0.24% 184.64 kB 185.08 kB
oss-experimental/react-dom/cjs/react-dom-profiling.development.js +0.24% 1,119.71 kB 1,122.40 kB +0.24% 187.47 kB 187.92 kB
oss-experimental/react-dom/cjs/react-dom-unstable_testing.development.js +0.24% 1,119.86 kB 1,122.55 kB +0.23% 188.27 kB 188.71 kB
oss-experimental/react/cjs/react.production.js = 19.69 kB 19.40 kB = 4.94 kB 4.94 kB

Generated by 🚫 dangerJS against 25af4b8

Copy link
Member

@rickhanlonii rickhanlonii left a comment

Choose a reason for hiding this comment

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

At some point (not now) it would be nice to have tests to reference as specs for the behavior. For this one I think we could add Scheduler.log in the ViewTransition event to assert on the types if we shimmed startViewTransition?

Don't want to slow this down yet though, this is a good example of a change harder to make if you have to update a bunch of tests too.

Reviewed: sebmarkbage/react@f57175d...3399b16

Conceptually they belong to each Transition object and should only be applied
where updates to that batch are applied.

However, since we batch regular Transitions together they go into the same set.
startTransition(() => {
  startTransition(() => {
    addTransitionType(...)
  });
  setState(...);
});

When the inner one finishes you could drop its transition types because it
has no work scheduled on it.
We store them on all roots that have been scheduled as part of the
startTransition call.

We shouldn't just let the first root that commits steal them all. Each root
gets its own set.
@sebmarkbage sebmarkbage force-pushed the addtransitionrefactor branch from 21cd559 to 25af4b8 Compare April 1, 2025 16:10
@sebmarkbage sebmarkbage merged commit 731ae3e into facebook:main Apr 1, 2025
14 of 16 checks passed
sebmarkbage added a commit that referenced this pull request Apr 1, 2025
…Map (#32800)

We have a high level concept for this used elsewhere.

We should use this for `transitionTypes` too:


https://github.com/facebook/react/blob/main/packages/react-reconciler/src/ReactInternalTypes.js#L285

As mentioned in #32797 we could also just use the `transitionLanes`
since the `types` are also on the `Transition` objects. If we always
stored this set.
github-actions bot pushed a commit that referenced this pull request Apr 1, 2025
…Map (#32800)

We have a high level concept for this used elsewhere.

We should use this for `transitionTypes` too:

https://github.com/facebook/react/blob/main/packages/react-reconciler/src/ReactInternalTypes.js#L285

As mentioned in #32797 we could also just use the `transitionLanes`
since the `types` are also on the `Transition` objects. If we always
stored this set.

DiffTrain build for [450f8df](450f8df)
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
CLA Signed React Core Team Opened by a member of the React Core Team
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants