mirror of
https://github.com/Kir-Antipov/mc-publish.git
synced 2025-01-22 18:14:45 +01:00
Added a possibility to make objects callable
This commit is contained in:
parent
7882738bce
commit
d263c15393
2 changed files with 96 additions and 0 deletions
47
src/utils/functions/callable.ts
Normal file
47
src/utils/functions/callable.ts
Normal file
|
@ -0,0 +1,47 @@
|
|||
/**
|
||||
* A symbol representing the `call` function of a {@link Callable} object.
|
||||
*/
|
||||
export const CALL = Symbol.for("call");
|
||||
|
||||
/**
|
||||
* Represents an object, which can be converted into a {@link Callable} one.
|
||||
*/
|
||||
interface SemiCallable {
|
||||
/**
|
||||
* A method that should be invoked, when an object is used as a function.
|
||||
*/
|
||||
[CALL](...args: unknown[]): unknown;
|
||||
}
|
||||
|
||||
/**
|
||||
* Represents an object, which can be called like a function.
|
||||
*
|
||||
* @template T - The type of the underlying object.
|
||||
*/
|
||||
export type Callable<T extends SemiCallable> = T & {
|
||||
/**
|
||||
* Redirects a call to the {@link CALL} function.
|
||||
*/
|
||||
(...args: Parameters<T[typeof CALL]>): ReturnType<T[typeof CALL]>;
|
||||
};
|
||||
|
||||
/**
|
||||
* Makes an object callable.
|
||||
*
|
||||
* @template T - The type of the object.
|
||||
* @param obj - The object to make callable.
|
||||
*
|
||||
* @returns A new {@link Callable} object with the same properties as the original one, but which can be called like a function.
|
||||
*/
|
||||
export function makeCallable<T extends SemiCallable>(obj: T): Callable<T> {
|
||||
/**
|
||||
* Redirects a call to the {@link CALL} function.
|
||||
*/
|
||||
function call(...args: unknown[]): unknown {
|
||||
return (call as unknown as T)[CALL](...args);
|
||||
}
|
||||
|
||||
Object.assign(call, obj);
|
||||
Object.setPrototypeOf(call, Object.getPrototypeOf(obj));
|
||||
return call as unknown as Callable<T>;
|
||||
}
|
49
tests/unit/utils/functions/callable.spec.ts
Normal file
49
tests/unit/utils/functions/callable.spec.ts
Normal file
|
@ -0,0 +1,49 @@
|
|||
import { CALL, makeCallable } from "@/utils/functions/callable";
|
||||
|
||||
describe("makeCallable", () => {
|
||||
test("makes an object callable", () => {
|
||||
const obj = {
|
||||
[CALL]: (a: number, b: number) => a + b,
|
||||
};
|
||||
|
||||
const callable = makeCallable(obj);
|
||||
|
||||
expect(callable(1, 2)).toBe(3);
|
||||
});
|
||||
|
||||
test("preserves object properties", () => {
|
||||
const obj = {
|
||||
foo: 42,
|
||||
|
||||
[CALL](): number {
|
||||
return this.foo;
|
||||
},
|
||||
};
|
||||
|
||||
const callable = makeCallable(obj);
|
||||
|
||||
expect(callable()).toBe(42);
|
||||
expect(callable.foo).toBe(42);
|
||||
});
|
||||
|
||||
test("preserves object prototype", () => {
|
||||
class FooClass {
|
||||
foo: number;
|
||||
|
||||
constructor(foo: number) {
|
||||
this.foo = foo;
|
||||
}
|
||||
|
||||
[CALL](): number {
|
||||
return this.foo;
|
||||
}
|
||||
}
|
||||
|
||||
const obj = new FooClass(42);
|
||||
const callable = makeCallable(obj);
|
||||
|
||||
expect(callable).toBeInstanceOf(FooClass);
|
||||
expect(callable.foo).toBe(42);
|
||||
expect(callable()).toBe(42);
|
||||
});
|
||||
});
|
Loading…
Reference in a new issue