forked from akinomyoga/cxxmatrix
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathconway.hpp
120 lines (106 loc) · 3.58 KB
/
conway.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
#ifndef cxxmatrix_conway_hpp
#define cxxmatrix_conway_hpp
#include "cxxmatrix.hpp"
#include <cstdint>
#include <algorithm>
#include <vector>
namespace cxxmatrix {
struct conway_t {
int width = 128, height = 128;
std::vector<byte> data1;
std::vector<byte> data2;
public:
void initialize() {
this->time = 1;
data1.resize(width * height);
data2.resize(width * height);
std::generate(data1.begin(), data1.end(), [] () { return util::rand() & 1; });
}
private:
int index(int x, int y) const {
x = util::mod(x, 2 * width);
if (x >= width) y = -1 - y, x -= width; // Klein boundary condition
//if (x >= width) x -= width; // T2 boundary condition
return util::mod(y, height) * width + x;
}
byte const& get1(int x, int y) const { return data1[index(x, y)]; }
byte const& get2(int x, int y) const { return data2[index(x, y)]; }
byte& get1(int x, int y) { return data1[index(x, y)]; }
byte& get2(int x, int y) { return data2[index(x, y)]; }
private:
std::uint32_t time = 1;
void create4x4() {
double const prob = (width / 100.0) * (height / 100.0);
if (util::rand() % std::min<int>(1, 100 / prob)== 0) {
int const x0 = util::rand() % width;
int const y0 = util::rand() % height;
std::uint32_t value = util::rand();
for (int a = 0; a < 4; a++) {
for (int b = 0; b < 4; b++) {
get1(x0 + a, y0 + b) = value & 1;
value >>= 1;
}
}
}
}
public:
void step(double time) {
if (time < this->time) return;
this->time++;
for (int y = 0; y < height; y++) {
for (int x = 0; x < width; x++) {
int count = 0;
if (get1(x + 1, y)) count++;
if (get1(x - 1, y)) count++;
if (get1(x, y + 1)) count++;
if (get1(x, y - 1)) count++;
if (get1(x + 1, y + 1)) count++;
if (get1(x + 1, y - 1)) count++;
if (get1(x - 1, y + 1)) count++;
if (get1(x - 1, y - 1)) count++;
get2(x, y) = count == 2 ? get1(x, y) : count == 3 ? 1 : 0;
}
}
data1.swap(data2);
create4x4();
}
private:
int origin_x, origin_y;
public:
void set_size(int cols, int rows) {
origin_x = cols / 2;
origin_y = rows / 2;
}
private:
static constexpr double xscale = 0.5 * 0.7;
static constexpr double yscale = -1.0;
double u_x, u_y, v_x, v_y;
public:
void set_transform(double scale, double theta) {
this->u_x = +scale * xscale * std::cos(theta);
this->u_y = -scale * yscale * std::sin(theta);
this->v_x = +scale * xscale * std::sin(theta);
this->v_y = +scale * yscale * std::cos(theta);
}
int get_pixel(int x, int y, double power) const {
double const u = 0.5 + u_x * (x - origin_x) + u_y * (y - origin_y);
double const v = 0.5 + v_x * (x - origin_x) + v_y * (y - origin_y);
if (get1(std::ceil(u), std::ceil(v))) return 1;
if (power >= 0.4) {
double const dx1A = 0.5, dy1A = +0.5;
double const dx1B = 0.5, dy1B = -0.5;
double const duA = dx1A * u_x + dy1A * u_y;
double const dvA = dy1A * v_x + dx1A * v_y;
double const duB = dx1B * u_x + dy1B * u_y;
double const dvB = dy1B * v_x + dx1B * v_y;
bool const sec = std::ceil(u + duA) != std::ceil(u - duA) ||
std::ceil(v + dvA) != std::ceil(v - dvA) ||
std::ceil(u + duB) != std::ceil(u - duB) ||
std::ceil(v + dvB) != std::ceil(v - dvB);
if (sec) return 2;
}
return 0;
}
};
}
#endif