When you don’t want to repeat yourself, sometimes a type needs to be based on another type.
Mapped types build on the syntax for index signatures, which are used to declare the types of properties which has not been declared ahead of time:
typeOnlyBoolsAndHorses = {[key : string]: boolean |Horse ;};Try
constconforms :OnlyBoolsAndHorses = {del : true,rodney : false,};
A mapped type is a generic type which uses a union created via a keyof
to iterate through the keys of one type to create another:
Try
typeOptionsFlags <Type > = {[Property in keyofType ]: boolean;};
In this example, OptionFlags
will take all the properties from the type Type
and change their values to be a boolean.
typeFeatureFlags = {darkMode : () => void;newUserProfile : () => void;};
type// ^ = type FeatureOptions = {FeatureOptions =OptionsFlags <FeatureFlags >;// darkMode: boolean;
// newUserProfile: boolean;
// }
Try
Mapping Modifiers
There are two additional modifiers which can be applied during mapping: readonly
and ?
which affect mutability and optionality respectively.
You can remove or add these modifiers by prefixing with -
or +
. If you don’t add a prefix, then +
is assumed.
// Removes 'readonly' attributes from a type's propertiestypeCreateMutable <Type > = {-readonly [Property in keyofType ]:Type [Property ];};
typeLockedAccount = {readonlyid : string;readonlyname : string;};
type// ^ = type UnlockedAccount = {UnlockedAccount =CreateMutable <LockedAccount >;// id: string;
// name: string;
// }
Try
// Removes 'optional' attributes from a type's propertiestypeConcrete <Type > = {[Property in keyofType ]-?:Type [Property ];};
typeMaybeUser = {id : string;name ?: string;age ?: number;};
type// ^ = type User = {User =Concrete <MaybeUser >;// id: string;
// name: string;
// age: number;
// }
Try
Key Remapping via as
In TypeScript 4.1 and onwards, you can re-map keys in mapped types with an as
clause in a mapped type:
type MappedTypeWithNewProperties<Type> = {[Properties in keyof Type as NewKeyType]: Type[Properties]}
You can leverage features like template literal types to create new property names from prior ones:
typeGetters <Type > = {[Property in keyofType as `get${Capitalize <string &Property >}`]: () =>Type [Property ]};
interfacePerson {name : string;age : number;location : string;}
type// ^ = type LazyPerson = {LazyPerson =Getters <Person >;// getName: () => string;
// getAge: () => number;
// getLocation: () => string;
// }
Try
You can filter out keys by producing never
via a conditional type:
// Remove the 'kind' propertytypeRemoveKindField <T > = {[K in keyofT asExclude <K , "kind">]:T [K ]};
interfaceCircle {kind : "circle";radius : number;}
type// ^ = type KindlessCircle = {KindlessCircle =RemoveKindField <Circle >;// radius: number;
// }
Try
Further Exploration
Mapped types work well with other features in this type manipulation section, for example here is a mapped type using a conditional type which returns either a true
or false
depending on whether an object has the property pii
set to the literal true
:
type ExtractPII<Type> = {[Property in keyof Type]: Type[Property] extends { pii: true } ? true : false;};type DBFields = {id: { format: "incrementing" };name: { type: string; pii: true };};type ObjectsNeedingGDPRDeletion = ExtractPII<DBFields>;// ^?