forked from sindresorhus/type-fest
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathset-parameter-type.d.ts
68 lines (56 loc) · 2.68 KB
/
set-parameter-type.d.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
import type {IsUnknown} from './is-unknown';
/**
Create a tuple that replaces the given `Tuple`'s elements with the given `Record_`'s values at the given indices.
*/
type MergeObjectToTuple<Tuple extends unknown[], Record_ extends Record<number, unknown>> = {
[K in keyof Tuple]: Record_ extends Record<string, unknown>
// Handle object case like `{0: string, 1: number}`
? K extends `${infer NumberK extends number}`
? NumberK extends keyof Record_ ? Record_[K] : Tuple[K]
: never // Should not happen
// Handle tuple case like `[string, number]`
: K extends keyof Record_ ? Record_[K] : Tuple[K]
};
/**
Create a function that replaces some parameters with the given parameters.
The parameters that are not specified will be kept as-is.
Note:
- This type will ignore the given function's generic type.
- If you change the parameter type that return type depends on, the return type will not change:
```
const fn = (a: number) => a;
//=> fn: (a: number) => number;
// We change type of `a` to `string`, but return type is still `number`.
type Fn = SetParameterType<typeof fn, {0: string}>;
//=> (a: string) => number;
```
Use-case:
- Define a wrapped function that receives something different while returning the same type.
- Mocking and testing.
- Overload function type. (See example)
@example
```
import type {SetParameterType} from 'type-fest';
type HandleMessage = (data: Data, message: string) => void;
type HandleOk = SetParameterType<HandleMessage, {0: SuccessData, 1: 'ok'}>;
//=> type HandleOk = (data: SuccessData, message: 'ok') => void;
// Another way to define the parameters to replace.
type HandleError = SetParameterType<HandleMessage, [data: ErrorData, message: 'error']>;
//=> type HandleError = (data: ErrorData, message: 'error') => void;
// Could change single parameter type.
type HandleWarn = SetParameterType<HandleMessage, {1: 'warn'}>;
//=> type HandleWarn = (data: Data, message: 'warn') => void;
```
@category Function
*/
export type SetParameterType<Fn extends (...arguments_: any[]) => unknown, P extends Record<number, unknown>> =
// Just using `Parameters<Fn>` isn't ideal because it doesn't handle the `this` fake parameter.
Fn extends (this: infer ThisArg, ...arguments_: infer Arguments) => unknown
? (
// If a function did not specify the `this` fake parameter, it will be inferred to `unknown`.
// We want to detect this situation just to display a friendlier type upon hovering on an IntelliSense-powered IDE.
IsUnknown<ThisArg> extends true
? (...arguments_: MergeObjectToTuple<Arguments, P>) => ReturnType<Fn>
: (this: ThisArg, ...arguments_: MergeObjectToTuple<Arguments, P>) => ReturnType<Fn>
)
: Fn; // This part should be unreachable