mirror of
https://github.com/Relintai/pandemonium_engine.git
synced 2025-01-25 18:39:18 +01:00
80 lines
2.5 KiB
C++
80 lines
2.5 KiB
C++
#ifndef FAST_WFC_WAVE_HPP_
|
|
#define FAST_WFC_WAVE_HPP_
|
|
|
|
#include "array_2d.h"
|
|
#include <random>
|
|
#include <vector>
|
|
|
|
// Struct containing the values needed to compute the entropy of all the cells.
|
|
// This struct is updated every time the wave is changed.
|
|
// p'(pattern) is equal to patterns_frequencies[pattern] if wave.get(cell,
|
|
// pattern) is set to true, otherwise 0.
|
|
struct EntropyMemoisation {
|
|
std::vector<double> plogp_sum; // The sum of p'(pattern)// log(p'(pattern)).
|
|
std::vector<double> sum; // The sum of p'(pattern).
|
|
std::vector<double> log_sum; // The log of sum.
|
|
std::vector<unsigned> nb_patterns; // The number of patterns present
|
|
std::vector<double> entropy; // The entropy of the cell.
|
|
};
|
|
|
|
// Contains the pattern possibilities in every cell.
|
|
// Also contains information about cell entropy.
|
|
class Wave {
|
|
private:
|
|
// The patterns frequencies p given to wfc.
|
|
const std::vector<double> patterns_frequencies;
|
|
|
|
// The precomputation of p * log(p).
|
|
const std::vector<double> plogp_patterns_frequencies;
|
|
|
|
// The precomputation of min (p * log(p)) / 2.
|
|
// This is used to define the maximum value of the noise.
|
|
const double min_abs_half_plogp;
|
|
|
|
EntropyMemoisation memoisation;
|
|
|
|
// This value is set to true if there is a contradiction in the wave (all elements set to false in a cell).
|
|
bool is_impossible;
|
|
|
|
// The number of distinct patterns.
|
|
const size_t nb_patterns;
|
|
|
|
// The actual wave. data.get(index, pattern) is equal to 0 if the pattern can
|
|
// be placed in the cell index.
|
|
Array2D<uint8_t> data;
|
|
|
|
public:
|
|
// The size of the wave.
|
|
const unsigned width;
|
|
const unsigned height;
|
|
const unsigned size;
|
|
|
|
// Initialize the wave with every cell being able to have every pattern.
|
|
Wave(unsigned height, unsigned width, const std::vector<double> &patterns_frequencies);
|
|
|
|
// Return true if pattern can be placed in cell index.
|
|
bool get(unsigned index, unsigned pattern) const {
|
|
return data.get(index, pattern);
|
|
}
|
|
|
|
// Return true if pattern can be placed in cell (i,j)
|
|
bool get(unsigned i, unsigned j, unsigned pattern) const {
|
|
return get(i * width + j, pattern);
|
|
}
|
|
|
|
// Set the value of pattern in cell index.
|
|
void set(unsigned index, unsigned pattern, bool value);
|
|
|
|
// Set the value of pattern in cell (i,j).
|
|
void set(unsigned i, unsigned j, unsigned pattern, bool value) {
|
|
set(i * width + j, pattern, value);
|
|
}
|
|
|
|
// Return the index of the cell with lowest entropy different of 0.
|
|
// If there is a contradiction in the wave, return -2.
|
|
// If every cell is decided, return -1.
|
|
int get_min_entropy(std::minstd_rand &gen) const;
|
|
};
|
|
|
|
#endif // FAST_WFC_WAVE_HPP_
|