1#include "internal/graph.h"
3#include "internal/bezier.h"
4#include "internal/douglas_peucker.h"
5#include "internal/Pixel.h"
6#include "internal/shared_contours.h"
21 static_cast<float>(a.red),
static_cast<float>(a.green),
static_cast<float>(a.blue)};
23 static_cast<float>(b.red),
static_cast<float>(b.green),
static_cast<float>(b.blue)};
25 (af.red - bf.red) * (af.red - bf.red) + (af.green - bf.green) * (af.green - bf.green) +
26 (af.blue - bf.blue) * (af.blue - bf.blue)
36void Graph::hash_node_ids() {
37 for (int32_t i {0}; i < m_nodes->size(); i++) {
38 const int32_t key {m_nodes->at(i)->id()};
43bool Graph::all_areas_bigger_than(int32_t min_area) {
44 for (
auto& n : *m_nodes) {
45 if (n->area() < min_area) {
53bool Graph::add_edge(int32_t node_id1, int32_t node_id2) {
54 auto end_node_ids {m_node_ids.end()};
55 auto node1_it {m_node_ids.find(node_id1)};
56 auto node2_it {m_node_ids.find(node_id2)};
58 if (node1_it == end_node_ids || node2_it == end_node_ids) {
62 const int32_t idx1 {node1_it->second};
63 const int32_t idx2 {node2_it->second};
65 m_nodes->at(idx1)->add_edge(m_nodes->at(idx2));
66 m_nodes->at(idx2)->add_edge(m_nodes->at(idx1));
70bool Graph::merge_nodes(
const Node_ptr& node_to_keep,
const Node_ptr& node_to_remove) {
71 auto end_node_ids {m_node_ids.end()};
72 auto node1_it {m_node_ids.find(node_to_keep->id())};
73 auto node2_it {m_node_ids.find(node_to_remove->id())};
75 if (node1_it == end_node_ids || node2_it == end_node_ids) {
79 const int32_t idx_k {node1_it->second};
80 const int32_t idx_r {node2_it->second};
83 for (Node_ptr n : m_nodes->at(idx_r)->edges()) {
84 if (n->id() != node_to_keep->id()) {
86 n->remove_edge(node_to_remove);
87 n->add_edge(node_to_keep);
88 node_to_keep->add_edge(n);
92 node_to_keep->add_pixels(node_to_remove->get_pixels());
94 node_to_remove->clear_all();
99void Graph::clear_unconnected_nodes() {
100 std::vector<Node_ptr>& nodes {*m_nodes};
104 nodes.begin(), nodes.end(), [](
const Node_ptr& n) { return n->area() == 0; }
112void Graph::discover_edges(
113 const std::vector<int32_t>& region_labels,
const int32_t width,
const int32_t height
116 constexpr int8_t dirs[8][2] {{1, 0}, {-1, 0}, {0, 1}, {0, -1},
117 {1, 1}, {-1, -1}, {-1, 1}, {1, -1}};
121 for (int32_t y {0}; y < height; ++y) {
122 for (int32_t x {0}; x < width; ++x) {
123 const int32_t idx {y * width + x};
124 const int32_t rid {region_labels[idx]};
126 for (int32_t k {0}; k < 8; ++k) {
127 const int32_t nx {x + dirs[k][0]};
128 const int32_t ny {y + dirs[k][1]};
130 if (nx >= 0 && nx < width && ny >= 0 && ny < height) {
131 rneigh[k] = region_labels[ny * width + nx];
137 for (int32_t r : rneigh) {
146void Graph::process_overlapping_edges() {
148 std::vector<int32_t> label_map(m_width * m_height, 0);
150 for (
const Node_ptr& n : get_nodes()) {
154 for (
auto& [_, p] : n->get_pixels()) {
155 label_map[p.y * m_width + p.x] = n->id();
159 constexpr int8_t dirs[8][2] {{1, 0}, {-1, 0}, {0, 1}, {0, -1},
160 {1, 1}, {-1, -1}, {-1, 1}, {1, -1}};
163 for (
const Node_ptr& n : get_nodes()) {
166 int32_t val = n->id();
168 for (
auto& [_, p] : n->get_pixels()) {
173 for (
int k = 0; k < 8; ++k) {
174 int nx = x + dirs[k][0];
175 int ny = y + dirs[k][1];
178 if (nx < 0 || nx >= m_width || ny < 0 || ny >= m_height)
181 int32_t n_val = label_map[ny * m_width + nx];
184 if (n_val != 0 && n_val != val && val < n_val) {
185 bool is_too_thin =
false;
188 for (
int mk = 0; mk < 8; ++mk) {
189 int mx = nx + dirs[mk][0];
190 int my = ny + dirs[mk][1];
192 if (mx < 0 || mx >= m_width || my < 0 || my >= m_height)
195 int32_t m_val = label_map[my * m_width + mx];
197 if (m_val != 0 && m_val != val && m_val != n_val) {
205 Node_ptr neighbor_node =
206 m_nodes->at(m_node_ids[n_val]);
209 neighbor_node->add_edge_pixel(
XY {x, y});
213 n->add_edge_pixel(
XY {nx, ny});
222 std::vector<uint8_t> junction_map(w * h, 0);
228 int dx[] = {0, 1, 1, 1, 0, -1, -1, -1};
229 int dy[] = {-1, -1, 0, 1, 1, 1, 0, -1};
231 for (
int y = 0; y < h; ++y) {
232 for (
int x = 0; x < w; ++x) {
233 if (
getPixel(skel, w, h, x, y) == 0)
238 for (
int k = 0; k < 8; ++k) {
239 p[k] =
getPixel(skel, w, h, x + dx[k], y + dy[k]) ? 1 : 0;
245 for (
int k = 0; k < 8; ++k) {
246 if (p[k] == 0 && p[(k + 1) % 8] == 1)
252 for (
int k = 0; k < 8; ++k)
256 if (transitions >= 3) {
257 junction_map[y * w + x] = 1;
264void Graph::compute_contours() {
274 std::vector<int32_t> labels(
static_cast<size_t>(m_width) * m_height, -1);
275 for (
const Node_ptr& n : get_nodes()) {
278 for (
auto& p : n->get_pixels())
279 labels[static_cast<size_t>(p.position.y) * m_width + p.position.x] = n->id();
282 auto loops = build_shared_loops(labels, m_width, m_height, eps);
284 for (
const Node_ptr& n : get_nodes()) {
288 auto it = loops.find(n->id());
289 if (it == loops.end())
293 for (std::vector<QuadBezier>& curve : it->second) {
294 std::vector<Point> anchors;
295 anchors.reserve(curve.size() + 1);
297 anchors.push_back(q.p0);
299 anchors.push_back(curve.back().p2);
300 n->m_contours.contours.push_back(std::move(anchors));
301 n->m_contours.curves.push_back(std::move(curve));
302 n->m_contours.colors.push_back(col);
303 n->m_contours.hierarchy.push_back({-1, -1, -1, -1});
304 n->m_contours.is_hole.push_back(
false);
313void dt_1d(
const std::vector<float>& f, std::vector<float>& d,
int n) {
314 constexpr float INF = 1e20f;
315 std::vector<int> v(n);
316 std::vector<float> z(n + 1);
321 for (
int q = 1; q < n; ++q) {
324 s = ((f[q] +
static_cast<float>(q) * q) - (f[v[k]] +
static_cast<float>(v[k]) * v[k])) /
325 (2.0f *
static_cast<float>(q - v[k]));
326 if (s <= z[k] && k > 0) {
338 for (
int q = 0; q < n; ++q) {
339 while (z[k + 1] <
static_cast<float>(q))
341 const float dq =
static_cast<float>(q - v[k]);
342 d[q] = dq * dq + f[v[k]];
351float max_inscribed_radius(
const Node_ptr& n) {
352 std::vector<uint8_t> mask;
353 const std::array<int, 4> xywh = n->create_binary_image(mask);
354 const int w = xywh[2];
355 const int h = xywh[3];
356 if (w <= 0 || h <= 0)
361 const int pw = w + 2;
362 const int ph = h + 2;
363 constexpr float INF = 1e20f;
365 std::vector<float> grid(
static_cast<size_t>(pw) * ph);
366 for (
int y = 0; y < ph; ++y) {
367 for (
int x = 0; x < pw; ++x) {
368 const bool inside = x >= 1 && x <= w && y >= 1 && y <= h &&
369 mask[
static_cast<size_t>(y - 1) * w + (x - 1)];
370 grid[
static_cast<size_t>(y) * pw + x] = inside ? INF : 0.0f;
375 std::vector<float> in, out(std::max(pw, ph));
377 for (
int x = 0; x < pw; ++x) {
378 for (
int y = 0; y < ph; ++y)
379 in[y] = grid[
static_cast<size_t>(y) * pw + x];
381 for (
int y = 0; y < ph; ++y)
382 grid[
static_cast<size_t>(y) * pw + x] = out[y];
386 for (
int y = 0; y < ph; ++y) {
387 for (
int x = 0; x < pw; ++x)
388 in[x] = grid[
static_cast<size_t>(y) * pw + x];
390 for (
int x = 0; x < pw; ++x)
394 return std::sqrt(max_d2);
399void Graph::merge_small_area_nodes(
const int32_t min_area,
const int32_t min_thickness) {
404 bool merged_any =
true;
408 for (
const Node_ptr& n : get_nodes()) {
412 bool needs_merge = n->area() <
static_cast<size_t>(min_area);
413 if (!needs_merge && min_thickness > 0) {
415 needs_merge = 2.0f * max_inscribed_radius(n) <
static_cast<float>(min_thickness);
422 Node_ptr best_neighbor =
nullptr;
423 float best_score = std::numeric_limits<float>::max();
424 for (
const Node_ptr& ne : n->edges()) {
425 if (ne->area() > 0) {
427 float score =
static_cast<float>(ne->area()) + 10.f * cdist;
428 if (score < best_score) {
436 if (!best_neighbor) {
440 if (best_neighbor->area() >= n->area()) {
441 merge_nodes(best_neighbor, n);
443 merge_nodes(n, best_neighbor);
448 clear_unconnected_nodes();
uint8_t getPixel(const std::vector< uint8_t > &img, int w, int h, int x, int y)
std::vector< uint8_t > analyzeJunctions(const std::vector< uint8_t > &skel, int w, int h)