1
- import { useReducer , useRef , useMemo , useContext } from 'react'
1
+ import { useReducer , useContext } from 'react'
2
2
import { useReduxContext as useDefaultReduxContext } from './useReduxContext'
3
- import Subscription from '../utils/Subscription'
4
3
import { useIsomorphicLayoutEffect } from '../utils/useIsomorphicLayoutEffect'
5
4
import { ReactReduxContext } from '../components/Context'
6
5
@@ -10,72 +9,29 @@ function useSelectorWithStoreAndSubscription(
10
9
selector ,
11
10
equalityFn ,
12
11
store ,
13
- contextSub
12
+ subscription
14
13
) {
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
67
20
}
21
+ return nextSelectedState
22
+ } ,
23
+ null ,
24
+ ( ) => selector ( store . getState ( ) )
25
+ )
68
26
69
- forceRender ( { } )
70
- }
71
-
72
- subscription . onStateChange = checkForUpdates
27
+ useIsomorphicLayoutEffect ( ( ) => {
28
+ subscription . onStateChange = dispatch
73
29
subscription . trySubscribe ( )
74
30
75
- checkForUpdates ( )
31
+ dispatch ( )
76
32
77
33
return ( ) => subscription . tryUnsubscribe ( )
78
- } , [ store , subscription ] )
34
+ } , [ subscription ] )
79
35
80
36
return selectedState
81
37
}
0 commit comments