@reduxjs/toolkit
Version:
The official, opinionated, batteries-included toolset for efficient Redux development
79 lines (63 loc) • 2.08 kB
text/typescript
import type { Reducer } from 'redux'
import type { ActionReducerMapBuilder } from '@reduxjs/toolkit'
import { createReducer, createAction } from '@reduxjs/toolkit'
import { expectType } from './helpers'
/*
* Test: createReducer() infers type of returned reducer.
*/
{
type CounterAction =
| { type: 'increment'; payload: number }
| { type: 'decrement'; payload: number }
const incrementHandler = (state: number, action: CounterAction) => state + 1
const decrementHandler = (state: number, action: CounterAction) => state - 1
const reducer = createReducer(0 as number, {
increment: incrementHandler,
decrement: decrementHandler,
})
const numberReducer: Reducer<number> = reducer
// @ts-expect-error
const stringReducer: Reducer<string> = reducer
}
/**
* Test: createReducer() state type can be specified expliclity.
*/
{
type CounterAction =
| { type: 'increment'; payload: number }
| { type: 'decrement'; payload: number }
const incrementHandler = (state: number, action: CounterAction) =>
state + action.payload
const decrementHandler = (state: number, action: CounterAction) =>
state - action.payload
createReducer<number>(0, {
increment: incrementHandler,
decrement: decrementHandler,
})
// @ts-expect-error
createReducer<string>(0, {
increment: incrementHandler,
decrement: decrementHandler,
})
}
/*
* Test: createReducer() ensures state type is mutable within a case reducer.
*/
{
const initialState: { readonly counter: number } = { counter: 0 }
createReducer(initialState, {
increment: (state) => {
state.counter += 1
},
})
}
/** Test: alternative builder callback for actionMap */
{
const increment = createAction<number, 'increment'>('increment')
const reducer = createReducer(0, (builder) =>
expectType<ActionReducerMapBuilder<number>>(builder)
)
expectType<number>(reducer(0, increment(5)))
// @ts-expect-error
expectType<string>(reducer(0, increment(5)))
}