Вам нужны различные параметры типа для каждого свойства в пути. Это позволяет компилятору рассуждать о конкретных полях, которые вы укажете:
type Props<TState, KStore extends keyof TState, KField extends keyof TState[KStore]> = {
state: KStore;
field: KField
data: TState[KStore][KField]
}
let p: Props<AppState, "cells", "duration"> = {
state: "cells",
field: "duration",
data: 1
}
Причина, по которой вы никогда не получаете, заключается в том, что когда компилятор пытается развернуть AppState[keyof AppState]
, он получит объединение CellsReducer | BarReducer
. Поскольку доступны только общие члены объединения keyof (CellsReducer | BarReducer)
- это never
(ключи недоступны).
Дополнительные параметры фиксируют фактическое поле, поэтому, если KStore
является строковым литералом, тип "cells"
keyof AppState["cells"]
будет ключами этого конкретного поля в состоянии приложения. KField
работает аналогично, позволяя нам правильно набирать data
.
Чтобы не указывать значения state
и field
дважды, вы можете написать вспомогательную функцию:
function propertyFactory<TState>() {
return function <KStore extends keyof TState, KField extends keyof TState[KStore]>(o: Props<TState, KStore, KField>) {
return o;
}
}
let p = propertyFactory<AppState>()({
state: "cells",
field: "duration",
data: 1
})