import { useMemo, useReducer } from 'react'; // Extracts property names from initial state of reducer to allow typesafe dispatch objects export type FieldNames = { [K in keyof T]: T[K] extends string ? K : K; }[keyof T]; // Returns the Action Type for the dispatch object to be used for typing in things like context export type ActionType = | { type: 'reset' } | { type?: 'change'; field: FieldNames; value: any }; // Returns a typed dispatch and state export const useCreateReducer = ({ initialState }: { initialState: T }) => { type Action = | { type: 'reset' } | { type?: 'change'; field: FieldNames; value: any }; const reducer = (state: T, action: Action) => { if (!action.type) return { ...state, [action.field]: action.value }; if (action.type === 'reset') return initialState; throw new Error(); }; const [state, dispatch] = useReducer(reducer, initialState); return useMemo(() => ({ state, dispatch }), [state, dispatch]); };