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
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
use crate::functional::{ApplyFunctor, Functor};
pub trait EmptyTuple {}
impl EmptyTuple for () {}
pub trait SingleTuple {}
impl<A> SingleTuple for (A,) {}
pub trait Pair {}
impl<A, B> Pair for (A, B) {}
pub trait Triple {}
impl<A, B, C> Triple for (A, B, C) {}
pub trait Quadruple {}
impl<A, B, C, D> Quadruple for (A, B, C, D) {}
pub struct FirstOfFunctor {}
pub type FirstOf<Tuple> = ApplyFunctor<FirstOfFunctor, Tuple>;
impl<A> Functor<(A,)> for FirstOfFunctor {
type Output = A;
}
impl<A, B> Functor<(A, B)> for FirstOfFunctor {
type Output = A;
}
impl<A, B, C> Functor<(A, B, C)> for FirstOfFunctor {
type Output = A;
}
impl<A, B, C, D> Functor<(A, B, C, D)> for FirstOfFunctor {
type Output = A;
}
impl<A, B, C, D, E> Functor<(A, B, C, D, E)> for FirstOfFunctor {
type Output = A;
}
pub struct SecondOfFunctor {}
pub type SecondOf<Tuple> = ApplyFunctor<SecondOfFunctor, Tuple>;
impl<A, B> Functor<(A, B)> for SecondOfFunctor {
type Output = B;
}
impl<A, B, C> Functor<(A, B, C)> for SecondOfFunctor {
type Output = B;
}
impl<A, B, C, D> Functor<(A, B, C, D)> for SecondOfFunctor {
type Output = B;
}
impl<A, B, C, D, E> Functor<(A, B, C, D, E)> for SecondOfFunctor {
type Output = B;
}
pub struct ThirdOfFunctor {}
pub type ThirdOf<Tuple> = ApplyFunctor<ThirdOfFunctor, Tuple>;
impl<A, B, C> Functor<(A, B, C)> for ThirdOfFunctor {
type Output = C;
}
impl<A, B, C, D> Functor<(A, B, C, D)> for ThirdOfFunctor {
type Output = C;
}
impl<A, B, C, D, E> Functor<(A, B, C, D, E)> for ThirdOfFunctor {
type Output = C;
}
pub struct LeftAssociateFunctor {}
pub type LeftAssociate<Tuple> = ApplyFunctor<LeftAssociateFunctor, Tuple>;
impl<A, B, C> Functor<(A, (B, C))> for LeftAssociateFunctor {
type Output = ((A, B), C);
}
pub struct RightAssociateFunctor {}
pub type RightAssociate<Tuple> = ApplyFunctor<RightAssociateFunctor, Tuple>;
impl<A, B, C> Functor<((A, B), C)> for RightAssociateFunctor {
type Output = (A, (B, C));
}