Typescript 2.8+: как захватить и сопоставить создателей действий с обработчиками действий? - PullRequest
0 голосов
/ 01 июня 2018

У меня есть следующие создатели действий:

/** simply captures the string literal type */
function actionType<T extends string, U>(type: T, u: U) {
  return Object.assign({}, { type }, u);
}

// ------------------------------------
// Action Creators
// ------------------------------------
const ACTION_CREATORS = {
  toggleFilter: (params: { filterValueKey: string; filterKey: string; filterValue: string }) => {
    const { filterKey, filterValue, filterValueKey } = params;
    return actionType('FILTERS_TOGGLE', {
      payload: {
        filterValueKey,
        filterKey,
        filterValue
      }
    });
  },
  deleteFilter: (params: { filterValueKey: string }) => {
    const { filterValueKey } = params;
    return actionType('FILTERS_DELETE', {
      payload: {
        filterValueKey
      }
    });
  },
  searchFilterValues: (params: { activeFilterKey: string; query: string }) => {
    const { query, activeFilterKey } = params;
    return actionType('FILTERS_QUERY_VALUES', {
      payload: {
        query,
        activeFilterKey
      }
    });
  }
};

, и я хотел бы написать

type ActionHandlers<T extends {
  [actionCreator: string]: (...args: any[]) => { type: string }
}> = { /* ... */ };

, который добавит правильные наборы для "обработчиков действий"объект, подобный приведенному ниже:

const ACTION_HANDLERS: ActionHandlers<typeof ACTION_CREATORS > = {
  FILTERS_TOGGLE: (state, action) => {
    action.payload.filterValueKey // this key is typed
    const newState = { ...state };
    return newState;
  },
  FILTERS_DELETE: () => {} /* ... */,
  FILTERS_QUERY_VALUES: (state, action) => {
    action.payload.query // this is also typed and typed differently
    const newState = { ...state };
    return newState;
  }
}

Пока я могу захватывать имена типов действий и получать возвращаемые типы создателей действий, но я застрял в этой точке:

type Action<T extends { type: string }> = T;

type ActionCreator<T extends string> = (...params: any[]) => { type: T };
type MapToNames<T extends { [key: string]: ActionCreator<any> }> = {
  [P in keyof T]: T[P] extends ActionCreator<infer U> ? U : never
};
type ActionNames<T extends { [key: string]: ActionCreator<any> }> = MapToNames<
  T
>[keyof MapToNames<T>];

type RemoveFunctions<T extends { [key: string]: (...args: any[]) => { type: string } }> = {
  [P in keyof T]: T[P] extends (...params: any[]) => Action<infer U> ? U : never
};

type Names = ActionNames<typeof ACTION_CREATORS>;
// type Names = "FILTERS_TOGGLE" | "FILTERS_DELETE" | "FILTERS_QUERY_VALUES"
type NoFunctions = RemoveFunctions<typeof ACTION_CREATORS>;

Любые идеи будут с благодарностью!

1 Ответ

0 голосов
/ 01 июня 2018

Это может быть одним из возможных решений:

type GetActionCreatorTypes<T extends { [key: string]: (...args: any[]) => { type: string } }> = {
    // Get the return type of the action creator
    [P in keyof T]: T[P] extends (...params: any[]) => Action<infer U> ? U : never
}[keyof T]; // Make a union of all action types 


type ActionHandlers<T extends {
    [actionCreator: string]: (...args: any[]) => { type: string }
}> = {
        // Take all properties named type (will be a union of "FILTERS_TOGGLE" | "FILTERS_DELETE" | "FILTERS_QUERY_VALUES" in our case)
        [P in GetActionCreatorTypes<T>['type']]: (
            state: any, 
            // Type the action as beeing the only type in the union with the type property named P
            action: Extract<GetActionCreatorTypes<T>, { type: P }>
        ) => any 
    };


const ACTION_HANDLERS: ActionHandlers<typeof ACTION_CREATORS> = {
    FILTERS_TOGGLE: (state, action) => {
        action.payload.filterValueKey // ok 
        const newState = { ...state };
        return newState;
    },
    FILTERS_DELETE: () => { } /* ... */,
    FILTERS_QUERY_VALUES: (state, action) => {
        action.payload.query // this is also ok
        const newState = { ...state };
        return newState;
    }
}

Детская площадка ссылка

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...