-
Notifications
You must be signed in to change notification settings - Fork 6
/
bimappable.ts
38 lines (34 loc) · 1.01 KB
/
bimappable.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
/**
* Bimappable is a structure that allows a function to be applied inside of the
* associated concrete structure but is specific to a second held value of that
* structure..
*
* @module Bimappable
* @since 2.0.0
*/
import type { $, Hold, Kind } from "./kind.ts";
import type { Mappable } from "./mappable.ts";
import { pipe } from "./fn.ts";
/**
* The Bimappable interface. Bimapple includes the methods map and mapSecond.
*/
export interface Bimappable<U extends Kind> extends Mappable<U>, Hold<U> {
readonly mapSecond: <B, J>(
fbj: (value: B) => J,
) => <A = never, C = never, D = unknown, E = unknown>(
ta: $<U, [A, B, C], [D], [E]>,
) => $<U, [A, J, C], [D], [E]>;
}
/**
* @since 2.0.0
*/
export function bimap<U extends Kind>(
{ map, mapSecond }: Bimappable<U>,
): <A, B, I, J>(
fai: (a: A) => I,
fbj: (b: B) => J,
) => <C = never, D = unknown, E = unknown>(
ua: $<U, [A, B, C], [D], [E]>,
) => $<U, [I, J, C], [D], [E]> {
return (fai, fbj) => (ua) => pipe(ua, map(fai), mapSecond(fbj));
}