Skip to content

Commit

Permalink
Add moveBefore Experiment (#31596)
Browse files Browse the repository at this point in the history
A long standing issue for React has been that if you reorder stateful
nodes, they may lose their state and reload. The thing moving loses its
state. There's no way to solve this in general where two stateful nodes
swap.

The [`moveBefore()`
proposal](https://chromestatus.com/feature/5135990159835136?gate=5177450351558656)
has now moved to
[intent-to-ship](https://groups.google.com/a/chromium.org/g/blink-dev/c/YE_xLH6MkRs/m/_7CD0NYMAAAJ).
This function is kind of like `insertBefore` but preserves state.

There's [a demo here](https://state-preserving-atomic-move.glitch.me/).
Ideally we'd port this demo to a fixture so we can try it.

Currently this flag is always off - even in experimental. That's because
this is still behind a Chrome flag so it's a little early to turn it on
even in experimental. So you need a custom build. It's on in RN but only
because it doesn't apply there which makes it easier to tell that it's
safe to ship once it's on everywhere else.

The other reason it's still off is because there's currently a semantic
breaking change. `moveBefore()` errors if both nodes are disconnected.
That happens if we're inside a completely disconnected React root.
That's not usually how you should use React because it means effects
can't read layout etc. However, it is currently supported. To handle
this we'd have to try/catch the `moveBefore` to handle this case but we
hope this semantic will change before it ships. Before we turn this on
in experimental we either have to wait for the implementation to not
error in the disconnected-disconnected case in Chrome or we'd have to
add try/catch.

DiffTrain build for [aba370f](aba370f)
  • Loading branch information
sebmarkbage committed Nov 22, 2024
1 parent b78a8b1 commit c88a1f2
Show file tree
Hide file tree
Showing 23 changed files with 146 additions and 92 deletions.
2 changes: 1 addition & 1 deletion compiled-rn/VERSION_NATIVE_FB
Original file line number Diff line number Diff line change
@@ -1 +1 @@
19.0.0-native-fb-c11c9510-20241120
19.0.0-native-fb-aba370f1-20241122
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<2185e4149a140a1970869525d77110da>>
* @generated SignedSource<<712b04c3a26023e4bea59df930cf674a>>
*/

"use strict";
Expand Down Expand Up @@ -420,5 +420,5 @@ __DEV__ &&
exports.useFormStatus = function () {
return resolveDispatcher().useHostTransitionStatus();
};
exports.version = "19.0.0-native-fb-c11c9510-20241120";
exports.version = "19.0.0-native-fb-aba370f1-20241122";
})();
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<fd16f72b5915df19d7034a54360181c3>>
* @generated SignedSource<<81c01fe21e27672bef833613e7a3e25c>>
*/

"use strict";
Expand Down Expand Up @@ -203,4 +203,4 @@ exports.useFormState = function (action, initialState, permalink) {
exports.useFormStatus = function () {
return ReactSharedInternals.H.useHostTransitionStatus();
};
exports.version = "19.0.0-native-fb-c11c9510-20241120";
exports.version = "19.0.0-native-fb-aba370f1-20241122";
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<fd16f72b5915df19d7034a54360181c3>>
* @generated SignedSource<<81c01fe21e27672bef833613e7a3e25c>>
*/

"use strict";
Expand Down Expand Up @@ -203,4 +203,4 @@ exports.useFormState = function (action, initialState, permalink) {
exports.useFormStatus = function () {
return ReactSharedInternals.H.useHostTransitionStatus();
};
exports.version = "19.0.0-native-fb-c11c9510-20241120";
exports.version = "19.0.0-native-fb-aba370f1-20241122";
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<6243eeb8f8a958dce0f7d61322ac04ff>>
* @generated SignedSource<<0d89e6bc8382f0cef2e9d75790b40d5e>>
*/

/*
Expand Down Expand Up @@ -11874,7 +11874,13 @@ __DEV__ &&
var tag = node.tag;
if (5 === tag || 6 === tag)
(node = node.stateNode),
before ? parent.insertBefore(node, before) : parent.appendChild(node);
before
? supportsMoveBefore
? parent.moveBefore(node, before)
: parent.insertBefore(node, before)
: supportsMoveBefore
? parent.moveBefore(node, null)
: parent.appendChild(node);
else if (4 !== tag && 27 !== tag && ((node = node.child), null !== node))
for (
insertOrAppendPlacementNode(node, before, parent),
Expand Down Expand Up @@ -25339,6 +25345,9 @@ __DEV__ &&
.catch(handleErrorInNextTick);
}
: scheduleTimeout,
supportsMoveBefore =
"undefined" !== typeof window &&
"function" === typeof window.Node.prototype.moveBefore,
NotLoaded = 0,
Loaded = 1,
Errored = 2,
Expand Down Expand Up @@ -25759,11 +25768,11 @@ __DEV__ &&
};
(function () {
var isomorphicReactPackageVersion = React.version;
if ("19.0.0-native-fb-c11c9510-20241120" !== isomorphicReactPackageVersion)
if ("19.0.0-native-fb-aba370f1-20241122" !== isomorphicReactPackageVersion)
throw Error(
'Incompatible React versions: The "react" and "react-dom" packages must have the exact same version. Instead got:\n - react: ' +
(isomorphicReactPackageVersion +
"\n - react-dom: 19.0.0-native-fb-c11c9510-20241120\nLearn more: https://react.dev/warnings/version-mismatch")
"\n - react-dom: 19.0.0-native-fb-aba370f1-20241122\nLearn more: https://react.dev/warnings/version-mismatch")
);
})();
("function" === typeof Map &&
Expand Down Expand Up @@ -25800,11 +25809,11 @@ __DEV__ &&
!(function () {
var internals = {
bundleType: 1,
version: "19.0.0-native-fb-c11c9510-20241120",
version: "19.0.0-native-fb-aba370f1-20241122",
rendererPackageName: "react-dom",
currentDispatcherRef: ReactSharedInternals,
findFiberByHostInstance: getClosestInstanceFromNode,
reconcilerVersion: "19.0.0-native-fb-c11c9510-20241120"
reconcilerVersion: "19.0.0-native-fb-aba370f1-20241122"
};
internals.overrideHookState = overrideHookState;
internals.overrideHookStateDeletePath = overrideHookStateDeletePath;
Expand Down Expand Up @@ -25948,5 +25957,5 @@ __DEV__ &&
listenToAllSupportedEvents(container);
return new ReactDOMHydrationRoot(initialChildren);
};
exports.version = "19.0.0-native-fb-c11c9510-20241120";
exports.version = "19.0.0-native-fb-aba370f1-20241122";
})();
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<7b01fb0b70c13edae17ecac2234f6026>>
* @generated SignedSource<<bbe9c9df4aeffbacb96929a845d03786>>
*/

/*
Expand Down Expand Up @@ -8271,7 +8271,13 @@ function insertOrAppendPlacementNode(node, before, parent) {
var tag = node.tag;
if (5 === tag || 6 === tag)
(node = node.stateNode),
before ? parent.insertBefore(node, before) : parent.appendChild(node);
before
? supportsMoveBefore
? parent.moveBefore(node, before)
: parent.insertBefore(node, before)
: supportsMoveBefore
? parent.moveBefore(node, null)
: parent.appendChild(node);
else if (4 !== tag && 27 !== tag && ((node = node.child), null !== node))
for (
insertOrAppendPlacementNode(node, before, parent), node = node.sibling;
Expand Down Expand Up @@ -14127,6 +14133,9 @@ function handleErrorInNextTick(error) {
throw error;
});
}
var supportsMoveBefore =
"undefined" !== typeof window &&
"function" === typeof window.Node.prototype.moveBefore;
function clearSuspenseBoundary(parentInstance, suspenseInstance) {
var node = suspenseInstance,
depth = 0;
Expand Down Expand Up @@ -15829,14 +15838,14 @@ ReactDOMHydrationRoot.prototype.unstable_scheduleHydration = function (target) {
};
var isomorphicReactPackageVersion$jscomp$inline_1735 = React.version;
if (
"19.0.0-native-fb-c11c9510-20241120" !==
"19.0.0-native-fb-aba370f1-20241122" !==
isomorphicReactPackageVersion$jscomp$inline_1735
)
throw Error(
formatProdErrorMessage(
527,
isomorphicReactPackageVersion$jscomp$inline_1735,
"19.0.0-native-fb-c11c9510-20241120"
"19.0.0-native-fb-aba370f1-20241122"
)
);
ReactDOMSharedInternals.findDOMNode = function (componentOrElement) {
Expand All @@ -15858,11 +15867,11 @@ ReactDOMSharedInternals.findDOMNode = function (componentOrElement) {
};
var internals$jscomp$inline_2193 = {
bundleType: 0,
version: "19.0.0-native-fb-c11c9510-20241120",
version: "19.0.0-native-fb-aba370f1-20241122",
rendererPackageName: "react-dom",
currentDispatcherRef: ReactSharedInternals,
findFiberByHostInstance: getClosestInstanceFromNode,
reconcilerVersion: "19.0.0-native-fb-c11c9510-20241120"
reconcilerVersion: "19.0.0-native-fb-aba370f1-20241122"
};
if ("undefined" !== typeof __REACT_DEVTOOLS_GLOBAL_HOOK__) {
var hook$jscomp$inline_2194 = __REACT_DEVTOOLS_GLOBAL_HOOK__;
Expand Down Expand Up @@ -15966,4 +15975,4 @@ exports.hydrateRoot = function (container, initialChildren, options) {
listenToAllSupportedEvents(container);
return new ReactDOMHydrationRoot(initialChildren);
};
exports.version = "19.0.0-native-fb-c11c9510-20241120";
exports.version = "19.0.0-native-fb-aba370f1-20241122";
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<9e48210564a143f24ae3d2ac4d1f0a42>>
* @generated SignedSource<<4646449a1fbaa134be54e6ac1df8a36f>>
*/

/*
Expand Down Expand Up @@ -8631,7 +8631,13 @@ function insertOrAppendPlacementNode(node, before, parent) {
var tag = node.tag;
if (5 === tag || 6 === tag)
(node = node.stateNode),
before ? parent.insertBefore(node, before) : parent.appendChild(node);
before
? supportsMoveBefore
? parent.moveBefore(node, before)
: parent.insertBefore(node, before)
: supportsMoveBefore
? parent.moveBefore(node, null)
: parent.appendChild(node);
else if (4 !== tag && 27 !== tag && ((node = node.child), null !== node))
for (
insertOrAppendPlacementNode(node, before, parent), node = node.sibling;
Expand Down Expand Up @@ -14772,6 +14778,9 @@ function handleErrorInNextTick(error) {
throw error;
});
}
var supportsMoveBefore =
"undefined" !== typeof window &&
"function" === typeof window.Node.prototype.moveBefore;
function clearSuspenseBoundary(parentInstance, suspenseInstance) {
var node = suspenseInstance,
depth = 0;
Expand Down Expand Up @@ -16482,14 +16491,14 @@ ReactDOMHydrationRoot.prototype.unstable_scheduleHydration = function (target) {
};
var isomorphicReactPackageVersion$jscomp$inline_1825 = React.version;
if (
"19.0.0-native-fb-c11c9510-20241120" !==
"19.0.0-native-fb-aba370f1-20241122" !==
isomorphicReactPackageVersion$jscomp$inline_1825
)
throw Error(
formatProdErrorMessage(
527,
isomorphicReactPackageVersion$jscomp$inline_1825,
"19.0.0-native-fb-c11c9510-20241120"
"19.0.0-native-fb-aba370f1-20241122"
)
);
ReactDOMSharedInternals.findDOMNode = function (componentOrElement) {
Expand All @@ -16511,11 +16520,11 @@ ReactDOMSharedInternals.findDOMNode = function (componentOrElement) {
};
var internals$jscomp$inline_1832 = {
bundleType: 0,
version: "19.0.0-native-fb-c11c9510-20241120",
version: "19.0.0-native-fb-aba370f1-20241122",
rendererPackageName: "react-dom",
currentDispatcherRef: ReactSharedInternals,
findFiberByHostInstance: getClosestInstanceFromNode,
reconcilerVersion: "19.0.0-native-fb-c11c9510-20241120",
reconcilerVersion: "19.0.0-native-fb-aba370f1-20241122",
getLaneLabelMap: function () {
for (
var map = new Map(), lane = 1, index$292 = 0;
Expand Down Expand Up @@ -16634,4 +16643,4 @@ exports.hydrateRoot = function (container, initialChildren, options) {
listenToAllSupportedEvents(container);
return new ReactDOMHydrationRoot(initialChildren);
};
exports.version = "19.0.0-native-fb-c11c9510-20241120";
exports.version = "19.0.0-native-fb-aba370f1-20241122";
Original file line number Diff line number Diff line change
Expand Up @@ -7,7 +7,7 @@
* @noflow
* @nolint
* @preventMunge
* @generated SignedSource<<f5d26777932583272414736718d386d4>>
* @generated SignedSource<<fdb540f0caca4805185f18de2b75187e>>
*/

/*
Expand Down Expand Up @@ -11882,7 +11882,13 @@ __DEV__ &&
var tag = node.tag;
if (5 === tag || 6 === tag)
(node = node.stateNode),
before ? parent.insertBefore(node, before) : parent.appendChild(node);
before
? supportsMoveBefore
? parent.moveBefore(node, before)
: parent.insertBefore(node, before)
: supportsMoveBefore
? parent.moveBefore(node, null)
: parent.appendChild(node);
else if (4 !== tag && 27 !== tag && ((node = node.child), null !== node))
for (
insertOrAppendPlacementNode(node, before, parent),
Expand Down Expand Up @@ -25400,6 +25406,9 @@ __DEV__ &&
.catch(handleErrorInNextTick);
}
: scheduleTimeout,
supportsMoveBefore =
"undefined" !== typeof window &&
"function" === typeof window.Node.prototype.moveBefore,
NotLoaded = 0,
Loaded = 1,
Errored = 2,
Expand Down Expand Up @@ -25820,11 +25829,11 @@ __DEV__ &&
};
(function () {
var isomorphicReactPackageVersion = React.version;
if ("19.0.0-native-fb-c11c9510-20241120" !== isomorphicReactPackageVersion)
if ("19.0.0-native-fb-aba370f1-20241122" !== isomorphicReactPackageVersion)
throw Error(
'Incompatible React versions: The "react" and "react-dom" packages must have the exact same version. Instead got:\n - react: ' +
(isomorphicReactPackageVersion +
"\n - react-dom: 19.0.0-native-fb-c11c9510-20241120\nLearn more: https://react.dev/warnings/version-mismatch")
"\n - react-dom: 19.0.0-native-fb-aba370f1-20241122\nLearn more: https://react.dev/warnings/version-mismatch")
);
})();
("function" === typeof Map &&
Expand Down Expand Up @@ -25861,11 +25870,11 @@ __DEV__ &&
!(function () {
var internals = {
bundleType: 1,
version: "19.0.0-native-fb-c11c9510-20241120",
version: "19.0.0-native-fb-aba370f1-20241122",
rendererPackageName: "react-dom",
currentDispatcherRef: ReactSharedInternals,
findFiberByHostInstance: getClosestInstanceFromNode,
reconcilerVersion: "19.0.0-native-fb-c11c9510-20241120"
reconcilerVersion: "19.0.0-native-fb-aba370f1-20241122"
};
internals.overrideHookState = overrideHookState;
internals.overrideHookStateDeletePath = overrideHookStateDeletePath;
Expand Down Expand Up @@ -26325,7 +26334,7 @@ __DEV__ &&
exports.useFormStatus = function () {
return resolveDispatcher().useHostTransitionStatus();
};
exports.version = "19.0.0-native-fb-c11c9510-20241120";
exports.version = "19.0.0-native-fb-aba370f1-20241122";
"undefined" !== typeof __REACT_DEVTOOLS_GLOBAL_HOOK__ &&
"function" ===
typeof __REACT_DEVTOOLS_GLOBAL_HOOK__.registerInternalModuleStop &&
Expand Down
Loading

0 comments on commit c88a1f2

Please sign in to comment.