From 9fdc18d24a63e2cafbfbba6a5d71cae3c89c0007 Mon Sep 17 00:00:00 2001 From: Mark Erikson Date: Wed, 21 Jun 2023 21:15:01 -0400 Subject: [PATCH 1/4] Remove GA --- website/docusaurus.config.js | 3 --- 1 file changed, 3 deletions(-) diff --git a/website/docusaurus.config.js b/website/docusaurus.config.js index 59d4459d9..64bc05374 100644 --- a/website/docusaurus.config.js +++ b/website/docusaurus.config.js @@ -31,9 +31,6 @@ const siteConfig = { require.resolve('./static/css/codeblock.css'), ], }, - googleAnalytics: { - trackingID: 'UA-130598673-2', - }, }, ], ], From a669a94e0968386eacedddac30031e14c8204ae3 Mon Sep 17 00:00:00 2001 From: Lenz Weber-Tronic Date: Sat, 29 Jul 2023 17:48:42 +0200 Subject: [PATCH 2/4] add "publish" workflow --- .github/workflows/publish.yaml | 24 ++++++++++++++++++++++++ 1 file changed, 24 insertions(+) create mode 100644 .github/workflows/publish.yaml diff --git a/.github/workflows/publish.yaml b/.github/workflows/publish.yaml new file mode 100644 index 000000000..994956277 --- /dev/null +++ b/.github/workflows/publish.yaml @@ -0,0 +1,24 @@ +name: Publish Package to npmjs +on: + # keeping it purely manual for now as to not accidentally trigger a release + #release: + # types: [published] + workflow_dispatch: +jobs: + publish: + runs-on: ubuntu-latest + permissions: + id-token: write + contents: read + steps: + - uses: actions/checkout@v3 + - uses: actions/setup-node@v3 + with: + node-version: '18.x' + registry-url: 'https://registry.npmjs.org/' + cache: 'yarn' + - run: yarn install --frozen-lockfile + - run: yarn test + - run: npm publish --access public --provenance + env: + NODE_AUTH_TOKEN: ${{ secrets.NPM_TOKEN }} \ No newline at end of file From 2ac527bc1681b9cf82b269369814419fa8e7a2bc Mon Sep 17 00:00:00 2001 From: Lenz Weber-Tronic Date: Sat, 29 Jul 2023 17:52:38 +0200 Subject: [PATCH 3/4] RSC-specific workarounds (#2050) * map context instances using `createContext` as key * use `{}` as ReactReduxContext in environments where no context exists (removes `Proxy` use) * switch React imports to namespace imports to prevent hook import detection * use `globalThis` only when availbalbe --- src/components/Context.ts | 39 +++++++++---------- src/components/Provider.tsx | 6 +-- src/components/connect.tsx | 36 ++++++++--------- src/next.ts | 4 +- src/utils/useIsomorphicLayoutEffect.native.ts | 4 +- src/utils/useIsomorphicLayoutEffect.ts | 6 ++- 6 files changed, 48 insertions(+), 47 deletions(-) diff --git a/src/components/Context.ts b/src/components/Context.ts index 72e3fb10e..6c5607a8f 100644 --- a/src/components/Context.ts +++ b/src/components/Context.ts @@ -1,4 +1,4 @@ -import { createContext, version as ReactVersion } from 'react' +import * as React from 'react' import type { Context } from 'react' import type { Action, AnyAction, Store } from 'redux' import type { Subscription } from '../utils/Subscription' @@ -15,34 +15,33 @@ export interface ReactReduxContextValue< noopCheck: CheckFrequency } -const ContextKey = Symbol.for(`react-redux-context-${ReactVersion}`) -const gT = globalThis as { [ContextKey]?: Context } +const ContextKey = Symbol.for(`react-redux-context`) +const gT: { + [ContextKey]?: Map< + typeof React.createContext, + Context + > +} = (typeof globalThis !== "undefined" ? globalThis : /* fall back to a per-module scope (pre-8.1 behaviour) if `globalThis` is not available */ {}) as any; -function getContext() { - let realContext = gT[ContextKey] +function getContext(): Context { + if (!React.createContext) return {} as any + + const contextMap = (gT[ContextKey] ??= new Map< + typeof React.createContext, + Context + >()) + let realContext = contextMap.get(React.createContext) if (!realContext) { - realContext = createContext(null as any) + realContext = React.createContext(null as any) if (process.env.NODE_ENV !== 'production') { realContext.displayName = 'ReactRedux' } - gT[ContextKey] = realContext + contextMap.set(React.createContext, realContext) } return realContext } -export const ReactReduxContext = /*#__PURE__*/ new Proxy( - {} as Context, - /*#__PURE__*/ new Proxy>>( - {}, - { - get(_, handler) { - const target = getContext() - // @ts-ignore - return (_target, ...args) => Reflect[handler](target, ...args) - }, - } - ) -) +export const ReactReduxContext = /*#__PURE__*/ getContext() export type ReactReduxContextInstance = typeof ReactReduxContext diff --git a/src/components/Provider.tsx b/src/components/Provider.tsx index 3d8bdf75c..170860c21 100644 --- a/src/components/Provider.tsx +++ b/src/components/Provider.tsx @@ -1,5 +1,5 @@ import type { Context, ReactNode } from 'react' -import React, { useMemo } from 'react' +import * as React from 'react' import type { ReactReduxContextValue } from './Context' import { ReactReduxContext } from './Context' import { createSubscription } from '../utils/Subscription' @@ -42,7 +42,7 @@ function Provider({ stabilityCheck = 'once', noopCheck = 'once', }: ProviderProps) { - const contextValue = useMemo(() => { + const contextValue = React.useMemo(() => { const subscription = createSubscription(store) return { store, @@ -53,7 +53,7 @@ function Provider({ } }, [store, serverState, stabilityCheck, noopCheck]) - const previousState = useMemo(() => store.getState(), [store]) + const previousState = React.useMemo(() => store.getState(), [store]) useIsomorphicLayoutEffect(() => { const { subscription } = contextValue diff --git a/src/components/connect.tsx b/src/components/connect.tsx index 77fb273d2..d76025eeb 100644 --- a/src/components/connect.tsx +++ b/src/components/connect.tsx @@ -1,7 +1,7 @@ /* eslint-disable valid-jsdoc, @typescript-eslint/no-unused-vars */ import hoistStatics from 'hoist-non-react-statics' import type { ComponentType } from 'react' -import React, { useContext, useMemo, useRef } from 'react' +import * as React from 'react' import { isValidElementType, isContextConsumer } from 'react-is' import type { Store } from 'redux' @@ -533,7 +533,7 @@ function connect< props: InternalConnectProps & TOwnProps ) { const [propsContext, reactReduxForwardedRef, wrapperProps] = - useMemo(() => { + React.useMemo(() => { // Distinguish between actual "data" props that were passed to the wrapper component, // and values needed to control behavior (forwarded refs, alternate context instances). // To maintain the wrapperProps object reference, memoize this destructuring. @@ -541,7 +541,7 @@ function connect< return [props.context, reactReduxForwardedRef, wrapperProps] }, [props]) - const ContextToUse: ReactReduxContextInstance = useMemo(() => { + const ContextToUse: ReactReduxContextInstance = React.useMemo(() => { // Users may optionally pass in a custom context instance to use instead of our ReactReduxContext. // Memoize the check that determines which context instance we should use. return propsContext && @@ -553,7 +553,7 @@ function connect< }, [propsContext, Context]) // Retrieve the store and ancestor subscription via context, if available - const contextValue = useContext(ContextToUse) + const contextValue = React.useContext(ContextToUse) // The store _must_ exist as either a prop or in context. // We'll check to see if it _looks_ like a Redux store first. @@ -587,13 +587,13 @@ function connect< ? contextValue.getServerState : store.getState - const childPropsSelector = useMemo(() => { + const childPropsSelector = React.useMemo(() => { // The child props selector needs the store reference as an input. // Re-create this selector whenever the store changes. return defaultSelectorFactory(store.dispatch, selectorFactoryOptions) }, [store]) - const [subscription, notifyNestedSubs] = useMemo(() => { + const [subscription, notifyNestedSubs] = React.useMemo(() => { if (!shouldHandleStateChanges) return NO_SUBSCRIPTION_ARRAY // This Subscription's source should match where store came from: props vs. context. A component @@ -615,7 +615,7 @@ function connect< // Determine what {store, subscription} value should be put into nested context, if necessary, // and memoize that value to avoid unnecessary context updates. - const overriddenContextValue = useMemo(() => { + const overriddenContextValue = React.useMemo(() => { if (didStoreComeFromProps) { // This component is directly subscribed to a store from props. // We don't want descendants reading from this store - pass down whatever @@ -632,14 +632,14 @@ function connect< }, [didStoreComeFromProps, contextValue, subscription]) // Set up refs to coordinate values between the subscription effect and the render logic - const lastChildProps = useRef() - const lastWrapperProps = useRef(wrapperProps) - const childPropsFromStoreUpdate = useRef() - const renderIsScheduled = useRef(false) - const isProcessingDispatch = useRef(false) - const isMounted = useRef(false) + const lastChildProps = React.useRef() + const lastWrapperProps = React.useRef(wrapperProps) + const childPropsFromStoreUpdate = React.useRef() + const renderIsScheduled = React.useRef(false) + const isProcessingDispatch = React.useRef(false) + const isMounted = React.useRef(false) - const latestSubscriptionCallbackError = useRef() + const latestSubscriptionCallbackError = React.useRef() useIsomorphicLayoutEffect(() => { isMounted.current = true @@ -648,7 +648,7 @@ function connect< } }, []) - const actualChildPropsSelector = useMemo(() => { + const actualChildPropsSelector = React.useMemo(() => { const selector = () => { // Tricky logic here: // - This render may have been triggered by a Redux store update that produced new child props @@ -676,7 +676,7 @@ function connect< // about useLayoutEffect in SSR, so we try to detect environment and fall back to // just useEffect instead to avoid the warning, since neither will run anyway. - const subscribeForReact = useMemo(() => { + const subscribeForReact = React.useMemo(() => { const subscribe = (reactListener: () => void) => { if (!subscription) { return () => {} @@ -741,7 +741,7 @@ function connect< // Now that all that's done, we can finally try to actually render the child component. // We memoize the elements for the rendered child component as an optimization. - const renderedWrappedComponent = useMemo(() => { + const renderedWrappedComponent = React.useMemo(() => { return ( // @ts-ignore { + const renderedChild = React.useMemo(() => { if (shouldHandleStateChanges) { // If this component is subscribed to store updates, we need to pass its own // subscription instance down to our descendants. That means rendering the same diff --git a/src/next.ts b/src/next.ts index c030ab101..0da9fcdc9 100644 --- a/src/next.ts +++ b/src/next.ts @@ -3,7 +3,7 @@ // The useSyncExternalStoreWithSelector has to be imported, but we can use the // non-shim version. This shaves off the byte size of the shim. -import { useSyncExternalStore } from 'react' +import * as React from 'react' import { useSyncExternalStoreWithSelector } from 'use-sync-external-store/with-selector' import { unstable_batchedUpdates as batch } from './utils/reactBatchedUpdates' @@ -13,7 +13,7 @@ import { initializeUseSelector } from './hooks/useSelector' import { initializeConnect } from './components/connect' initializeUseSelector(useSyncExternalStoreWithSelector) -initializeConnect(useSyncExternalStore) +initializeConnect(React.useSyncExternalStore) // Enable batched updates in our subscriptions for use // with standard React renderers (ReactDOM, React Native) diff --git a/src/utils/useIsomorphicLayoutEffect.native.ts b/src/utils/useIsomorphicLayoutEffect.native.ts index e80393ad9..80a30f17a 100644 --- a/src/utils/useIsomorphicLayoutEffect.native.ts +++ b/src/utils/useIsomorphicLayoutEffect.native.ts @@ -1,5 +1,5 @@ -import { useLayoutEffect } from 'react' +import * as React from 'react' // Under React Native, we know that we always want to use useLayoutEffect -export const useIsomorphicLayoutEffect = useLayoutEffect +export const useIsomorphicLayoutEffect = React.useLayoutEffect diff --git a/src/utils/useIsomorphicLayoutEffect.ts b/src/utils/useIsomorphicLayoutEffect.ts index d60f51c2e..0cb1776f1 100644 --- a/src/utils/useIsomorphicLayoutEffect.ts +++ b/src/utils/useIsomorphicLayoutEffect.ts @@ -1,4 +1,4 @@ -import { useEffect, useLayoutEffect } from 'react' +import * as React from '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 @@ -16,4 +16,6 @@ export const canUseDOM = !!( typeof window.document.createElement !== 'undefined' ) -export const useIsomorphicLayoutEffect = canUseDOM ? useLayoutEffect : useEffect +export const useIsomorphicLayoutEffect = canUseDOM + ? React.useLayoutEffect + : React.useEffect From 4a7e129ee537d35c53da258ccf7924a0376391ee Mon Sep 17 00:00:00 2001 From: Lenz Weber-Tronic Date: Sat, 29 Jul 2023 18:02:15 +0200 Subject: [PATCH 4/4] Release 8.1.2 --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 99f275685..bb229f126 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "react-redux", - "version": "8.1.1", + "version": "8.1.2", "description": "Official React bindings for Redux", "keywords": [ "react",