-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathNonogram.cpp
118 lines (102 loc) · 3.41 KB
/
Nonogram.cpp
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
#include "Nonogram.h"
Nonogram::Nonogram(std::vector<std::vector<int>> &&rows, std::vector<std::vector<int>> &&cols, size_t width,
size_t height) {
rows_ = rows;
cols_ = cols;
width_ = width;
height_ = height;
}
void Nonogram::Solve() {
matrix_.assign(height_, std::vector<int>(width_, 0));
bool done_at_least_one_changing = true;
while (done_at_least_one_changing) {
done_at_least_one_changing = DoOneIteration();
}
}
void Nonogram::Increase(size_t coefficient) {
height_ *= coefficient;
width_ *= coefficient;
std::vector<std::vector<int>> matrix(height_, std::vector<int>(width_, 0));
for (size_t i = 0; i < height_; ++i) {
for (size_t j = 0; j < width_; ++j) {
matrix[i][j] = matrix_[i / coefficient][j / coefficient];
}
}
matrix_ = matrix;
}
size_t Nonogram::GetWidth() const {
return width_;
}
size_t Nonogram::GetHeight() const {
return height_;
}
int Nonogram::GetCell(size_t i, size_t j) const {
return matrix_[i][j];
}
bool Nonogram::CanMakeCorrect(std::vector<int> &correct, std::vector<int> ¤t) {
std::vector<size_t> black_pref(current.size() + 1, 0);
std::vector<size_t> white_pref(current.size() + 1, 0);
std::vector<int> dp(current.size(), 0);
std::vector<int> last_dp(current.size(), 0);
for (size_t i = 0; i < current.size(); ++i) {
black_pref[i + 1] = black_pref[i];
white_pref[i + 1] = white_pref[i];
black_pref[i + 1] += current[i] == 1;
white_pref[i + 1] += current[i] == -1;
}
for (size_t j = 0; j < correct.size(); ++j) {
for (size_t i = 0; i < current.size(); ++i) {
dp[i] = 0;
if (i > 0 && current[i] != 1 && dp[i - 1]) {
dp[i] = 1;
continue;
}
if (j == 0) {
dp[i] = (i + 1 >= correct[j]) && (white_pref[i + 1] - white_pref[i + 1 - correct[j]] == 0) &&
(black_pref[i + 1 - correct[j]] == 0);
} else {
dp[i] = (i >= correct[j] + 1) && (white_pref[i + 1] - white_pref[i + 1 - correct[j]] == 0) &&
(current[i - correct[j]] != 1) && (last_dp[i - correct[j] - 1]);
}
}
last_dp = dp;
}
return dp.back();
}
void Nonogram::TryPaintCell(size_t i, size_t j) {
std::vector<int> col = GetColumn(j);
std::vector<int> row = GetRow(i);
for (int cell_color = -1; cell_color <= 1; cell_color += 2) {
matrix_[i][j] = row[j] = col[i] = cell_color;
if (!CanMakeCorrect(rows_[i], row) || !CanMakeCorrect(cols_[j], col)) {
matrix_[i][j] = -cell_color;
return;
}
}
matrix_[i][j] = 0;
}
bool Nonogram::DoOneIteration() {
bool done_at_least_one_changing = false;
for (size_t i = 0; i < height_; ++i) {
for (size_t j = 0; j < width_; ++j) {
if (matrix_[i][j] != 0) {
continue;
}
TryPaintCell(i, j);
if (matrix_[i][j] != 0) {
done_at_least_one_changing = true;
}
}
}
return done_at_least_one_changing;
}
std::vector<int> Nonogram::GetColumn(size_t i) const {
std::vector<int> result(height_, 0);
for (size_t j = 0; j < height_; ++j) {
result[j] = (matrix_[j][i]);
}
return result;
}
std::vector<int> Nonogram::GetRow(size_t i) const {
return matrix_[i];
}