Skip to content

Commit 1f4d970

Browse files
authored
chore: modify jest config, ts-jest => 26.xx.xx, tsconfig'include add … (reduxjs#1770)
1 parent cc63d0c commit 1f4d970

12 files changed

+88
-99
lines changed

jest.config.js

+12-7
Original file line numberDiff line numberDiff line change
@@ -1,14 +1,9 @@
1-
const { default: tsJestPreset } = require('ts-jest')
2-
31
const defaults = {
4-
...tsJestPreset,
52
coverageDirectory: './coverage/',
63
collectCoverage: true,
74
testURL: 'http://localhost',
85
}
9-
10-
const testFolderPath = (folderName) =>
11-
`<rootDir>/test/${folderName}/**/*.{js,ts,tsx}`
6+
const testFolderPath = (folderName) => `<rootDir>/test/${folderName}/**/*.js`
127

138
const NORMAL_TEST_FOLDERS = ['components', 'hooks', 'integration', 'utils']
149

@@ -18,6 +13,16 @@ const standardConfig = {
1813
testMatch: NORMAL_TEST_FOLDERS.map(testFolderPath),
1914
}
2015

16+
const tsTestFolderPath = (folderName) =>
17+
`<rootDir>/test/${folderName}/**/*.{ts,tsx}`
18+
19+
const tsStandardConfig = {
20+
...defaults,
21+
displayName: 'ReactDOM',
22+
preset: 'ts-jest',
23+
testMatch: NORMAL_TEST_FOLDERS.map(tsTestFolderPath),
24+
}
25+
2126
const rnConfig = {
2227
...defaults,
2328
displayName: 'React Native',
@@ -29,5 +34,5 @@ const rnConfig = {
2934
}
3035

3136
module.exports = {
32-
projects: [standardConfig, rnConfig],
37+
projects: [tsStandardConfig, standardConfig, rnConfig],
3338
}

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -109,7 +109,7 @@
109109
"rimraf": "^3.0.2",
110110
"rollup": "^2.32.1",
111111
"rollup-plugin-terser": "^7.0.2",
112-
"ts-jest": "^27.0.3",
112+
"ts-jest": "26.5.6",
113113
"typescript": "^4.3.4"
114114
},
115115
"browserify": {

src/components/Provider.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,7 @@ export interface ProviderProps<A extends Action = AnyAction> {
1515
* If this is used, generate own connect HOC by using connectAdvanced, supplying the same context provided to the
1616
* Provider. Initial value doesn't matter, as it is overwritten with the internal state of Provider.
1717
*/
18-
context?: Context<ReactReduxContextValue>
18+
context?: Context<ReactReduxContextValue | null>
1919
children: ReactNode
2020
}
2121

src/connect/wrapMapToProps.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@ export function wrapMapToPropsConstant(
4747
// A length of zero is assumed to mean mapToProps is getting args via arguments or ...args and
4848
// therefore not reporting its length accurately..
4949
export function getDependsOnOwnProps(mapToProps: MapToProps) {
50-
return mapToProps?.dependsOnOwnProps
50+
return mapToProps.dependsOnOwnProps
5151
? Boolean(mapToProps.dependsOnOwnProps)
5252
: mapToProps.length !== 1
5353
}

src/types.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -272,6 +272,6 @@ export type ResolveArrayThunks<TDispatchProps extends ReadonlyArray<any>> =
272272
export interface TypedUseSelectorHook<TState> {
273273
<TSelected>(
274274
selector: (state: TState) => TSelected,
275-
equalityFn?: (left: TSelected, right: TSelected) => boolean
275+
equalityFn?: EqualityFn<TSelected>
276276
): TSelected
277277
}

test/components/hooks.spec.tsx

+19-11
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ describe('React', () => {
4343
const mapStateSpy1 = jest.fn()
4444
const renderSpy1 = jest.fn()
4545

46-
let component1StateList
46+
let component1StateList: number[]
4747

4848
const component1Decorator = connect<
4949
Omit<RootStateType, 'byId'>,
@@ -77,23 +77,31 @@ describe('React', () => {
7777
const mapStateSpy2 = jest.fn()
7878
const renderSpy2 = jest.fn()
7979

80-
interface Component2PropsType {
81-
mappedProp: Array<string>
80+
interface Component2Tstate {
81+
mappedProp: string[]
8282
}
8383
const component2Decorator = connect<
84-
Component2PropsType,
84+
Component2Tstate,
8585
unknown,
8686
Omit<RootStateType, 'byId'>,
8787
RootStateType
88-
>((state, ownProps) => {
89-
mapStateSpy2()
90-
91-
return {
92-
mappedProp: ownProps.list.map((id) => state.byId[id]),
88+
>(
89+
(
90+
state: RootStateType,
91+
ownProps: Omit<RootStateType, 'byId'>
92+
): Component2Tstate => {
93+
mapStateSpy2()
94+
95+
return {
96+
mappedProp: ownProps.list.map((id) => state.byId[id]),
97+
}
9398
}
94-
})
99+
)
100+
interface Component2PropsType {
101+
list: number[]
102+
}
95103

96-
const component2 = (props) => {
104+
const component2 = (props: Component2PropsType) => {
97105
renderSpy2()
98106

99107
expect(props.list).toBe(component1StateList)

test/hooks/useDispatch.spec.tsx

+5-3
Original file line numberDiff line numberDiff line change
@@ -7,9 +7,10 @@ import {
77
createDispatchHook,
88
} from '../../src/index'
99
import type { ProviderProps } from '../../src/'
10+
import type { ReactReduxContextValue } from '../../src/components/Context'
1011

11-
const store = createStore((c: number): number => c + 1)
12-
const store2 = createStore((c: number): number => c + 2)
12+
const store = createStore((c: number = 1): number => c + 1)
13+
const store2 = createStore((c: number = 1): number => c + 2)
1314

1415
describe('React', () => {
1516
describe('hooks', () => {
@@ -27,7 +28,8 @@ describe('React', () => {
2728
})
2829
describe('createDispatchHook', () => {
2930
it("returns the correct store's dispatch function", () => {
30-
const nestedContext = React.createContext(null)
31+
const nestedContext =
32+
React.createContext<ReactReduxContextValue | null>(null)
3133
const useCustomDispatch = createDispatchHook(nestedContext)
3234
const { result } = renderHook(() => useDispatch(), {
3335
// eslint-disable-next-line react/prop-types

test/hooks/useSelector.spec.tsx

+28-18
Original file line numberDiff line numberDiff line change
@@ -11,11 +11,12 @@ import {
1111
connect,
1212
createSelectorHook,
1313
} from '../../src/index'
14-
import type { FunctionComponent } from 'react'
1514
import { useReduxContext } from '../../src/hooks/useReduxContext'
15+
import type { FunctionComponent, DispatchWithoutAction, ReactNode } from 'react'
1616
import type { Store, AnyAction } from 'redux'
1717
import type { ProviderProps, TypedUseSelectorHook } from '../../src/'
1818
import type { Subscription } from '../../src/utils/Subscription'
19+
import type { ReactReduxContextValue } from '../../src/components/Context'
1920

2021
describe('React', () => {
2122
describe('hooks', () => {
@@ -24,7 +25,7 @@ describe('React', () => {
2425
count: number
2526
}
2627
let normalStore: Store<NormalStateType, AnyAction>
27-
let renderedItems = []
28+
let renderedItems: any[] = []
2829
type RootState = ReturnType<typeof normalStore.getState>
2930
let useNormalSelector: TypedUseSelectorHook<RootState> = useSelector
3031

@@ -56,7 +57,8 @@ describe('React', () => {
5657
})
5758

5859
it('selects the state and renders the component when the store updates', () => {
59-
const selector: jest.Mock<number, [s: NormalStateType]> = jest.fn(
60+
type MockParams = [NormalStateType]
61+
const selector: jest.Mock<number, MockParams> = jest.fn(
6062
(s) => s.count
6163
)
6264

@@ -80,7 +82,7 @@ describe('React', () => {
8082

8183
describe('lifecycle interactions', () => {
8284
it('always uses the latest state', () => {
83-
const store = createStore((c: number): number => c + 1, -1)
85+
const store = createStore((c: number = 1): number => c + 1, -1)
8486

8587
const Comp = () => {
8688
const selector = useCallback((c: number): number => c + 1, [])
@@ -106,7 +108,7 @@ describe('React', () => {
106108
let rootSubscription: Subscription
107109

108110
const Parent = () => {
109-
const { subscription } = useReduxContext()
111+
const { subscription } = useReduxContext() as ReactReduxContextValue
110112
rootSubscription = subscription
111113
const count = useNormalSelector((s) => s.count)
112114
return count === 1 ? <Child /> : null
@@ -122,19 +124,19 @@ describe('React', () => {
122124
<Parent />
123125
</ProviderMock>
124126
)
125-
127+
// @ts-ignore ts(2454)
126128
expect(rootSubscription.getListeners().get().length).toBe(1)
127129

128130
normalStore.dispatch({ type: '' })
129-
131+
// @ts-ignore ts(2454)
130132
expect(rootSubscription.getListeners().get().length).toBe(2)
131133
})
132134

133135
it('unsubscribes when the component is unmounted', () => {
134136
let rootSubscription: Subscription
135137

136138
const Parent = () => {
137-
const { subscription } = useReduxContext()
139+
const { subscription } = useReduxContext() as ReactReduxContextValue
138140
rootSubscription = subscription
139141
const count = useNormalSelector((s) => s.count)
140142
return count === 0 ? <Child /> : null
@@ -150,11 +152,11 @@ describe('React', () => {
150152
<Parent />
151153
</ProviderMock>
152154
)
153-
155+
// @ts-ignore ts(2454)
154156
expect(rootSubscription.getListeners().get().length).toBe(2)
155157

156158
normalStore.dispatch({ type: '' })
157-
159+
// @ts-ignore ts(2454)
158160
expect(rootSubscription.getListeners().get().length).toBe(1)
159161
})
160162

@@ -183,7 +185,7 @@ describe('React', () => {
183185
})
184186

185187
it('works properly with memoized selector with dispatch in Child useLayoutEffect', () => {
186-
const store = createStore((c: number): number => c + 1, -1)
188+
const store = createStore((c: number = 1): number => c + 1, -1)
187189

188190
const Comp = () => {
189191
const selector = useCallback((c: number): number => c, [])
@@ -282,7 +284,7 @@ describe('React', () => {
282284

283285
it('uses the latest selector', () => {
284286
let selectorId = 0
285-
let forceRender
287+
let forceRender: DispatchWithoutAction
286288

287289
const Comp = () => {
288290
const [, f] = useReducer((c) => c + 1, 0)
@@ -301,15 +303,19 @@ describe('React', () => {
301303

302304
expect(renderedItems).toEqual([0])
303305

304-
rtl.act(forceRender)
306+
rtl.act(() => {
307+
forceRender()
308+
})
305309
expect(renderedItems).toEqual([0, 1])
306310

307311
rtl.act(() => {
308312
normalStore.dispatch({ type: '' })
309313
})
310314
expect(renderedItems).toEqual([0, 1])
311315

312-
rtl.act(forceRender)
316+
rtl.act(() => {
317+
forceRender()
318+
})
313319
expect(renderedItems).toEqual([0, 1, 2])
314320
})
315321

@@ -503,8 +509,8 @@ describe('React', () => {
503509
})
504510

505511
describe('createSelectorHook', () => {
506-
let defaultStore
507-
let customStore
512+
let defaultStore: Store
513+
let customStore: Store
508514
type StateType = {
509515
count: number
510516
}
@@ -519,7 +525,8 @@ describe('React', () => {
519525
})
520526

521527
it('subscribes to the correct store', () => {
522-
const nestedContext = React.createContext(null)
528+
const nestedContext =
529+
React.createContext<ReactReduxContextValue | null>(null)
523530
const useCustomSelector = createSelectorHook(nestedContext)
524531
let defaultCount = null
525532
let customCount = null
@@ -531,7 +538,10 @@ describe('React', () => {
531538
defaultCount = count
532539
return <>{children}</>
533540
}
534-
const DisplayCustomCount = ({ children = null }) => {
541+
interface DisplayCustomCountType {
542+
children: ReactNode
543+
}
544+
const DisplayCustomCount = ({ children }: DisplayCustomCountType) => {
535545
const count = useCustomSelector<StateType>(getCount)
536546
customCount = count
537547
return <>{children}</>

test/typetests/react-redux-types.typetest.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
/* eslint-disable @typescript-eslint/no-unused-vars, no-inner-declarations */
22
import { Component, ReactElement } from 'react'
3-
import * as React from 'react'
4-
import * as ReactDOM from 'react-dom'
3+
import React from 'react'
4+
import ReactDOM from 'react-dom'
55
import { Store, Dispatch, bindActionCreators, AnyAction } from 'redux'
66
import { connect, Provider, ConnectedProps } from '../../src/index'
77
import { expectType } from '../typeTestHelpers'

tsconfig.json

+3-2
Original file line numberDiff line numberDiff line change
@@ -11,8 +11,9 @@
1111
"declaration": true,
1212
"emitDeclarationOnly": true,
1313
"outDir": "./es",
14-
"forceConsistentCasingInFileNames": true
14+
"forceConsistentCasingInFileNames": true,
15+
"experimentalDecorators":true
1516
},
16-
"include": ["src/**/*", "types"],
17+
"include": ["src/**/*", "test/**/*", "types"],
1718
"exclude": ["node_modules", "dist"]
1819
}

types/index.d.ts

-12
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,4 @@
11
/* eslint-disable no-unused-vars */
2-
declare module 'react-dom' {
3-
export function unstable_batchedUpdates<A, B>(
4-
callback: (a: A, b: B) => any,
5-
a: A,
6-
b: B
7-
): void
8-
export function unstable_batchedUpdates<A>(
9-
callback: (a: A) => any,
10-
a: A
11-
): void
12-
export function unstable_batchedUpdates(callback: () => any): void
13-
}
142

153
declare module 'react-native' {
164
export function unstable_batchedUpdates<A, B>(

0 commit comments

Comments
 (0)