When intersecting two object types, typescript rightfully intersects its properties too.
type Type = { foo: "yes" | "no" } & { foo: "maybe"; bar: string }// type Type = { foo: "yes" | "no" | "maybe", bar: string }
Notice that not only the properties foo
and bar
were "merged", but also the union types from foo
since properties are also intersected by typescript. Also note that the order doesn't matter, since everything will be intersected.
It's the expected behavior. But sometimes you may want to merge objects where the right member properties takes precedence. Like the behavior of { ...left, ...right }
.
Then you can use Override type generic:
type Override<Left, Right> = Omit<Left, keyof Right> & Right;
Given Left and Right example types:
type Left = { foo: string; bar: string };type Right = { foo: number };
Intersecting
type Intersected = Left & Right;// type Intersected = { foo: never; bar: string }// └╮ `foo` becomes `string & number`// │ which evaluates to `never` since it can't// │ be string and number at the same time
Overriding
type Overrided = Override<Left, Right>;// type Overrided = { foo: number; bar: string }// └╮ `foo` becomes `number` because// │ Right properties take precedence over Left,// │ but preserving Left properties as expected
# Flatten
To a better readability you might want to combo it with Flatten
type utility.
This tricky type is also known as Simplify
in some libs.
type Flatten<T> = { [Key in keyof T]: T[Key] } & {};type Override1<L extends object, R extends object> = Omit<L, keyof R> & R;type Override2<L extends object, R extends object> = Flatten<Omit<L, keyof R> & R>;
type Left = { foo: string };type Right = { foo: number };
type A = Override1<Left, Right>;// type A = Omit<Left, "foo"> & Right
type B = Override2<Left, Right>;// type B = { foo: number }// └ the type becomes "flat" i.e. easier to read with intellisense
# Assign !==
Spread
Taking advantage of the subject, I want to address a related common confusion.
In TypeScript, object assignment and object spreading leads to different types.
const left = { foo: "string" };const right = { foo: 1 };
const assign1 = Object.assign(left, right);assign1.foo// ^? (property) foo: never
const assign2 = Object.assign({}, left, right);assign2.foo// ^? (property) foo: never
const spread = { ...left, ...right };spread.foo// ^? (property) foo: number
However it seems to be a TypeScript-only problem(?). In runtime it overwrite from right to left aswell.
const assign = Object.assign(left, right);console.log(assign.foo) // logs 1// ^? (property) foo: never