export interface UserActions extends Action { type: UserKind; payload: UserPayload; }
..TypeScript will raise an error on combineReducers:
data:image/s3,"s3://crabby-images/f5548/f55488c24198abaf28ef7e7195c164933eae6c1d" alt=""
To fix the compilation error, make the Redux action property be optional:
export interface UserActions extends Action { type: UserKind; payload?: UserPayload; }
Result:
data:image/s3,"s3://crabby-images/da5c1/da5c1613dac94dcc234b39271acc93973a729c69" alt=""
However, TypeScript won't be able to catch missed properties, it won't raise an error that we didn't pass the payload property on User 6:
data:image/s3,"s3://crabby-images/06291/06291586bfaf3858cf7a35db19b60404691a0a97" alt=""
To fix that, remove the optional indicator on Redux action:
export interface UserActions extends Action { type: UserKind; payload: UserPayload; }
Result, TypeScript is now able to catch missed properties:
data:image/s3,"s3://crabby-images/a1a87/a1a87df866109320b35613cf419f4bcf043616e2" alt=""
However, we are back to this error on combineReducers:
data:image/s3,"s3://crabby-images/f529d/f529d89434ca0291239e725da623b8e1fd467893" alt=""
To comply with combineReducers requirement of Reducer actions containing just a type property only, create another interface that has optional properties (e.g., payload), and use that for the Redux reducer's action parameter (UserActionsNoPayload). And then on UserActions, inherit UserActionsNoPayload from UserActions:
data:image/s3,"s3://crabby-images/9f9c2/9f9c221d6bb64b3ee5cc19c89d7a42ae78616620" alt=""
Now, no more combineReducers compilation error, and the compiler is still able to catch missed properties:
data:image/s3,"s3://crabby-images/ab12c/ab12c673272f61f10288bf83540e4e84b36f3178" alt=""
Here are the TypeScript definitions for the routeToUser:
export interface StateProps { user: Dto.Client.User; component: Component; counter: Dto.Client.Counter; } export interface ActionProps { routeToUser: UserDispatcher; doCounter: CounterDispatcher; } interface Props extends StateProps, ActionProps { } export type Dispatcher = (action: UserActions | CounterActions) => void; export type UserDispatcher = (action: UserActions) => void; export type CounterDispatcher = (action: CounterActions) => void; export default class ReduxFirstRouterApp extends React.Component<Props, {}> {
And here is how are the states and dispatchers are mapped as properties to a component via Redux's connect:
const mapStateToProps = ({user, component, counter}: StateProps) => ({user, component, counter}); const mapDispatchToProps = (dispatch: Dispatcher): ActionProps => ({ routeToUser: (action: UserActions) => dispatch(action), doCounter: (action: CounterActions) => dispatch(action) }); this.AppContainer = connect(mapStateToProps, mapDispatchToProps)(ReduxFirstRouterApp);
Happy Coding!
P.S.
At first, I tried creating UserActions with complete properties, and then made UserActionsNoPayload inherits UserActions and define payload as optional on UserActionsNoPayload; however, it's a compiler error:
data:image/s3,"s3://crabby-images/43cf7/43cf72715deb263be7eab850e2257f8f60ee8e17" alt=""