|
1 | 1 | /*eslint-disable react/prop-types*/
|
2 | 2 |
|
3 |
| -import React from 'react' |
| 3 | +import React, { useCallback, useReducer } from 'react' |
4 | 4 | import { createStore } from 'redux'
|
5 | 5 | import { renderHook, act } from 'react-hooks-testing-library'
|
6 | 6 | import * as rtl from 'react-testing-library'
|
@@ -47,6 +47,29 @@ describe('React', () => {
|
47 | 47 | })
|
48 | 48 |
|
49 | 49 | describe('lifeycle interactions', () => {
|
| 50 | + it('always uses the latest state', () => { |
| 51 | + store = createStore(c => c + 1, -1) |
| 52 | + |
| 53 | + const Comp = () => { |
| 54 | + const selector = useCallback(c => c + 1, []) |
| 55 | + const value = useSelector(selector) |
| 56 | + renderedItems.push(value) |
| 57 | + return <div /> |
| 58 | + } |
| 59 | + |
| 60 | + rtl.render( |
| 61 | + <ProviderMock store={store}> |
| 62 | + <Comp /> |
| 63 | + </ProviderMock> |
| 64 | + ) |
| 65 | + |
| 66 | + expect(renderedItems).toEqual([1]) |
| 67 | + |
| 68 | + store.dispatch({ type: '' }) |
| 69 | + |
| 70 | + expect(renderedItems).toEqual([1, 2]) |
| 71 | + }) |
| 72 | + |
50 | 73 | it('subscribes to the store synchronously', () => {
|
51 | 74 | let rootSubscription
|
52 | 75 |
|
@@ -156,6 +179,39 @@ describe('React', () => {
|
156 | 179 | })
|
157 | 180 | })
|
158 | 181 |
|
| 182 | + it('uses the latest selector', () => { |
| 183 | + let selectorId = 0 |
| 184 | + let forceRender |
| 185 | + |
| 186 | + const Comp = () => { |
| 187 | + const [, f] = useReducer(c => c + 1, 0) |
| 188 | + forceRender = f |
| 189 | + const renderedSelectorId = selectorId++ |
| 190 | + const value = useSelector(() => renderedSelectorId) |
| 191 | + renderedItems.push(value) |
| 192 | + return <div /> |
| 193 | + } |
| 194 | + |
| 195 | + rtl.render( |
| 196 | + <ProviderMock store={store}> |
| 197 | + <Comp /> |
| 198 | + </ProviderMock> |
| 199 | + ) |
| 200 | + |
| 201 | + expect(renderedItems).toEqual([0]) |
| 202 | + |
| 203 | + rtl.act(forceRender) |
| 204 | + expect(renderedItems).toEqual([0, 1]) |
| 205 | + |
| 206 | + rtl.act(() => { |
| 207 | + store.dispatch({ type: '' }) |
| 208 | + }) |
| 209 | + expect(renderedItems).toEqual([0, 1]) |
| 210 | + |
| 211 | + rtl.act(forceRender) |
| 212 | + expect(renderedItems).toEqual([0, 1, 2]) |
| 213 | + }) |
| 214 | + |
159 | 215 | describe('edge cases', () => {
|
160 | 216 | it('ignores transient errors in selector (e.g. due to stale props)', () => {
|
161 | 217 | const spy = jest.spyOn(console, 'error').mockImplementation(() => {})
|
|
0 commit comments