2022-04-20 03:05:34 +02:00
|
|
|
#ifndef FAST_WFC_PROPAGATOR_HPP_
|
|
|
|
#define FAST_WFC_PROPAGATOR_HPP_
|
|
|
|
|
|
|
|
#include "array_3d.h"
|
2022-04-21 17:05:26 +02:00
|
|
|
#include "core/vector.h"
|
2022-04-20 03:24:50 +02:00
|
|
|
#include "direction.h"
|
2022-04-20 03:05:34 +02:00
|
|
|
|
|
|
|
class Wave;
|
|
|
|
|
|
|
|
class Propagator {
|
|
|
|
public:
|
2022-04-21 17:05:26 +02:00
|
|
|
struct PropagatorEntry {
|
|
|
|
Vector<uint32_t> directions[4];
|
|
|
|
};
|
|
|
|
|
2022-04-20 03:05:34 +02:00
|
|
|
private:
|
2022-04-21 16:36:33 +02:00
|
|
|
const uint32_t patterns_size;
|
2022-04-20 03:05:34 +02:00
|
|
|
|
2022-04-21 17:45:30 +02:00
|
|
|
Vector<PropagatorEntry> propagator_state;
|
2022-04-20 03:05:34 +02:00
|
|
|
|
2022-04-21 16:43:04 +02:00
|
|
|
const uint32_t wave_width;
|
|
|
|
const uint32_t wave_height;
|
2022-04-20 03:05:34 +02:00
|
|
|
|
|
|
|
const bool periodic_output;
|
|
|
|
|
2022-04-21 17:17:50 +02:00
|
|
|
struct PropagatingEntry {
|
|
|
|
uint32_t data[3];
|
|
|
|
|
|
|
|
PropagatingEntry() {
|
|
|
|
for (int i = 0; i < 3; ++i) {
|
|
|
|
data[i] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
PropagatingEntry(uint32_t x, uint32_t y, uint32_t z) {
|
|
|
|
data[0] = x;
|
|
|
|
data[1] = y;
|
|
|
|
data[2] = z;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-04-20 03:24:50 +02:00
|
|
|
// All the tuples (y, x, pattern) that should be propagated.
|
|
|
|
// The tuple should be propagated when wave.get(y, x, pattern) is set to
|
|
|
|
// false.
|
2022-04-21 17:17:50 +02:00
|
|
|
Vector<PropagatingEntry> propagating;
|
2022-04-20 03:05:34 +02:00
|
|
|
|
2022-04-21 17:05:26 +02:00
|
|
|
struct CompatibilityEntry {
|
|
|
|
int direction[4];
|
|
|
|
|
|
|
|
CompatibilityEntry() {
|
|
|
|
for (int i = 0; i < 4; ++i) {
|
|
|
|
direction[i] = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-04-20 03:24:50 +02:00
|
|
|
// compatible.get(y, x, pattern)[direction] contains the number of patterns
|
|
|
|
// present in the wave that can be placed in the cell next to (y,x) in the
|
|
|
|
// opposite direction of direction without being in contradiction with pattern
|
|
|
|
// placed in (y,x). If wave.get(y, x, pattern) is set to false, then
|
|
|
|
// compatible.get(y, x, pattern) has every element negative or null
|
2022-04-21 17:05:26 +02:00
|
|
|
Array3D<CompatibilityEntry> compatible;
|
2022-04-20 03:05:34 +02:00
|
|
|
|
2022-04-20 03:24:50 +02:00
|
|
|
void init_compatible();
|
2022-04-20 03:05:34 +02:00
|
|
|
|
|
|
|
public:
|
2022-04-21 17:45:30 +02:00
|
|
|
Propagator(uint32_t wave_height, uint32_t wave_width, bool periodic_output, Vector<PropagatorEntry> propagator_state) :
|
2022-04-20 03:05:34 +02:00
|
|
|
patterns_size(propagator_state.size()),
|
|
|
|
propagator_state(propagator_state),
|
|
|
|
wave_width(wave_width),
|
|
|
|
wave_height(wave_height),
|
|
|
|
periodic_output(periodic_output),
|
|
|
|
compatible(wave_height, wave_width, patterns_size) {
|
|
|
|
init_compatible();
|
|
|
|
}
|
|
|
|
|
2022-04-21 16:43:04 +02:00
|
|
|
void add_to_propagator(uint32_t y, uint32_t x, uint32_t pattern) {
|
2022-04-20 03:05:34 +02:00
|
|
|
// All the direction are set to 0, since the pattern cannot be set in (y,x).
|
2022-04-21 17:05:26 +02:00
|
|
|
CompatibilityEntry temp;
|
2022-04-20 03:05:34 +02:00
|
|
|
compatible.get(y, x, pattern) = temp;
|
2022-04-21 17:05:26 +02:00
|
|
|
|
2022-04-21 17:17:50 +02:00
|
|
|
propagating.push_back(PropagatingEntry(y, x, pattern));
|
2022-04-20 03:05:34 +02:00
|
|
|
}
|
|
|
|
|
2022-04-20 03:24:50 +02:00
|
|
|
void propagate(Wave &wave);
|
2022-04-20 03:05:34 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif // FAST_WFC_PROPAGATOR_HPP_
|