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;
|