Mapped Types #
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 have not been declared ahead of time:
type OnlyBoolsAndHorses = {
[key: string]: boolean | Horse;
};
Â
const conforms: OnlyBoolsAndHorses = {
del: true,
rodney: false,
};
A mapped type is a generic type which uses a union of PropertyKey
s
(frequently created
via a keyof
) to iterate
through keys to create a type:
type OptionsFlags<Type> = {
[Property in keyof Type]: boolean;
};
In this example, OptionsFlags
will take all the properties from the
type Type
and change their values to be a boolean.
type Features = {
darkMode: () => void;
newUserProfile: () => void;
};
Â
type FeatureOptions = OptionsFlags<Features>;
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 properties
type CreateMutable<Type> = {
-readonly [Property in keyof Type]: Type[Property];
};
Â
type LockedAccount = {
readonly id: string;
readonly name: string;
};
Â
type UnlockedAccount = CreateMutable<LockedAccount>;
// Removes 'optional' attributes from a type's properties
type Concrete<Type> = {
[Property in keyof Type]-?: Type[Property];
};
Â
type MaybeUser = {
id: string;
name?: string;
age?: number;
};
Â
type User = Concrete<MaybeUser>;
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:
type Getters<Type> = {
[Property in keyof Type as `get${Capitalize<string & Property>}`]: () => Type[Property]
};
Â
interface Person {
name: string;
age: number;
location: string;
}
Â
type LazyPerson = Getters<Person>;
You can filter out keys by producing never
via a conditional type:
// Remove the 'kind' property
type RemoveKindField<Type> = {
[Property in keyof Type as Exclude<Property, "kind">]: Type[Property]
};
Â
interface Circle {
kind: "circle";
radius: number;
}
Â
type KindlessCircle = RemoveKindField<Circle>;
You can map over arbitrary unions, not just unions of
string | number | symbol
, but unions of any type:
type EventConfig<Events extends { kind: string }> = {
[E in Events as E["kind"]]: (event: E) => void;
}
Â
type SquareEvent = { kind: "square", x: number, y: number };
type CircleEvent = { kind: "circle", radius: number };
Â
type Config = EventConfig<SquareEvent | CircleEvent>
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>;
::: _attribution
© 2012-2023 Microsoft
Licensed under the Apache License, Version 2.0.
https://www.typescriptlang.org/docs/handbook/2/mapped-types.html{._attribution-link}
:::