-
Notifications
You must be signed in to change notification settings - Fork 464
/
permutate.hpp
140 lines (119 loc) · 3.09 KB
/
permutate.hpp
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
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
#ifndef SASS_PATHS_H
#define SASS_PATHS_H
#include <vector>
namespace Sass {
// Returns a list of all possible paths through the given lists.
//
// For example, given `[[1, 2], [3, 4], [5]]`, this returns:
//
// ```
// [[1, 3, 5],
// [2, 3, 5],
// [1, 4, 5],
// [2, 4, 5]]
// ```
//
// Note: called `paths` in dart-sass
template <class T>
std::vector<std::vector<T>> permutate(
const std::vector<std::vector<T>>& in)
{
size_t L = in.size(), n = 0;
// Exit early if any entry is empty
for (size_t i = 0; i < L; i += 1) {
if (in[i].size() == 0) return {};
}
size_t* state = new size_t[L + 1];
std::vector<std::vector<T>> out;
// First initialize all states for every permutation group
for (size_t i = 0; i < L; i += 1) {
state[i] = in[i].size() - 1;
}
while (true) {
std::vector<T> perm;
// Create one permutation for state
for (size_t i = 0; i < L; i += 1) {
perm.push_back(in.at(i).at(in[i].size() - state[i] - 1));
}
// Current group finished
if (state[n] == 0) {
// Find position of next decrement
while (n < L && state[++n] == 0) {}
if (n == L) {
out.push_back(perm);
break;
}
state[n] -= 1;
for (size_t p = 0; p < n; p += 1) {
state[p] = in[p].size() - 1;
}
// Restart from front
n = 0;
}
else {
state[n] -= 1;
}
out.push_back(perm);
}
delete[] state;
return out;
}
// EO permutate
// ToDo: this variant is used in resolve_parent_refs
template <class T>
std::vector<std::vector<T>>
permutateAlt(const std::vector<std::vector<T>>& in) {
size_t L = in.size();
size_t n = in.size() - 1;
size_t* state = new size_t[L];
std::vector< std::vector<T>> out;
// First initialize all states for every permutation group
for (size_t i = 0; i < L; i += 1) {
if (in[i].size() == 0) return {};
state[i] = in[i].size() - 1;
}
while (true) {
/*
// std::cerr << "PERM: ";
for (size_t p = 0; p < L; p++)
{ // std::cerr << state[p] << " "; }
// std::cerr << "\n";
*/
std::vector<T> perm;
// Create one permutation for state
for (size_t i = 0; i < L; i += 1) {
perm.push_back(in.at(i).at(in[i].size() - state[i] - 1));
}
// Current group finished
if (state[n] == 0) {
// Find position of next decrement
while (n > 0 && state[--n] == 0)
{
}
// Check for end condition
if (state[n] != 0) {
// Decrease next on the left side
state[n] -= 1;
// Reset all counters to the right
for (size_t p = n + 1; p < L; p += 1) {
state[p] = in[p].size() - 1;
}
// Restart from end
n = L - 1;
}
else {
out.push_back(perm);
break;
}
}
else {
state[n] -= 1;
}
out.push_back(perm);
}
delete[] state;
return out;
}
// EO permutateAlt
}
#endif