Skip to content

Commit eb7f08b

Browse files
authoredSep 5, 2022
Merge: Fix index signature type overwrite (#460)
1 parent ec07bff commit eb7f08b

File tree

2 files changed

+42
-7
lines changed

2 files changed

+42
-7
lines changed
 

‎source/merge.d.ts

+12-7
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,15 @@ import type {OmitIndexSignature} from './omit-index-signature';
22
import type {PickIndexSignature} from './pick-index-signature';
33
import type {EnforceOptional} from './enforce-optional';
44

5+
// Merges two objects without worrying about index signatures or optional keys.
6+
type SimpleMerge<Destination, Source> = {
7+
[Key in keyof Destination | keyof Source]: Key extends keyof Source
8+
? Source[Key]
9+
: Key extends keyof Destination
10+
? Destination[Key]
11+
: never;
12+
};
13+
514
/**
615
Merge two types into a new type. Keys of the second type overrides keys of the first type.
716
@@ -36,10 +45,6 @@ export type FooBar = Merge<Foo, Bar>;
3645
3746
@category Object
3847
*/
39-
export type Merge<Destination, Source> = EnforceOptional<{
40-
[Key in keyof OmitIndexSignature<Destination> | keyof OmitIndexSignature<Source>]: Key extends keyof Source
41-
? Source[Key]
42-
: Key extends keyof Destination
43-
? Destination[Key]
44-
: never;
45-
} & PickIndexSignature<Destination> & PickIndexSignature<Source>>;
48+
export type Merge<Destination, Source> = EnforceOptional<
49+
SimpleMerge<PickIndexSignature<Destination>, PickIndexSignature<Source>>
50+
& SimpleMerge<OmitIndexSignature<Destination>, OmitIndexSignature<Source>>>;

‎test-d/merge.ts

+30
Original file line numberDiff line numberDiff line change
@@ -104,3 +104,33 @@ expectType<{
104104
f?: number;
105105
g: undefined;
106106
}>(fooBarWithOptionalKeys);
107+
108+
// Checks that an indexed key type can be overwritten.
109+
type FooWithIndexSignature = {
110+
[x: string]: unknown;
111+
[x: number]: boolean;
112+
[x: symbol]: number;
113+
foo: boolean;
114+
fooBar: boolean;
115+
};
116+
117+
type BarWithIndexSignatureOverwrite = {
118+
[x: string]: number | string | boolean;
119+
[x: number]: number | string;
120+
[x: symbol]: symbol;
121+
bar: string;
122+
fooBar: string;
123+
};
124+
125+
type FooBarWithIndexSignature = Merge<FooWithIndexSignature, BarWithIndexSignatureOverwrite>;
126+
127+
declare const fooBarWithIndexSignature: FooBarWithIndexSignature;
128+
129+
expectType<{
130+
[x: string]: string | number | boolean;
131+
[x: number]: string | number;
132+
[x: symbol]: symbol;
133+
foo: boolean;
134+
bar: string;
135+
fooBar: string;
136+
}>(fooBarWithIndexSignature);

0 commit comments

Comments
 (0)
Please sign in to comment.