-
Notifications
You must be signed in to change notification settings - Fork 1.1k
Forking upstream files with react‐native‐platform‐override
We often need to patch (or derive new files from) upstream react-native
files in react-native-windows
. As we're not a proper fork of react-native
, we rely on a tool called react-native-platform-override
to help us maintain our changes.
Let's say we need to make some RNW-specific changes to an upstream RN file, either temporarily to unblock a build (until we can get the file fixed upstream) or something that wouldn't even make sense to submit an PR upstream for. To do that we're going to create an override file in the RNW repo that will replace the RN version.
RNW maintains the src-win
folder to hold its overrides. Its structure mirrors the root structure of the upstream react-native
package.
So for example, let's say we want to patch RN's Libraries/Pressability/HoverState.js
file.
- First, copy the upstream file into the appropriate place in RNW's
src-win
folder. In this example, that would mean copyingnode_modules/react-native/Libraries/Pressability/HoverState.js
tovnext/src-win/Libraries/Pressability/HoverState.js
. - Next, make you changes to that new file. Be sure to clearly mark your changes by surrounding code blocks with starting
// [Windows
and ending// Windows]
comments. For this example, here we're adding a newelse if
clause to anif
statement performing a platform check:+ // [Windows +} else if (Platform.OS === 'windows') { + isEnabled = true; + // Windows]
- Finally, we need to record the patched file by creating a new "override" using the
react-native-platform-override
command. From thevnext
folder, run:yarn react-native-platform-override add src-win/Libraries/Pressability/HoverState.js
- When prompted, select the appropriate "override type". In this case, since we're patching an existing file, choose:
Patches to upstream
. - When prompted, enter a GH issue number to track removing the patch (useful if the patch is only temporary while waiting for it to be merged upstream). For a "permanent" patch like in this example, simply enter
0
. - When prompted for the original file this override is based off of, you need to give the path of the upstream file including a prefix indicating which package it came from. In this example, overriding a
react-native
package, we'll prefix the relative path withpackages/react-native
, giving us:packages/react-native/Libraries/Pressability/HoverState.js
.
There should now be a new entry in the vnext/overrides.json
file. This will ensure that during integrations our tooling will keep this file up-to-date with upstream changes (while still maintaining our patches).
To test your changes, the final step is to run yarn build
in the vnext
folder to make sure the new override is applied locally. You may need to close and restart any Metro servers you have running to see the changes applied.
Let's say we need to make a RNW-specific version of an upstream file. The Metro bundler allows you to create platform-specific modules by creating a new file with the platform's identifier at the end of the name. For example, let's say there's the upstream LogBoxInspectorCodeFrame.js
and we want to create our own LogBoxInspectorCodeFrame.windows.js
version.
The steps are almost entirely the same as for patching upstream JS files. First we'd want to copy the file from node_modules/react-native/Libraries/LogBox/UI/LogBoxInspectorCodeFrame.js
to vnext/src-win/Libraries/LogBox/UI/LogBoxInspectorCodeFrame.windows.js
.
Continue with the steps above, the only difference is, when asked for the "override type", choose: Derived from upstream
. This lets the tooling know that we're not just patching an existing file, we're providing a new file that is derived from it.
Let's say there's an issue in one of the upstream native files in ReactCommon. We build ReactCommon in vnext\ReactCommon
but sometimes upstream changes either throw warnings (or don't build at all) with MSVC. So until we can get the issue fixed upstream, we can create fork the upstream file into the structure under vnext/ReactCommon/TEMP_UntilReactCommonUpdate
, which we will use in our builds. For example, let's say there's the upstream NativeDOM.cpp
that is throwing some build warnings that we want to address.
The steps are almost entirely the same as for patching upstream JS files. First we'd want to copy the file from node_modules/react-native/ReactCommon/react/nativemodule/dom/NativeDOM.cpp
to vnext/ReactCommon/TEMP_UntilReactCommonUpdate/react/nativemodule/dom/NativeDOM.cpp
. Note that we need to replicate the relative path structure under the upstream ReactCommon
under our TEMP_UntilReactCommonUpdate
.
Continue with the steps above, and be sure to open a tracking issue for removal of the override. There are some exceptions where we've had long-standing forks of ReactCommon files, but in general, our native builds are just "stricter" about some things than they are upstream, so it's usually not that hard to get upstream fixes and the override removed in a future integration.