Skip to content

Commit 28e55b9

Browse files
skarab42sindresorhus
andauthoredAug 31, 2022
Add OmitIndexSignature and PickIndexSignature types (#453)
Co-authored-by: Sindre Sorhus <sindresorhus@gmail.com>
1 parent 1003d5d commit 28e55b9

6 files changed

+157
-16
lines changed
 

‎index.d.ts

+8-1
Original file line numberDiff line numberDiff line change
@@ -13,7 +13,14 @@ export type {MergeExclusive} from './source/merge-exclusive';
1313
export type {RequireAtLeastOne} from './source/require-at-least-one';
1414
export type {RequireExactlyOne} from './source/require-exactly-one';
1515
export type {RequireAllOrNone} from './source/require-all-or-none';
16-
export type {RemoveIndexSignature} from './source/remove-index-signature';
16+
export type {
17+
OmitIndexSignature,
18+
/**
19+
@deprecated Renamed to {@link OmitIndexSignature}.
20+
*/
21+
OmitIndexSignature as RemoveIndexSignature,
22+
} from './source/omit-index-signature';
23+
export type {PickIndexSignature} from './source/pick-index-signature';
1724
export type {PartialDeep, PartialDeepOptions} from './source/partial-deep';
1825
export type {PartialOnUndefinedDeep, PartialOnUndefinedDeepOptions} from './source/partial-on-undefined-deep';
1926
export type {ReadonlyDeep} from './source/readonly-deep';

‎readme.md

+2-1
Original file line numberDiff line numberDiff line change
@@ -134,7 +134,8 @@ Click the type names for complete docs.
134134
- [`RequireAtLeastOne`](source/require-at-least-one.d.ts) - Create a type that requires at least one of the given keys.
135135
- [`RequireExactlyOne`](source/require-exactly-one.d.ts) - Create a type that requires exactly a single key of the given keys and disallows more.
136136
- [`RequireAllOrNone`](source/require-all-or-none.d.ts) - Create a type that requires all of the given keys or none of the given keys.
137-
- [`RemoveIndexSignature`](source/remove-index-signature.d.ts) - Create a type that only has explicitly defined properties, absent of any index signatures.
137+
- [`OmitIndexSignature`](source/omit-index-signature.d.ts) - Omit any index signatures from the given object type, leaving only explicitly defined properties.
138+
- [`PickIndexSignature`](source/pick-index-signature.d.ts) - Pick only index signatures from the given object type, leaving out all explicitly defined properties.
138139
- [`PartialDeep`](source/partial-deep.d.ts) - Create a deeply optional version of another type. Use [`Partial<T>`](https://www.typescriptlang.org/docs/handbook/utility-types.html#partialtype) if you only need one level deep.
139140
- [`PartialOnUndefinedDeep`](source/partial-on-undefined-deep.d.ts) - Create a deep version of another type where all keys accepting `undefined` type are set to optional.
140141
- [`ReadonlyDeep`](source/readonly-deep.d.ts) - Create a deeply immutable version of an `object`/`Map`/`Set`/`Array` type. Use [`Readonly<T>`](https://www.typescriptlang.org/docs/handbook/utility-types.html#readonlytype) if you only need one level deep.

‎source/remove-index-signature.d.ts ‎source/omit-index-signature.d.ts

+14-11
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,7 @@
11
/**
2-
Remove any index signatures from the given object type, so that only explicitly defined properties remain.
2+
Omit any index signatures from the given object type, leaving only explicitly defined properties.
3+
4+
This is the counterpart of `PickIndexSignature`.
35
46
Use-cases:
57
- Remove overly permissive signatures from third-party types.
@@ -34,20 +36,20 @@ type Keyed = {} extends Record<'foo' | 'bar', unknown>
3436
Using a [mapped type](https://www.typescriptlang.org/docs/handbook/2/mapped-types.html#further-exploration), you can then check for each `KeyType` of `ObjectType`...
3537
3638
```
37-
import type {RemoveIndexSignature} from 'type-fest';
39+
import type {OmitIndexSignature} from 'type-fest';
3840
39-
type RemoveIndexSignature<ObjectType> = {
41+
type OmitIndexSignature<ObjectType> = {
4042
[KeyType in keyof ObjectType // Map each key of `ObjectType`...
41-
]: ObjectType[KeyType]; // ...to its original value, i.e. `RemoveIndexSignature<Foo> == Foo`.
43+
]: ObjectType[KeyType]; // ...to its original value, i.e. `OmitIndexSignature<Foo> == Foo`.
4244
};
4345
```
4446
4547
...whether an empty object (`{}`) would be assignable to an object with that `KeyType` (`Record<KeyType, unknown>`)...
4648
4749
```
48-
import type {RemoveIndexSignature} from 'type-fest';
50+
import type {OmitIndexSignature} from 'type-fest';
4951
50-
type RemoveIndexSignature<ObjectType> = {
52+
type OmitIndexSignature<ObjectType> = {
5153
[KeyType in keyof ObjectType
5254
// Is `{}` assignable to `Record<KeyType, unknown>`?
5355
as {} extends Record<KeyType, unknown>
@@ -60,9 +62,9 @@ type RemoveIndexSignature<ObjectType> = {
6062
If `{}` is assignable, it means that `KeyType` is an index signature and we want to remove it. If it is not assignable, `KeyType` is a "real" key and we want to keep it.
6163
6264
```
63-
import type {RemoveIndexSignature} from 'type-fest';
65+
import type {OmitIndexSignature} from 'type-fest';
6466
65-
type RemoveIndexSignature<ObjectType> = {
67+
type OmitIndexSignature<ObjectType> = {
6668
[KeyType in keyof ObjectType
6769
as {} extends Record<KeyType, unknown>
6870
? never // => Remove this `KeyType`.
@@ -73,7 +75,7 @@ type RemoveIndexSignature<ObjectType> = {
7375
7476
@example
7577
```
76-
import type {RemoveIndexSignature} from 'type-fest';
78+
import type {OmitIndexSignature} from 'type-fest';
7779
7880
interface Example {
7981
// These index signatures will be removed.
@@ -91,13 +93,14 @@ interface Example {
9193
qux?: 'baz';
9294
}
9395
94-
type ExampleWithoutIndexSignatures = RemoveIndexSignature<Example>;
96+
type ExampleWithoutIndexSignatures = OmitIndexSignature<Example>;
9597
// => { foo: 'bar'; qux?: 'baz' | undefined; }
9698
```
9799
100+
@see PickIndexSignature
98101
@category Object
99102
*/
100-
export type RemoveIndexSignature<ObjectType> = {
103+
export type OmitIndexSignature<ObjectType> = {
101104
[KeyType in keyof ObjectType as {} extends Record<KeyType, unknown>
102105
? never
103106
: KeyType]: ObjectType[KeyType];

‎source/pick-index-signature.d.ts

+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
/**
2+
Pick only index signatures from the given object type, leaving out all explicitly defined properties.
3+
4+
This is the counterpart of `OmitIndexSignature`.
5+
6+
When you use a type that will iterate through an object that has indexed keys and explicitly defined keys you end up with a type where only the indexed keys are kept. This is because `keyof` of an indexed type always returns `string | number | symbol`, because every key is possible in that object. With this type, you can save the indexed keys and reinject them later, like in the second example below.
7+
8+
@example
9+
```
10+
import type {PickIndexSignature} from 'type-fest';
11+
12+
declare const symbolKey: unique symbol;
13+
14+
type Example = {
15+
// These index signatures will remain.
16+
[x: string]: unknown;
17+
[x: number]: unknown;
18+
[x: symbol]: unknown;
19+
[x: `head-${string}`]: string;
20+
[x: `${string}-tail`]: string;
21+
[x: `head-${string}-tail`]: string;
22+
[x: `${bigint}`]: string;
23+
[x: `embedded-${number}`]: string;
24+
25+
// These explicitly defined keys will be removed.
26+
['snake-case-key']: string;
27+
[symbolKey]: string;
28+
foo: 'bar';
29+
qux?: 'baz';
30+
};
31+
32+
type ExampleIndexSignature = PickIndexSignature<Example>;
33+
// {
34+
// [x: string]: unknown;
35+
// [x: number]: unknown;
36+
// [x: symbol]: unknown;
37+
// [x: `head-${string}`]: string;
38+
// [x: `${string}-tail`]: string;
39+
// [x: `head-${string}-tail`]: string;
40+
// [x: `${bigint}`]: string;
41+
// [x: `embedded-${number}`]: string;
42+
// }
43+
```
44+
45+
@example
46+
```
47+
import type {OmitIndexSignature, PickIndexSignature, Simplify} from 'type-fest';
48+
49+
type Foo = {
50+
[x: string]: string;
51+
foo: string;
52+
bar: number;
53+
};
54+
55+
// Imagine that you want a new type `Bar` that comes from `Foo`.
56+
// => {
57+
// [x: string]: string;
58+
// bar: number;
59+
// };
60+
61+
type Bar = Omit<Foo, 'foo'>;
62+
// This is not working because `Omit` returns only indexed keys.
63+
// => {
64+
// [x: string]: string;
65+
// [x: number]: string;
66+
// }
67+
68+
// One solution is to save the indexed signatures to new type.
69+
type FooIndexSignature = PickIndexSignature<Foo>;
70+
// => {
71+
// [x: string]: string;
72+
// }
73+
74+
// Get a new type without index signatures.
75+
type FooWithoutIndexSignature = OmitIndexSignature<Foo>;
76+
// => {
77+
// foo: string;
78+
// bar: number;
79+
// }
80+
81+
// At this point we can use Omit to get our new type.
82+
type BarWithoutIndexSignature = Omit<FooWithoutIndexSignature, 'foo'>;
83+
// => {
84+
// bar: number;
85+
// }
86+
87+
// And finally we can merge back the indexed signatures.
88+
type BarWithIndexSignature = Simplify<BarWithoutIndexSignature & FooIndexSignature>;
89+
// => {
90+
// [x: string]: string;
91+
// bar: number;
92+
// }
93+
```
94+
95+
@see OmitIndexSignature
96+
@category Object
97+
*/
98+
export type PickIndexSignature<ObjectType> = {
99+
[KeyType in keyof ObjectType as {} extends Record<KeyType, unknown>
100+
? KeyType
101+
: never]: ObjectType[KeyType];
102+
};

‎test-d/remove-index-signature.ts ‎test-d/omit-index-signature.ts

+3-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import {expectType} from 'tsd';
2-
import type {RemoveIndexSignature} from '../index';
2+
import type {OmitIndexSignature} from '../index';
33

44
type ExampleInterface = {
55
// These index signatures will be removed.
@@ -24,13 +24,13 @@ type MappedType<ObjectType> = {
2424
};
2525
};
2626

27-
declare const exampleInterfaceKnownKeys: RemoveIndexSignature<ExampleInterface>;
27+
declare const exampleInterfaceKnownKeys: OmitIndexSignature<ExampleInterface>;
2828
expectType<{
2929
foo: 'bar';
3030
qux?: 'baz';
3131
}>(exampleInterfaceKnownKeys);
3232

33-
declare const exampleMappedTypeKnownKeys: RemoveIndexSignature<
33+
declare const exampleMappedTypeKnownKeys: OmitIndexSignature<
3434
MappedType<ExampleInterface>
3535
>;
3636
expectType<{

‎test-d/pick-index-signature.ts

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import {expectType} from 'tsd';
2+
import type {PickIndexSignature, Simplify} from '../index';
3+
4+
declare const symbolKey: unique symbol;
5+
6+
type Foo = {
7+
[x: string]: unknown;
8+
[x: number]: unknown;
9+
[x: symbol]: unknown;
10+
[x: `head-${string}`]: string;
11+
[x: `${string}-tail`]: string;
12+
[x: `head-${string}-tail`]: string;
13+
[x: `${bigint}`]: string;
14+
[x: `embedded-${number}`]: string;
15+
};
16+
17+
type Bar = {
18+
['snake-case-key']: string;
19+
[symbolKey]: string;
20+
foo: 'bar';
21+
qux?: 'baz';
22+
};
23+
24+
type FooBar = Simplify<Foo & Bar>;
25+
26+
declare const indexSignature: PickIndexSignature<FooBar>;
27+
28+
expectType<Foo>(indexSignature);

0 commit comments

Comments
 (0)
Please sign in to comment.