1 module nxt.combinations;
2 
3 /** Given non-negative integers $(D m) and $(D n), generate all size $(D m)
4    combinations of the integers from 0 to $(D n)-1 in sorted order (each
5    combination is sorted and the entire table is sorted).
6 
7    For example, 3 comb 5 is
8    0 1 2
9    0 1 3
10    0 1 4
11    0 2 3
12    0 2 4
13    0 3 4
14    1 2 3
15    1 2 4
16    1 3 4
17    2 3 4
18 
19    See_Also: http://rosettacode.org/wiki/Combinations
20 */
21 struct Combinations(T, bool copy = true, bool useArray = true)
22 {
23     import std.container: Array;
24     import std.traits: Unqual;
25 
26     static if (useArray)
27         alias Indices = Array!size_t;
28     else
29         alias Indices = size_t[];
30 
31     Unqual!T[] pool, front;
32     size_t r, n;
33     bool empty = false;
34 
35     Indices indices;
36 
37     size_t len;
38     bool lenComputed = false;
39 
40     this(T[] pool_, in size_t r_) // pure nothrow @safe
41     {
42         this.pool = pool_.dup;
43         this.r = r_;
44         this.n = pool.length;
45         if (r > n)
46             empty = true;
47 
48         indices.length = r;
49 
50         size_t i;
51 
52         i = 0;
53         foreach (ref ini; indices[])
54             ini = i++;
55 
56         front.length = r;
57 
58         i = 0;
59         foreach (immutable idx; indices[])
60             front[i++] = pool[idx];
61     }
62 
63     @property size_t length() /*logic_const*/ // pure nothrow @nogc
64     {
65         static size_t binomial(size_t n, size_t k) // pure nothrow @safe @nogc
66         in
67         {
68             assert(n > 0, "binomial: n must be > 0.");
69         }
70         do
71         {
72             if (k < 0 || k > n)
73                 return 0;
74             if (k > (n / 2))
75                 k = n - k;
76             size_t result = 1;
77             foreach (size_t d; 1 .. k + 1) {
78                 result *= n;
79                 n--;
80                 result /= d;
81             }
82             return result;
83         }
84 
85         if (!lenComputed)
86         {
87             // Set cache.
88             len = binomial(n, r);
89             lenComputed = true;
90         }
91         return len;
92     }
93 
94     void popFront() // pure nothrow @safe
95     {
96         if (!empty)
97         {
98             bool broken = false;
99             size_t pos = 0;
100             foreach_reverse (immutable i; 0 .. r)
101             {
102                 pos = i;
103                 if (indices[i] != i + n - r)
104                 {
105                     broken = true;
106                     break;
107                 }
108             }
109             if (!broken)
110             {
111                 empty = true;
112                 return;
113             }
114             indices[pos]++;
115             foreach (immutable j; pos + 1 .. r)
116                 indices[j] = indices[j - 1] + 1;
117             static if (copy)
118                 front = new Unqual!T[front.length];
119 
120             size_t i = 0;
121             foreach (immutable idx; indices[])
122             {
123                 front[i] = pool[idx];
124                 i++;
125             }
126         }
127     }
128 }
129 
130 auto combinations(bool copy = true, T, bool useArray = false)(T[] items, in size_t k)
131 in(items.length)
132 	=> Combinations!(T, copy, useArray)(items, k);
133 
134 unittest
135 {
136     import std.algorithm: equal, map;
137     // assert(equal([1, 2, 3, 4].combinations!false(2), [[3, 4], [3, 4], [3, 4], [3, 4], [3, 4], [3, 4]]));
138     enum solution = [[1, 2],
139                      [1, 3],
140                      [1, 4],
141                      [2, 3],
142                      [2, 4],
143                      [3, 4]];
144     assert(equal([1, 2, 3, 4].combinations!true(2), solution));
145     assert(equal([1, 2, 3, 4].combinations(2).map!(x => x), solution));
146 }
147 
148 import std.range.primitives : isInputRange;
149 
150 /** All Unordered Element Pairs (2-Element Subsets) of a $(D Range).
151 
152     TODO: Add template parameter to decide if .array should be used internally.
153 
154     See_Also: http://forum.dlang.org/thread/iqkybajwdzcvdytakgvw@forum.dlang.org#post-vhufbwsqbssyqwfxxbuu:40forum.dlang.org
155     See_Also: https://issues.dlang.org/show_bug.cgi?id=6788
156     See_Also: https://issues.dlang.org/show_bug.cgi?id=7128
157 */
158 auto pairwise(R)(R r)
159 if (isInputRange!R)
160 {
161     struct Pairwise(R)
162     {
163         import core.internal.traits : Unqual;
164         import std.traits : ForeachType;
165         import std.typecons: Tuple;
166 
167         alias UR = Unqual!R;
168         alias E = ForeachType!UR;
169         alias Pair = Tuple!(E, E);
170 
171         import std.range.primitives : isRandomAccessRange, hasLength;
172         import std.traits : isNarrowString;
173 
174         static if (isRandomAccessRange!R &&
175                    hasLength!R &&
176                    !isNarrowString!R)
177         {
178 
179             this(R r_)
180             {
181                 this._input = r_;
182                 j = 1;
183             }
184             @property bool empty() => j >= _input.length;
185             @property Pair front() => typeof(return)(_input[i], _input[j]);
186             void popFront()
187             {
188                 if (j >= _input.length - 1)
189                 {
190                     i++;
191                     j = i + 1;
192                 }
193                 else
194                     j++;
195             }
196             private size_t i, j;
197         }
198         else // isInputRange!UR
199         {
200             import std.range : dropOne;
201             this(R r_)
202             {
203                 this._input = r_;
204                 i = r_;
205                 if (!i.empty)
206                     j = i.dropOne;
207                 else
208                     j = UR.init;
209             }
210             @property bool empty() => j.empty;
211             @property Pair front() => typeof(return)(i.front, j.front);
212             void popFront()
213             {
214                 j.popFront();
215                 if (j.empty)
216                 {
217                     i.popFront();
218                     j = i.dropOne;
219                 }
220             }
221             private UR i, j; // temporary copies of $(D _input)
222         }
223 
224     private:
225         UR _input;
226     }
227 
228     return Pairwise!R(r);
229 }
230 
231 /// test RandomAccessRange input
232 unittest
233 {
234     import std.algorithm: equal, filter;
235     import std.typecons: Tuple;
236 
237     assert((new int[0]).pairwise.empty);
238     assert([1].pairwise.empty);
239 
240     alias T = Tuple!(int, int);
241     assert(equal([1, 2].pairwise,
242                  [T(1, 2)]));
243     assert(equal([1, 2, 3].pairwise,
244                  [T(1, 2), T(1, 3), T(2, 3)]));
245     assert(equal([1, 2, 3, 4].pairwise,
246                  [T(1, 2), T(1, 3), T(1, 4),
247                   T(2, 3), T(2, 4), T(3, 4)]));
248 }
249 
250 /// test ForwardRange input
251 unittest
252 {
253     import std.algorithm: equal, filter;
254     import std.array : array;
255 
256     auto p = [1].filter!"a < 4".pairwise;
257     assert(p.empty);
258 
259     assert(equal(p.array,
260                  [1].pairwise.array));
261 
262     assert(equal([1, 2, 3, 4].filter!"a < 4".pairwise,
263                  [1, 2, 3].pairwise));
264 }