diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml index ce50c33ef..df51cd4fa 100644 --- a/.github/workflows/test.yml +++ b/.github/workflows/test.yml @@ -121,10 +121,6 @@ jobs: - name: Clone RTK repo run: git clone https://github.com/reduxjs/redux-toolkit.git ./redux-toolkit - - name: Check out v2.0-integration - working-directory: ./redux-toolkit - run: git checkout v2.0-integration - - name: Check folder contents run: ls -l . diff --git a/package.json b/package.json index 45a3a0a2a..ed1e7d6c0 100644 --- a/package.json +++ b/package.json @@ -14,12 +14,14 @@ "bugs": "https://github.com/reduxjs/react-redux/issues", "module": "dist/react-redux.legacy-esm.js", "main": "dist/cjs/index.js", + "react-native": "./dist/react-redux.react-native.mjs", "types": "dist/react-redux.d.ts", "exports": { "./package.json": "./package.json", ".": { "types": "./dist/react-redux.d.ts", "react-server": "./dist/rsc.mjs", + "react-native": "./dist/react-redux.react-native.mjs", "import": "./dist/react-redux.mjs", "default": "./dist/cjs/index.js" }, @@ -50,7 +52,7 @@ "react": "^18.0", "react-dom": "^18.0", "react-native": ">=0.71", - "redux": "^5.0.0-rc" + "redux": "^5.0.0" }, "peerDependenciesMeta": { "@types/react": { diff --git a/src/alternate-renderers.ts b/src/alternate-renderers.ts index d08c5add4..88a92f03a 100644 --- a/src/alternate-renderers.ts +++ b/src/alternate-renderers.ts @@ -3,7 +3,7 @@ // Examples include React-Three-Fiber, Ink, etc. // We'll assume they're built with React 18 and thus have `useSyncExternalStore` available. -import * as React from 'react' +import { React } from './utils/react' import { useSyncExternalStoreWithSelector } from 'use-sync-external-store/with-selector.js' import { initializeUseSelector } from './hooks/useSelector' diff --git a/src/components/Context.ts b/src/components/Context.ts index 006942227..66238ed5e 100644 --- a/src/components/Context.ts +++ b/src/components/Context.ts @@ -1,5 +1,5 @@ import type { Context } from 'react' -import * as React from 'react' +import { React } from '../utils/react' import type { Action, Store, UnknownAction } from 'redux' import type { Subscription } from '../utils/Subscription' import type { ProviderProps } from './Provider' diff --git a/src/components/Provider.tsx b/src/components/Provider.tsx index e93a94fad..afac66540 100644 --- a/src/components/Provider.tsx +++ b/src/components/Provider.tsx @@ -1,5 +1,5 @@ import type { Context, ReactNode } from 'react' -import * as React from 'react' +import { React } from '../utils/react' import type { Action, Store, UnknownAction } from 'redux' import type { DevModeCheckFrequency } from '../hooks/useSelector' import { createSubscription } from '../utils/Subscription' diff --git a/src/components/connect.tsx b/src/components/connect.tsx index bcec8eec8..769b18bac 100644 --- a/src/components/connect.tsx +++ b/src/components/connect.tsx @@ -1,6 +1,6 @@ /* eslint-disable valid-jsdoc, @typescript-eslint/no-unused-vars */ import type { ComponentType } from 'react' -import * as React from 'react' +import { React } from '../utils/react' import { isValidElementType, isContextConsumer } from '../utils/react-is' import type { Store } from 'redux' diff --git a/src/hooks/useReduxContext.ts b/src/hooks/useReduxContext.ts index fdae9b5b7..86dae5954 100644 --- a/src/hooks/useReduxContext.ts +++ b/src/hooks/useReduxContext.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { React } from '../utils/react' import { ReactReduxContext } from '../components/Context' import type { ReactReduxContextValue } from '../components/Context' diff --git a/src/hooks/useSelector.ts b/src/hooks/useSelector.ts index 9f1d7b697..629ead80d 100644 --- a/src/hooks/useSelector.ts +++ b/src/hooks/useSelector.ts @@ -1,4 +1,5 @@ -import * as React from 'react' +//import * as React from 'react' +import { React } from '../utils/react' import type { ReactReduxContextValue } from '../components/Context' import { ReactReduxContext } from '../components/Context' diff --git a/src/react-native.ts b/src/react-native.ts new file mode 100644 index 000000000..27bc27f67 --- /dev/null +++ b/src/react-native.ts @@ -0,0 +1,28 @@ +// The primary entry point assumes we are working with React 18, and thus have +// useSyncExternalStore available. We can import that directly from React itself. +// The useSyncExternalStoreWithSelector has to be imported, but we can use the +// non-shim version. This shaves off the byte size of the shim. + +import { React } from './utils/react' +import { useSyncExternalStoreWithSelector } from 'use-sync-external-store/with-selector.js' + +import { unstable_batchedUpdates as batchInternal } from './utils/reactBatchedUpdates.native' +import { setBatch } from './utils/batch' + +import { initializeUseSelector } from './hooks/useSelector' +import { initializeConnect } from './components/connect' + +initializeUseSelector(useSyncExternalStoreWithSelector) +initializeConnect(React.useSyncExternalStore) + +// Enable batched updates in our subscriptions for use +// with standard React renderers (ReactDOM, React Native) +setBatch(batchInternal) + +// Avoid needing `react-dom` in the final TS types +// by providing a simpler type for `batch` +const batch: (cb: () => void) => void = batchInternal + +export { batch } + +export * from './exports' diff --git a/src/utils/react.ts b/src/utils/react.ts new file mode 100644 index 000000000..faab8eceb --- /dev/null +++ b/src/utils/react.ts @@ -0,0 +1,7 @@ +import * as ReactOriginal from 'react' +import type * as ReactNamespace from 'react' + +export const React: typeof ReactNamespace = + // prettier-ignore + // @ts-ignore + 'default' in ReactOriginal ? ReactOriginal['default'] : ReactOriginal as any diff --git a/src/utils/useIsomorphicLayoutEffect.native.ts b/src/utils/useIsomorphicLayoutEffect.native.ts index 80a30f17a..c3b22c296 100644 --- a/src/utils/useIsomorphicLayoutEffect.native.ts +++ b/src/utils/useIsomorphicLayoutEffect.native.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { React } from '../utils/react' // Under React Native, we know that we always want to use useLayoutEffect diff --git a/src/utils/useIsomorphicLayoutEffect.ts b/src/utils/useIsomorphicLayoutEffect.ts index 0cb1776f1..329c00e10 100644 --- a/src/utils/useIsomorphicLayoutEffect.ts +++ b/src/utils/useIsomorphicLayoutEffect.ts @@ -1,4 +1,4 @@ -import * as React from 'react' +import { React } from '../utils/react' // React currently throws a warning when using useLayoutEffect on the server. // To get around it, we can conditionally useEffect on the server (no-op) and diff --git a/tsup.config.ts b/tsup.config.ts index 81a61b1c2..95925fd44 100644 --- a/tsup.config.ts +++ b/tsup.config.ts @@ -76,6 +76,15 @@ export default defineConfig((options) => { format: ['esm'], outExtension: () => ({ js: '.mjs' }), }, + // React Native requires a separate entry point for `"react-native"` batch dep + { + ...commonOptions, + entry: { + 'react-redux.react-native': 'src/react-native.ts', + }, + format: ['esm'], + outExtension: () => ({ js: '.mjs' }), + }, // CJS development { ...commonOptions, diff --git a/yarn.lock b/yarn.lock index 38c2859b8..17f8ee656 100644 --- a/yarn.lock +++ b/yarn.lock @@ -9417,7 +9417,7 @@ __metadata: react: ^18.0 react-dom: ^18.0 react-native: ">=0.71" - redux: ^5.0.0-rc + redux: ^5.0.0 peerDependenciesMeta: "@types/react": optional: true