Skip to content

Commit 7f92f16

Browse files
committed
Experimental useSelector to avoid stale props
1 parent cff554d commit 7f92f16

File tree

2 files changed

+94
-61
lines changed

2 files changed

+94
-61
lines changed

src/hooks/useSelector.js

+17-61
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
1-
import { useReducer, useRef, useMemo, useContext } from 'react'
1+
import { useReducer, useContext } from 'react'
22
import { useReduxContext as useDefaultReduxContext } from './useReduxContext'
3-
import Subscription from '../utils/Subscription'
43
import { useIsomorphicLayoutEffect } from '../utils/useIsomorphicLayoutEffect'
54
import { ReactReduxContext } from '../components/Context'
65

@@ -10,72 +9,29 @@ function useSelectorWithStoreAndSubscription(
109
selector,
1110
equalityFn,
1211
store,
13-
contextSub
12+
subscription
1413
) {
15-
const [, forceRender] = useReducer(s => s + 1, 0)
16-
17-
const subscription = useMemo(() => new Subscription(store, contextSub), [
18-
store,
19-
contextSub
20-
])
21-
22-
const latestSubscriptionCallbackError = useRef()
23-
const latestSelector = useRef()
24-
const latestSelectedState = useRef()
25-
26-
let selectedState
27-
28-
try {
29-
if (
30-
selector !== latestSelector.current ||
31-
latestSubscriptionCallbackError.current
32-
) {
33-
selectedState = selector(store.getState())
34-
} else {
35-
selectedState = latestSelectedState.current
36-
}
37-
} catch (err) {
38-
if (latestSubscriptionCallbackError.current) {
39-
err.message += `\nThe error may be correlated with this previous error:\n${latestSubscriptionCallbackError.current.stack}\n\n`
40-
}
41-
42-
throw err
43-
}
44-
45-
useIsomorphicLayoutEffect(() => {
46-
latestSelector.current = selector
47-
latestSelectedState.current = selectedState
48-
latestSubscriptionCallbackError.current = undefined
49-
})
50-
51-
useIsomorphicLayoutEffect(() => {
52-
function checkForUpdates() {
53-
try {
54-
const newSelectedState = latestSelector.current(store.getState())
55-
56-
if (equalityFn(newSelectedState, latestSelectedState.current)) {
57-
return
58-
}
59-
60-
latestSelectedState.current = newSelectedState
61-
} catch (err) {
62-
// we ignore all errors here, since when the component
63-
// is re-rendered, the selectors are called again, and
64-
// will throw again, if neither props nor store state
65-
// changed
66-
latestSubscriptionCallbackError.current = err
14+
const [selectedState, dispatch] = useReducer(
15+
prevSelectedState => {
16+
const nextState = store.getState()
17+
const nextSelectedState = selector(nextState)
18+
if (equalityFn(prevSelectedState, nextSelectedState)) {
19+
return prevSelectedState
6720
}
21+
return nextSelectedState
22+
},
23+
null,
24+
() => selector(store.getState())
25+
)
6826

69-
forceRender({})
70-
}
71-
72-
subscription.onStateChange = checkForUpdates
27+
useIsomorphicLayoutEffect(() => {
28+
subscription.onStateChange = dispatch
7329
subscription.trySubscribe()
7430

75-
checkForUpdates()
31+
dispatch()
7632

7733
return () => subscription.tryUnsubscribe()
78-
}, [store, subscription])
34+
}, [subscription])
7935

8036
return selectedState
8137
}

test/integration/stale-props.spec.js

+77
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
/*eslint-disable react/prop-types*/
2+
3+
import React from 'react'
4+
import { createStore } from 'redux'
5+
import { Provider, useSelector } from '../../src/index.js'
6+
import * as rtl from '@testing-library/react'
7+
8+
describe('React', () => {
9+
describe('stale props with useSelector', () => {
10+
it('ignores transient errors in selector (e.g. due to stale props)', () => {
11+
const Parent = () => {
12+
const count = useSelector(count => count)
13+
return <Child parentCount={count} />
14+
}
15+
16+
const Child = ({ parentCount }) => {
17+
const result = useSelector(count => {
18+
if (count !== parentCount) {
19+
throw new Error()
20+
}
21+
22+
return count + parentCount
23+
})
24+
25+
return <div>{result}</div>
26+
}
27+
28+
const store = createStore((count = -1) => count + 1)
29+
30+
const App = () => (
31+
<Provider store={store}>
32+
<Parent />
33+
</Provider>
34+
)
35+
36+
rtl.render(<App />)
37+
38+
rtl.act(() => {
39+
expect(() => store.dispatch({ type: '' })).not.toThrowError()
40+
})
41+
})
42+
43+
it('ensures consistency of state and props in selector', () => {
44+
let selectorSawInconsistencies = false
45+
46+
const Parent = () => {
47+
const count = useSelector(count => count)
48+
return <Child parentCount={count} />
49+
}
50+
51+
const Child = ({ parentCount }) => {
52+
const result = useSelector(count => {
53+
selectorSawInconsistencies =
54+
selectorSawInconsistencies || count !== parentCount
55+
return count + parentCount
56+
})
57+
58+
return <div>{result}</div>
59+
}
60+
61+
const store = createStore((count = -1) => count + 1)
62+
63+
const App = () => (
64+
<Provider store={store}>
65+
<Parent />
66+
</Provider>
67+
)
68+
69+
rtl.render(<App />)
70+
71+
rtl.act(() => {
72+
store.dispatch({ type: '' })
73+
})
74+
expect(selectorSawInconsistencies).toBe(false)
75+
})
76+
})
77+
})

0 commit comments

Comments
 (0)