Skip to content

Commit eac8c6f

Browse files
authoredAug 26, 2022
Add TupleToUnion type (#440)
1 parent beaabe1 commit eac8c6f

File tree

4 files changed

+79
-0
lines changed

4 files changed

+79
-0
lines changed
 

‎index.d.ts

+1
Original file line numberDiff line numberDiff line change
@@ -64,6 +64,7 @@ export {HasOptionalKeys} from './source/has-optional-keys';
6464
export {RequiredKeysOf} from './source/required-keys-of';
6565
export {HasRequiredKeys} from './source/has-required-keys';
6666
export {Spread} from './source/spread';
67+
export {TupleToUnion} from './source/tuple-to-union';
6768

6869
// Template literal types
6970
export {CamelCase} from './source/camel-case';

‎readme.md

+1
Original file line numberDiff line numberDiff line change
@@ -194,6 +194,7 @@ Click the type names for complete docs.
194194
- [`MultidimensionalArray`](source/multidimensional-array.d.ts) - Create a type that represents a multidimensional array of the given type and dimensions.
195195
- [`MultidimensionalReadonlyArray`](source/multidimensional-readonly-array.d.ts) - Create a type that represents a multidimensional readonly array of the given type and dimensions.
196196
- [`ReadonlyTuple`](source/readonly-tuple.d.ts) - Create a type that represents a read-only tuple of the given type and length.
197+
- [`TupleToUnion`](source/tuple-to-union.d.ts) - Convert a tuple into a union type of its elements.
197198

198199
### Numeric
199200

‎source/tuple-to-union.d.ts

+51
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,51 @@
1+
/**
2+
Convert a tuple into a union type of its elements.
3+
4+
This can be useful when you have a fixed set of allowed values and want a type defining only the allowed values, but do not want to repeat yourself.
5+
6+
@example
7+
```
8+
import type {TupleToUnion} from 'type-fest';
9+
10+
const destinations = ['a', 'b', 'c'] as const;
11+
12+
type Destination = TupleToUnion<typeof destinations>;
13+
//=> 'a' | 'b' | 'c'
14+
15+
function verifyDestination(destination: unknown): destination is Destination {
16+
return destinations.includes(destination as any);
17+
}
18+
19+
type RequestBody = {
20+
deliverTo: Destination;
21+
};
22+
23+
function verifyRequestBody(body: unknown): body is RequestBody {
24+
const deliverTo = (body as any).deliverTo;
25+
return typeof body === 'object' && body !== null && verifyDestination(deliverTo);
26+
}
27+
```
28+
29+
Alternatively, you may use `typeof destinations[number]`. If `destinations` is a tuple, there is no difference. However if `destinations` is a string, the resulting type will the union of the characters in the string. Other types of `destinations` may result in a compile error. In comparison, TupleToUnion will return `never` if a tuple is not provided.
30+
31+
@example
32+
```
33+
const destinations = ['a', 'b', 'c'] as const;
34+
35+
type Destination = typeof destinations[number];
36+
//=> 'a' | 'b' | 'c'
37+
38+
const erroringType = new Set(['a', 'b', 'c']);
39+
40+
type ErroringType = typeof erroringType[number];
41+
//=> Type 'Set<string>' has no matching index signature for type 'number'. ts(2537)
42+
43+
const numberBool: { [n: number]: boolean } = { 1: true };
44+
45+
type NumberBool = typeof numberBool[number];
46+
//=> boolean
47+
```
48+
49+
@category Array
50+
*/
51+
export type TupleToUnion<ArrayType> = ArrayType extends readonly [infer Head, ...(infer Rest)] ? Head | TupleToUnion<Rest> : never;

‎test-d/tuple-to-union.ts

+26
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,26 @@
1+
import {expectAssignable, expectNotType, expectType} from 'tsd';
2+
import type {TupleToUnion} from '../index';
3+
4+
const options = ['a', 'b', 'c'] as const;
5+
type Options = TupleToUnion<typeof options>;
6+
7+
const a: Options = 'a';
8+
expectAssignable<Options>(a);
9+
expectType<'a'>(a);
10+
expectNotType<'b'>(a);
11+
expectNotType<'c'>(a);
12+
13+
const b: Options = 'b';
14+
expectAssignable<Options>(b);
15+
expectNotType<'a'>(b);
16+
expectType<'b'>(b);
17+
expectNotType<'c'>(b);
18+
19+
const c: Options = 'c';
20+
expectAssignable<Options>(c);
21+
expectNotType<'a'>(c);
22+
expectNotType<'b'>(c);
23+
expectType<'c'>(c);
24+
25+
declare const notAnArray: TupleToUnion<[]>;
26+
expectType<never>(notAnArray);

0 commit comments

Comments
 (0)
Please sign in to comment.