Skip to content

Conversation

@sebmarkbage
Copy link
Collaborator

@sebmarkbage sebmarkbage commented Nov 18, 2025

When dealing with optimistic state, a common problem is not knowing the id of the thing we're waiting on. Items in lists need keys (and single items should often have keys too to reset their state). As a result you have to generate fake keys. It's a pain to manage those and when the real item comes in, you often end up rendering that with a different key which resets the state of the component tree. That in turns works against the grain of React and a lot of negatives fall out of it.

This adds a special optimisticKey symbol that can be used in place of a string key.

import {optimisticKey} from 'react';
...
const [optimisticItems, setOptimisticItems] = useOptimistic([]);
const children = savedItems.concat(
  optimisticItems.map(item =>
    <Item key={optimisticKey} item={item} />
  )
);
return <div>{children}</div>;

The semantics of this optimisticKey is that the assumption is that the newly saved item will be rendered in the same slot as the previous optimistic items. State is transferred into whatever real key ends up in the same slot.

This might lead to some incorrect transferring of state in some cases where things don't end up lining up - but it's worth it for simplicity in many cases since dealing with true matching of optimistic state is often very complex for something that only lasts a blink of an eye.

If a new item matches a key elsewhere in the set, then that's favored over reconciling against the old slot.

One quirk with the current algorithm is if the savedItems has items removed, then the slots won't line up by index anymore and will be skewed. We might be able to add something where the optimistic set is always reconciled against the end. However, it's probably better to just assume that the set will line up perfectly and otherwise it's just best effort that can lead to weird artifacts.

An optimisticKey will match itself for updates to the same slot, but it will not match any existing slot that is not an optimisticKey. So it's not an any, which I originally called it, because it doesn't match existing real keys against new optimistic keys. Only one direction.

Treat it same as null for now.
Since we might collapse multiple keys into a single key, we have to deal
with that special case. In that case we make the collapsed key optimistic.
For purposes of resuming, we treat optimistic keys as an index since they
have to appear in the same slot. Same as null.
We match new Fibers regardless of what their key is if the current key is
optimistic.

This means that the new Fiber might have a different key for the first time
so we need to handle updating it and restoring it properly.

For the complex cases of maps, I use a negative index to represent previous
optimistic keys so that we can look up to see if we match one of those.
@meta-cla meta-cla bot added the CLA Signed label Nov 18, 2025
@github-actions github-actions bot added the React Core Team Opened by a member of the React Core Team label Nov 18, 2025
@react-sizebot
Copy link

react-sizebot commented Nov 18, 2025

Comparing: ea4899e...072d397

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 +2.35% 6.68 kB 6.84 kB +2.90% 1.83 kB 1.88 kB
oss-stable/react-dom/cjs/react-dom-client.production.js +0.02% 608.16 kB 608.26 kB +0.02% 107.65 kB 107.67 kB
oss-experimental/react-dom/cjs/react-dom.production.js +2.35% 6.69 kB 6.84 kB +2.90% 1.83 kB 1.88 kB
oss-experimental/react-dom/cjs/react-dom-client.production.js +0.18% 666.18 kB 667.37 kB +0.17% 117.35 kB 117.55 kB
facebook-www/ReactDOM-prod.classic.js +0.04% 693.31 kB 693.56 kB +0.06% 121.98 kB 122.05 kB
facebook-www/ReactDOM-prod.modern.js +0.04% 683.73 kB 683.99 kB +0.06% 120.36 kB 120.43 kB
oss-experimental/react/cjs/react-jsx-runtime.profiling.js +22.05% 0.98 kB 1.19 kB +11.04% 0.50 kB 0.55 kB
oss-experimental/react/cjs/react-jsx-runtime.production.js +22.03% 0.98 kB 1.19 kB +10.82% 0.50 kB 0.55 kB
oss-experimental/react/cjs/react-jsx-runtime.react-server.production.js +16.39% 1.31 kB 1.53 kB +8.37% 0.68 kB 0.74 kB
oss-experimental/react/cjs/react-jsx-dev-runtime.react-server.production.js +16.34% 1.32 kB 1.53 kB +7.89% 0.68 kB 0.74 kB
facebook-react-native/react-dom/cjs/ReactDOM-prod.js +2.42% 6.48 kB 6.63 kB +2.88% 1.81 kB 1.86 kB
facebook-react-native/react-dom/cjs/ReactDOM-profiling.js +2.42% 6.48 kB 6.63 kB +2.88% 1.81 kB 1.86 kB
oss-stable-semver/react-dom/cjs/react-dom.production.js +2.36% 6.66 kB 6.81 kB +2.88% 1.80 kB 1.86 kB
oss-experimental/react/cjs/react-jsx-dev-runtime.development.js +2.05% 12.42 kB 12.68 kB +1.54% 3.44 kB 3.50 kB
oss-experimental/react/cjs/react.react-server.production.js +2.01% 18.74 kB 19.11 kB +1.46% 4.93 kB 5.00 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-jsx-runtime.profiling.js +22.05% 0.98 kB 1.19 kB +11.04% 0.50 kB 0.55 kB
oss-experimental/react/cjs/react-jsx-runtime.production.js +22.03% 0.98 kB 1.19 kB +10.82% 0.50 kB 0.55 kB
oss-experimental/react/cjs/react-jsx-runtime.react-server.production.js +16.39% 1.31 kB 1.53 kB +8.37% 0.68 kB 0.74 kB
oss-experimental/react/cjs/react-jsx-dev-runtime.react-server.production.js +16.34% 1.32 kB 1.53 kB +7.89% 0.68 kB 0.74 kB
facebook-react-native/react-dom/cjs/ReactDOM-prod.js +2.42% 6.48 kB 6.63 kB +2.88% 1.81 kB 1.86 kB
facebook-react-native/react-dom/cjs/ReactDOM-profiling.js +2.42% 6.48 kB 6.63 kB +2.88% 1.81 kB 1.86 kB
oss-stable-semver/react-dom/cjs/react-dom.production.js +2.36% 6.66 kB 6.81 kB +2.88% 1.80 kB 1.86 kB
oss-stable/react-dom/cjs/react-dom.production.js +2.35% 6.68 kB 6.84 kB +2.90% 1.83 kB 1.88 kB
oss-experimental/react-dom/cjs/react-dom.production.js +2.35% 6.69 kB 6.84 kB +2.90% 1.83 kB 1.88 kB
oss-experimental/react/cjs/react-jsx-dev-runtime.development.js +2.05% 12.42 kB 12.68 kB +1.54% 3.44 kB 3.50 kB
oss-experimental/react/cjs/react.react-server.production.js +2.01% 18.74 kB 19.11 kB +1.46% 4.93 kB 5.00 kB
oss-experimental/react/cjs/react.production.js +1.98% 19.09 kB 19.46 kB +1.51% 4.85 kB 4.92 kB
oss-experimental/react/cjs/react-jsx-runtime.development.js +1.96% 13.03 kB 13.29 kB +1.51% 3.46 kB 3.51 kB
oss-experimental/react/cjs/react-jsx-runtime.react-server.development.js +1.82% 14.01 kB 14.26 kB +1.42% 3.59 kB 3.64 kB
oss-experimental/react/cjs/react-jsx-dev-runtime.react-server.development.js +1.82% 14.01 kB 14.26 kB +1.42% 3.59 kB 3.64 kB
oss-experimental/react/cjs/react.react-server.development.js +1.45% 39.00 kB 39.57 kB +1.18% 9.13 kB 9.24 kB
facebook-react-native/react-dom/cjs/ReactDOM-dev.js +1.34% 16.74 kB 16.96 kB +1.72% 3.60 kB 3.66 kB
oss-stable-semver/react-dom/cjs/react-dom.development.js +1.27% 17.67 kB 17.90 kB +1.49% 3.82 kB 3.88 kB
oss-stable/react-dom/cjs/react-dom.development.js +1.27% 17.70 kB 17.92 kB +1.53% 3.85 kB 3.90 kB
oss-experimental/react-dom/cjs/react-dom.development.js +1.27% 17.70 kB 17.93 kB +1.53% 3.85 kB 3.91 kB
oss-experimental/react/cjs/react.development.js +1.11% 51.12 kB 51.69 kB +1.01% 11.50 kB 11.62 kB
oss-stable-semver/react-server/cjs/react-server-flight.production.js +0.43% 65.83 kB 66.11 kB +0.48% 13.00 kB 13.06 kB
oss-stable/react-server/cjs/react-server-flight.production.js +0.43% 65.83 kB 66.11 kB +0.48% 13.00 kB 13.06 kB
oss-experimental/react-server/cjs/react-server-flight.production.js +0.42% 67.78 kB 68.07 kB +0.45% 13.42 kB 13.48 kB
oss-stable-semver/react-server-dom-parcel/cjs/react-server-dom-parcel-server.browser.production.js +0.35% 94.52 kB 94.85 kB +0.33% 19.37 kB 19.44 kB
oss-stable/react-server-dom-parcel/cjs/react-server-dom-parcel-server.browser.production.js +0.35% 94.52 kB 94.85 kB +0.33% 19.37 kB 19.44 kB
oss-stable-semver/react-server-dom-parcel/cjs/react-server-dom-parcel-server.edge.production.js +0.34% 95.66 kB 95.99 kB +0.35% 19.64 kB 19.70 kB
oss-stable/react-server-dom-parcel/cjs/react-server-dom-parcel-server.edge.production.js +0.34% 95.66 kB 95.99 kB +0.35% 19.64 kB 19.70 kB
oss-experimental/react-server-dom-parcel/cjs/react-server-dom-parcel-server.browser.production.js +0.34% 96.37 kB 96.70 kB +0.32% 19.75 kB 19.81 kB
oss-experimental/react-server-dom-parcel/cjs/react-server-dom-parcel-server.edge.production.js +0.34% 97.51 kB 97.84 kB +0.42% 20.01 kB 20.10 kB
oss-stable-semver/react-server-dom-esm/cjs/react-server-dom-esm-server.node.production.js +0.34% 98.13 kB 98.46 kB +0.34% 20.13 kB 20.20 kB
oss-stable/react-server-dom-esm/cjs/react-server-dom-esm-server.node.production.js +0.34% 98.13 kB 98.46 kB +0.34% 20.13 kB 20.20 kB
oss-experimental/react-art/cjs/react-art.production.js +0.33% 356.07 kB 357.25 kB +0.37% 59.95 kB 60.17 kB
oss-experimental/react-server-dom-esm/cjs/react-server-dom-esm-server.node.production.js +0.33% 99.98 kB 100.31 kB +0.31% 20.50 kB 20.57 kB
oss-stable-semver/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.browser.production.js +0.32% 101.58 kB 101.91 kB +0.28% 20.51 kB 20.57 kB
oss-stable/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.browser.production.js +0.32% 101.58 kB 101.91 kB +0.28% 20.51 kB 20.57 kB
oss-stable-semver/react-server-dom-webpack/cjs/react-server-dom-webpack-server.browser.production.js +0.32% 101.93 kB 102.26 kB +0.29% 20.61 kB 20.67 kB
oss-stable/react-server-dom-webpack/cjs/react-server-dom-webpack-server.browser.production.js +0.32% 101.93 kB 102.26 kB +0.29% 20.61 kB 20.67 kB
oss-stable-semver/react-server-dom-parcel/cjs/react-server-dom-parcel-server.node.production.js +0.32% 101.96 kB 102.29 kB +0.38% 20.65 kB 20.73 kB
oss-stable/react-server-dom-parcel/cjs/react-server-dom-parcel-server.node.production.js +0.32% 101.96 kB 102.29 kB +0.38% 20.65 kB 20.73 kB
oss-stable-semver/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.production.js +0.32% 102.74 kB 103.07 kB +0.35% 20.80 kB 20.87 kB
oss-stable/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.production.js +0.32% 102.74 kB 103.07 kB +0.35% 20.80 kB 20.87 kB
oss-stable-semver/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.edge.production.js +0.32% 102.74 kB 103.07 kB +0.35% 20.80 kB 20.87 kB
oss-stable/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.edge.production.js +0.32% 102.74 kB 103.07 kB +0.35% 20.80 kB 20.87 kB
oss-experimental/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.browser.production.js +0.32% 103.43 kB 103.76 kB +0.27% 20.95 kB 21.01 kB
oss-experimental/react-server-dom-webpack/cjs/react-server-dom-webpack-server.browser.production.js +0.32% 103.79 kB 104.12 kB +0.27% 21.05 kB 21.11 kB
oss-experimental/react-server-dom-parcel/cjs/react-server-dom-parcel-server.node.production.js +0.32% 103.82 kB 104.15 kB +0.36% 21.02 kB 21.09 kB
oss-experimental/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.production.js +0.31% 104.59 kB 104.92 kB +0.35% 21.24 kB 21.31 kB
oss-experimental/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.edge.production.js +0.31% 104.59 kB 104.92 kB +0.35% 21.24 kB 21.31 kB
oss-stable-semver/react-server-dom-webpack/cjs/react-server-dom-webpack-server.node.unbundled.production.js +0.30% 108.04 kB 108.37 kB +0.31% 21.61 kB 21.67 kB
oss-stable/react-server-dom-webpack/cjs/react-server-dom-webpack-server.node.unbundled.production.js +0.30% 108.04 kB 108.37 kB +0.31% 21.61 kB 21.67 kB
oss-stable-semver/react-server-dom-webpack/cjs/react-server-dom-webpack-server.node.production.js +0.30% 109.09 kB 109.42 kB +0.32% 21.85 kB 21.92 kB
oss-stable/react-server-dom-webpack/cjs/react-server-dom-webpack-server.node.production.js +0.30% 109.09 kB 109.42 kB +0.32% 21.85 kB 21.92 kB
oss-stable-semver/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.node.production.js +0.30% 109.11 kB 109.44 kB +0.32% 21.85 kB 21.92 kB
oss-stable/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.node.production.js +0.30% 109.11 kB 109.44 kB +0.32% 21.85 kB 21.92 kB
oss-experimental/react-server-dom-webpack/cjs/react-server-dom-webpack-server.node.unbundled.production.js +0.30% 109.89 kB 110.22 kB +0.38% 21.97 kB 22.05 kB
oss-experimental/react-server-dom-webpack/cjs/react-server-dom-webpack-server.node.production.js +0.30% 110.95 kB 111.27 kB +0.38% 22.26 kB 22.34 kB
oss-experimental/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.node.production.js +0.30% 110.96 kB 111.29 kB +0.38% 22.26 kB 22.35 kB
oss-experimental/react-reconciler/cjs/react-reconciler.production.js +0.28% 486.65 kB 487.99 kB +0.28% 77.37 kB 77.58 kB
oss-experimental/react-server-dom-esm/esm/react-server-dom-esm-client.browser.production.js +0.28% 96.35 kB 96.61 kB +0.59% 19.80 kB 19.92 kB
oss-stable-semver/react-server-dom-esm/esm/react-server-dom-esm-client.browser.production.js +0.28% 96.35 kB 96.61 kB +0.59% 19.80 kB 19.92 kB
oss-stable/react-server-dom-esm/esm/react-server-dom-esm-client.browser.production.js +0.28% 96.35 kB 96.61 kB +0.59% 19.80 kB 19.92 kB
oss-stable-semver/react-server/cjs/react-server-flight.development.js +0.27% 143.25 kB 143.63 kB +0.34% 25.54 kB 25.63 kB
oss-stable/react-server/cjs/react-server-flight.development.js +0.27% 143.25 kB 143.63 kB +0.34% 25.54 kB 25.63 kB
oss-experimental/react-server/cjs/react-server-flight.development.js +0.26% 145.34 kB 145.72 kB +0.32% 26.00 kB 26.08 kB
oss-stable-semver/react-server-dom-parcel/cjs/react-server-dom-parcel-server.browser.development.js +0.24% 186.82 kB 187.27 kB +0.27% 33.90 kB 34.00 kB
oss-stable/react-server-dom-parcel/cjs/react-server-dom-parcel-server.browser.development.js +0.24% 186.82 kB 187.27 kB +0.27% 33.90 kB 34.00 kB
oss-experimental/react-server-dom-parcel/cjs/react-server-dom-parcel-server.browser.development.js +0.24% 188.92 kB 189.37 kB +0.30% 34.35 kB 34.46 kB
oss-experimental/react-reconciler/cjs/react-reconciler.profiling.js +0.24% 562.09 kB 563.44 kB +0.24% 87.38 kB 87.59 kB
oss-stable-semver/react-server-dom-parcel/cjs/react-server-dom-parcel-server.edge.development.js +0.24% 190.49 kB 190.94 kB +0.28% 34.40 kB 34.50 kB
oss-stable/react-server-dom-parcel/cjs/react-server-dom-parcel-server.edge.development.js +0.24% 190.49 kB 190.94 kB +0.28% 34.40 kB 34.50 kB
oss-experimental/react-server-dom-parcel/cjs/react-server-dom-parcel-server.edge.development.js +0.23% 192.58 kB 193.03 kB +0.31% 34.85 kB 34.96 kB
oss-stable-semver/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.browser.development.js +0.23% 194.66 kB 195.12 kB +0.27% 35.28 kB 35.38 kB
oss-stable/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.browser.development.js +0.23% 194.66 kB 195.12 kB +0.27% 35.28 kB 35.38 kB
oss-stable-semver/react-server-dom-webpack/cjs/react-server-dom-webpack-server.browser.development.js +0.23% 195.14 kB 195.59 kB +0.27% 35.38 kB 35.48 kB
oss-stable/react-server-dom-webpack/cjs/react-server-dom-webpack-server.browser.development.js +0.23% 195.14 kB 195.59 kB +0.27% 35.38 kB 35.48 kB
oss-experimental/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.browser.development.js +0.23% 196.77 kB 197.22 kB +0.26% 35.72 kB 35.81 kB
oss-experimental/react-server-dom-webpack/cjs/react-server-dom-webpack-server.browser.development.js +0.23% 197.24 kB 197.69 kB +0.25% 35.82 kB 35.91 kB
oss-stable-semver/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.development.js +0.23% 198.37 kB 198.82 kB +0.29% 35.76 kB 35.87 kB
oss-stable/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.development.js +0.23% 198.37 kB 198.82 kB +0.29% 35.76 kB 35.87 kB
oss-stable-semver/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.edge.development.js +0.23% 198.37 kB 198.82 kB +0.28% 35.77 kB 35.87 kB
oss-stable/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.edge.development.js +0.23% 198.37 kB 198.82 kB +0.28% 35.77 kB 35.87 kB
oss-experimental/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.development.js +0.22% 200.46 kB 200.91 kB +0.31% 36.19 kB 36.30 kB
oss-experimental/react-server-dom-turbopack/cjs/react-server-dom-turbopack-server.edge.development.js +0.22% 200.46 kB 200.91 kB +0.31% 36.19 kB 36.30 kB

Generated by 🚫 dangerJS against 072d397

Copy link
Collaborator

@acdlite acdlite left a comment

Choose a reason for hiding this comment

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

Exciting! Approving since it's behind a flag.

@sebmarkbage sebmarkbage merged commit eb89912 into facebook:main Nov 18, 2025
237 of 238 checks passed
github-actions bot pushed a commit that referenced this pull request Nov 19, 2025
When dealing with optimistic state, a common problem is not knowing the
id of the thing we're waiting on. Items in lists need keys (and single
items should often have keys too to reset their state). As a result you
have to generate fake keys. It's a pain to manage those and when the
real item comes in, you often end up rendering that with a different
`key` which resets the state of the component tree. That in turns works
against the grain of React and a lot of negatives fall out of it.

This adds a special `optimisticKey` symbol that can be used in place of
a `string` key.

```js
import {optimisticKey} from 'react';
...
const [optimisticItems, setOptimisticItems] = useOptimistic([]);
const children = savedItems.concat(
  optimisticItems.map(item =>
    <Item key={optimisticKey} item={item} />
  )
);
return <div>{children}</div>;
```

The semantics of this `optimisticKey` is that the assumption is that the
newly saved item will be rendered in the same slot as the previous
optimistic items. State is transferred into whatever real key ends up in
the same slot.

This might lead to some incorrect transferring of state in some cases
where things don't end up lining up - but it's worth it for simplicity
in many cases since dealing with true matching of optimistic state is
often very complex for something that only lasts a blink of an eye.

If a new item matches a `key` elsewhere in the set, then that's favored
over reconciling against the old slot.

One quirk with the current algorithm is if the `savedItems` has items
removed, then the slots won't line up by index anymore and will be
skewed. We might be able to add something where the optimistic set is
always reconciled against the end. However, it's probably better to just
assume that the set will line up perfectly and otherwise it's just best
effort that can lead to weird artifacts.

An `optimisticKey` will match itself for updates to the same slot, but
it will not match any existing slot that is not an `optimisticKey`. So
it's not an `any`, which I originally called it, because it doesn't
match existing real keys against new optimistic keys. Only one
direction.

DiffTrain build for [eb89912](eb89912)
github-actions bot pushed a commit that referenced this pull request Nov 19, 2025
When dealing with optimistic state, a common problem is not knowing the
id of the thing we're waiting on. Items in lists need keys (and single
items should often have keys too to reset their state). As a result you
have to generate fake keys. It's a pain to manage those and when the
real item comes in, you often end up rendering that with a different
`key` which resets the state of the component tree. That in turns works
against the grain of React and a lot of negatives fall out of it.

This adds a special `optimisticKey` symbol that can be used in place of
a `string` key.

```js
import {optimisticKey} from 'react';
...
const [optimisticItems, setOptimisticItems] = useOptimistic([]);
const children = savedItems.concat(
  optimisticItems.map(item =>
    <Item key={optimisticKey} item={item} />
  )
);
return <div>{children}</div>;
```

The semantics of this `optimisticKey` is that the assumption is that the
newly saved item will be rendered in the same slot as the previous
optimistic items. State is transferred into whatever real key ends up in
the same slot.

This might lead to some incorrect transferring of state in some cases
where things don't end up lining up - but it's worth it for simplicity
in many cases since dealing with true matching of optimistic state is
often very complex for something that only lasts a blink of an eye.

If a new item matches a `key` elsewhere in the set, then that's favored
over reconciling against the old slot.

One quirk with the current algorithm is if the `savedItems` has items
removed, then the slots won't line up by index anymore and will be
skewed. We might be able to add something where the optimistic set is
always reconciled against the end. However, it's probably better to just
assume that the set will line up perfectly and otherwise it's just best
effort that can lead to weird artifacts.

An `optimisticKey` will match itself for updates to the same slot, but
it will not match any existing slot that is not an `optimisticKey`. So
it's not an `any`, which I originally called it, because it doesn't
match existing real keys against new optimistic keys. Only one
direction.

DiffTrain build for [eb89912](eb89912)
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.

3 participants