65 lines
2.2 KiB
TypeScript
65 lines
2.2 KiB
TypeScript
|
type RemoveFromTuple<
|
||
|
Tuple extends readonly unknown[],
|
||
|
RemoveCount extends number,
|
||
|
Index extends 1[] = []
|
||
|
> = Index["length"] extends RemoveCount
|
||
|
? Tuple
|
||
|
: Tuple extends [infer First, ...infer Rest]
|
||
|
? RemoveFromTuple<Rest, RemoveCount, [...Index, 1]>
|
||
|
: Tuple;
|
||
|
|
||
|
type ConcatTuples<
|
||
|
Prefix extends readonly unknown[],
|
||
|
Suffix extends readonly unknown[]
|
||
|
> = [...Prefix, ...Suffix];
|
||
|
|
||
|
type ExtractFunctionParams<T> = T extends (this: infer TThis, ...args: infer P extends readonly unknown[]) => infer R
|
||
|
? { thisArg: TThis; params: P; returnType: R }
|
||
|
: never;
|
||
|
|
||
|
type BindFunction<
|
||
|
T extends (this: any, ...args: any[]) => any,
|
||
|
TThis,
|
||
|
TBoundArgs extends readonly unknown[],
|
||
|
ReceiverBound extends boolean
|
||
|
> = ExtractFunctionParams<T> extends {
|
||
|
thisArg: infer OrigThis;
|
||
|
params: infer P extends readonly unknown[];
|
||
|
returnType: infer R;
|
||
|
}
|
||
|
? ReceiverBound extends true
|
||
|
? (...args: RemoveFromTuple<P, Extract<TBoundArgs["length"], number>>) => R extends [OrigThis, ...infer Rest]
|
||
|
? [TThis, ...Rest] // Replace `this` with `thisArg`
|
||
|
: R
|
||
|
: <U, RemainingArgs extends RemoveFromTuple<P, Extract<TBoundArgs["length"], number>>>(
|
||
|
thisArg: U,
|
||
|
...args: RemainingArgs
|
||
|
) => R extends [OrigThis, ...infer Rest]
|
||
|
? [U, ...ConcatTuples<TBoundArgs, Rest>] // Preserve bound args in return type
|
||
|
: R
|
||
|
: never;
|
||
|
|
||
|
declare function callBind<
|
||
|
const T extends (this: any, ...args: any[]) => any,
|
||
|
Extracted extends ExtractFunctionParams<T>,
|
||
|
const TBoundArgs extends Partial<Extracted["params"]> & readonly unknown[],
|
||
|
const TThis extends Extracted["thisArg"]
|
||
|
>(
|
||
|
args: [fn: T, thisArg: TThis, ...boundArgs: TBoundArgs]
|
||
|
): BindFunction<T, TThis, TBoundArgs, true>;
|
||
|
|
||
|
declare function callBind<
|
||
|
const T extends (this: any, ...args: any[]) => any,
|
||
|
Extracted extends ExtractFunctionParams<T>,
|
||
|
const TBoundArgs extends Partial<Extracted["params"]> & readonly unknown[]
|
||
|
>(
|
||
|
args: [fn: T, ...boundArgs: TBoundArgs]
|
||
|
): BindFunction<T, Extracted["thisArg"], TBoundArgs, false>;
|
||
|
|
||
|
declare function callBind<const TArgs extends readonly unknown[]>(
|
||
|
args: [fn: Exclude<TArgs[0], Function>, ...rest: TArgs]
|
||
|
): never;
|
||
|
|
||
|
// export as namespace callBind;
|
||
|
export = callBind;
|