Skip to content

Commit

Permalink
Fix useSelector() in combination with lazy loaded components breaks w…
Browse files Browse the repository at this point in the history
…ith react v18 (#1977)
  • Loading branch information
jeroenpx committed Sep 14, 2023
1 parent a1a32d1 commit 6c0ad9e
Show file tree
Hide file tree
Showing 2 changed files with 164 additions and 5 deletions.
43 changes: 38 additions & 5 deletions src/utils/Subscription.ts
Original file line number Diff line number Diff line change
Expand Up @@ -99,9 +99,26 @@ export function createSubscription(store: any, parentSub?: Subscription) {
let unsubscribe: VoidFunc | undefined
let listeners: ListenerCollection = nullListeners

// Reasons to keep the subscription active
let subscriptionsAmount = 0

// Is this specific subscription subscribed (or only nested ones?)
let selfSubscribed = false

function addNestedSub(listener: () => void) {
trySubscribe()
return listeners.subscribe(listener)

const cleanupListener = listeners.subscribe(listener)

// cleanup nested sub
let removed = false
return () => {
if (!removed) {
removed = true
cleanupListener()
tryUnsubscribe()
}
}
}

function notifyNestedSubs() {
Expand All @@ -115,10 +132,11 @@ export function createSubscription(store: any, parentSub?: Subscription) {
}

function isSubscribed() {
return Boolean(unsubscribe)
return selfSubscribed
}

function trySubscribe() {
subscriptionsAmount++
if (!unsubscribe) {
unsubscribe = parentSub
? parentSub.addNestedSub(handleChangeWrapper)
Expand All @@ -129,21 +147,36 @@ export function createSubscription(store: any, parentSub?: Subscription) {
}

function tryUnsubscribe() {
if (unsubscribe) {
subscriptionsAmount--
if (unsubscribe && subscriptionsAmount === 0) {
unsubscribe()
unsubscribe = undefined
listeners.clear()
listeners = nullListeners
}
}

function trySubscribeSelf() {
if (!selfSubscribed) {
selfSubscribed = true
trySubscribe()
}
}

function tryUnsubscribeSelf() {
if (selfSubscribed) {
selfSubscribed = false
tryUnsubscribe()
}
}

const subscription: Subscription = {
addNestedSub,
notifyNestedSubs,
handleChangeWrapper,
isSubscribed,
trySubscribe,
tryUnsubscribe,
trySubscribe: trySubscribeSelf,
tryUnsubscribe: tryUnsubscribeSelf,
getListeners: () => listeners,
}

Expand Down
126 changes: 126 additions & 0 deletions test/hooks/useSelector.spec.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@ import React, {
useLayoutEffect,
useState,
useContext,
Suspense,
useEffect,
} from 'react'
import { createStore } from 'redux'
import * as rtl from '@testing-library/react'
Expand Down Expand Up @@ -723,6 +725,130 @@ describe('React', () => {
const expectedMaxUnmountTime = IS_REACT_18 ? 500 : 7000
expect(elapsedTime).toBeLessThan(expectedMaxUnmountTime)
})

it('keeps working when used inside a Suspense', async () => {
let result: number | undefined
let expectedResult: number | undefined
let lazyComponentAdded = false
let lazyComponentLoaded = false

// A lazy loaded component in the Suspense
// This component does nothing really. It is lazy loaded to trigger the issue
// Lazy loading this component will break other useSelectors in the same Suspense
// See issue https://github.com/reduxjs/react-redux/issues/1977
const OtherComp = () => {
useLayoutEffect(() => {
lazyComponentLoaded = true
}, [])

return <div></div>
}
let otherCompFinishLoading: () => void = () => {}
const OtherComponentLazy = React.lazy(
() =>
new Promise<{ default: React.ComponentType<any> }>((resolve) => {
otherCompFinishLoading = () =>
resolve({
default: OtherComp,
})
})
)
let addOtherComponent: () => void = () => {}
const Dispatcher = React.memo(() => {
const [load, setLoad] = useState(false)

useEffect(() => {
addOtherComponent = () => setLoad(true)
}, [])
useEffect(() => {
lazyComponentAdded = true
})
return load ? <OtherComponentLazy /> : null
})
// End of lazy loading component

// The test component inside the suspense (uses the useSelector which breaks)
const CompInsideSuspense = () => {
const count = useNormalSelector((state) => state.count)

result = count
return (
<div>
{count}
<Dispatcher />
</div>
)
}
// The test component outside the suspense (uses the useSelector which keeps working - for comparison)
const CompOutsideSuspense = () => {
const count = useNormalSelector((state) => state.count)

expectedResult = count
return <div>{count}</div>
}

// Now, steps to reproduce
// step 1. make sure the component with the useSelector inside the Suspsense is rendered
// -> it will register the subscription
// step 2. make sure the suspense is switched back to "Loading..." state by adding a component
// -> this will remove our useSelector component from the page temporary!
// step 3. Finish loading the other component, so the suspense is no longer loading
// -> this will re-add our <Provider> and useSelector component
// step 4. Check that the useSelectors in our re-added components still work

// step 1: render will render our component with the useSelector
rtl.render(
<>
<Suspense fallback={<div>Loading... </div>}>
<ProviderMock store={normalStore}>
<CompInsideSuspense />
</ProviderMock>
</Suspense>
<ProviderMock store={normalStore}>
<CompOutsideSuspense />
</ProviderMock>
</>
)

// step 2: make sure the suspense is switched back to "Loading..." state by adding a component
rtl.act(() => {
addOtherComponent()
})
await rtl.waitFor(() => {
if (!lazyComponentAdded) {
throw new Error('Suspense is not back in loading yet')
}
})
expect(lazyComponentAdded).toEqual(true)

// step 3. Finish loading the other component, so the suspense is no longer loading
// This will re-add our components under the suspense, but will NOT rerender them!
rtl.act(() => {
otherCompFinishLoading()
})
await rtl.waitFor(() => {
if (!lazyComponentLoaded) {
throw new Error('Suspense is not back to loaded yet')
}
})
expect(lazyComponentLoaded).toEqual(true)

// step 4. Check that the useSelectors in our re-added components still work
// Do an update to the redux store
rtl.act(() => {
normalStore.dispatch({ type: '' })
})

// Check the component *outside* the Suspense to check whether React rerendered
await rtl.waitFor(() => {
if (expectedResult !== 1) {
throw new Error('useSelector did not return 1 yet')
}
})

// Expect the useSelector *inside* the Suspense to also update (this was broken)
expect(result).toEqual(expectedResult)
})
})

describe('error handling for invalid arguments', () => {
Expand Down

0 comments on commit 6c0ad9e

Please sign in to comment.