2022-04-20 03:05:34 +02:00
|
|
|
#include "propagator.h"
|
|
|
|
#include "wave.h"
|
2022-04-20 02:39:35 +02:00
|
|
|
|
2022-04-20 03:24:50 +02:00
|
|
|
void Propagator::init_compatible() {
|
2022-04-21 17:05:26 +02:00
|
|
|
CompatibilityEntry value;
|
2022-04-20 03:05:34 +02:00
|
|
|
// We compute the number of pattern compatible in all directions.
|
2022-04-21 16:43:04 +02:00
|
|
|
for (uint32_t y = 0; y < wave_height; y++) {
|
|
|
|
for (uint32_t x = 0; x < wave_width; x++) {
|
|
|
|
for (uint32_t pattern = 0; pattern < patterns_size; pattern++) {
|
2022-04-21 17:05:26 +02:00
|
|
|
|
2022-04-20 03:05:34 +02:00
|
|
|
for (int direction = 0; direction < 4; direction++) {
|
2022-04-21 17:05:26 +02:00
|
|
|
value.direction[direction] = static_cast<uint32_t>(propagator_state[pattern].directions[get_opposite_direction(direction)].size());
|
2022-04-20 03:05:34 +02:00
|
|
|
}
|
2022-04-21 16:43:04 +02:00
|
|
|
|
2022-04-20 03:05:34 +02:00
|
|
|
compatible.get(y, x, pattern) = value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-04-20 02:39:35 +02:00
|
|
|
}
|
|
|
|
|
2022-04-20 03:24:50 +02:00
|
|
|
void Propagator::propagate(Wave &wave) {
|
2022-04-20 03:05:34 +02:00
|
|
|
// We propagate every element while there is element to propagate.
|
|
|
|
while (propagating.size() != 0) {
|
|
|
|
// The cell and pattern that has been set to false.
|
2022-04-21 16:43:04 +02:00
|
|
|
uint32_t y1, x1, pattern;
|
2022-04-21 16:31:03 +02:00
|
|
|
std::tie(y1, x1, pattern) = propagating[propagating.size() - 1];
|
|
|
|
propagating.resize(propagating.size() - 1);
|
2022-04-20 02:39:35 +02:00
|
|
|
|
2022-04-20 03:05:34 +02:00
|
|
|
// We propagate the information in all 4 directions.
|
2022-04-21 16:43:04 +02:00
|
|
|
for (uint32_t direction = 0; direction < 4; direction++) {
|
2022-04-20 03:05:34 +02:00
|
|
|
// We get the next cell in the direction direction.
|
|
|
|
int dx = directions_x[direction];
|
|
|
|
int dy = directions_y[direction];
|
|
|
|
int x2, y2;
|
|
|
|
if (periodic_output) {
|
|
|
|
x2 = ((int)x1 + dx + (int)wave.width) % wave.width;
|
|
|
|
y2 = ((int)y1 + dy + (int)wave.height) % wave.height;
|
|
|
|
} else {
|
|
|
|
x2 = x1 + dx;
|
|
|
|
y2 = y1 + dy;
|
|
|
|
if (x2 < 0 || x2 >= (int)wave.width) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (y2 < 0 || y2 >= (int)wave.height) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
}
|
2022-04-20 02:39:35 +02:00
|
|
|
|
2022-04-20 03:05:34 +02:00
|
|
|
// The index of the second cell, and the patterns compatible
|
2022-04-21 16:43:04 +02:00
|
|
|
uint32_t i2 = x2 + y2 * wave.width;
|
2022-04-21 17:05:26 +02:00
|
|
|
const Vector<uint32_t> &patterns = propagator_state[pattern].directions[direction];
|
2022-04-20 02:39:35 +02:00
|
|
|
|
2022-04-20 03:05:34 +02:00
|
|
|
// For every pattern that could be placed in that cell without being in
|
|
|
|
// contradiction with pattern1
|
2022-04-21 16:31:03 +02:00
|
|
|
int size = patterns.size();
|
|
|
|
for (int i = 0; i < size; ++i) {
|
2022-04-21 16:43:04 +02:00
|
|
|
uint32_t pattern = patterns[i];
|
2022-04-21 16:31:03 +02:00
|
|
|
|
2022-04-20 03:05:34 +02:00
|
|
|
// We decrease the number of compatible patterns in the opposite
|
|
|
|
// direction If the pattern was discarded from the wave, the element
|
|
|
|
// is still negative, which is not a problem
|
2022-04-21 17:05:26 +02:00
|
|
|
CompatibilityEntry &value = compatible.get(y2, x2, pattern);
|
|
|
|
value.direction[direction]--;
|
2022-04-20 02:39:35 +02:00
|
|
|
|
2022-04-20 03:05:34 +02:00
|
|
|
// If the element was set to 0 with this operation, we need to remove
|
|
|
|
// the pattern from the wave, and propagate the information
|
2022-04-21 17:05:26 +02:00
|
|
|
if (value.direction[direction] == 0) {
|
2022-04-21 16:31:03 +02:00
|
|
|
add_to_propagator(y2, x2, pattern);
|
|
|
|
wave.set(i2, pattern, false);
|
2022-04-20 03:05:34 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-04-20 02:39:35 +02:00
|
|
|
}
|