Img2Num C++ (Internal Developer Docs) dev
API Documentation
Loading...
Searching...
No Matches
Graph Class Reference
+ Collaboration diagram for Graph:

Public Member Functions

 Graph (std::unique_ptr< std::vector< Node_ptr > > &nodes, int width, int height)
 
bool add_edge (int32_t node_id1, int32_t node_id2)
 
bool merge_nodes (const Node_ptr &node_to_keep, const Node_ptr &node_to_remove)
 
void clear_unconnected_nodes ()
 
const std::vector< Node_ptr > & get_nodes () const
 
bool all_areas_bigger_than (int32_t min_area)
 
const size_t size ()
 
void discover_edges (const std::vector< int32_t > &region_labels, const int32_t width, const int32_t height)
 
void merge_small_area_nodes (const int32_t min_area, const int32_t min_thickness=0)
 
void compute_contours ()
 

Protected Member Functions

void hash_node_ids (void)
 
void process_overlapping_edges ()
 
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)
 

Protected Attributes

int m_width
 
int m_height
 
std::unique_ptr< std::vector< Node_ptr > > m_nodes
 
std::unordered_map< int32_t, int32_t > m_node_ids
 

Detailed Description

Definition at line 33 of file graph.h.

Constructor & Destructor Documentation

◆ Graph()

Graph::Graph ( std::unique_ptr< std::vector< Node_ptr > > &  nodes,
int  width,
int  height 
)
inline

Definition at line 73 of file graph.h.

74 : m_nodes(std::move(nodes))
75 , m_width(width)
76 , m_height(height) {
77 hash_node_ids();
78 }

◆ ~Graph()

Graph::~Graph ( )
inline

Definition at line 80 of file graph.h.

80 {
81 // Break the circular references so the shared_ptrs can reach 0
82 for (auto& node : *m_nodes) {
83 node->clear_all();
84 }
85 }

Member Function Documentation

◆ add_edge()

bool Graph::add_edge ( int32_t  node_id1,
int32_t  node_id2 
)

Definition at line 53 of file graph.cpp.

53 {
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)};
57
58 if (node1_it == end_node_ids || node2_it == end_node_ids) {
59 return false;
60 }
61
62 const int32_t idx1 {node1_it->second};
63 const int32_t idx2 {node2_it->second};
64
65 m_nodes->at(idx1)->add_edge(m_nodes->at(idx2));
66 m_nodes->at(idx2)->add_edge(m_nodes->at(idx1));
67 return true;
68}

◆ all_areas_bigger_than()

bool Graph::all_areas_bigger_than ( int32_t  min_area)

Definition at line 43 of file graph.cpp.

43 {
44 for (auto& n : *m_nodes) {
45 if (n->area() < min_area) {
46 return false;
47 }
48 }
49
50 return true;
51}

◆ analyzeJunctions()

std::vector< uint8_t > Graph::analyzeJunctions ( const std::vector< uint8_t > &  skel,
int  w,
int  h 
)
protected

@brief Analyzes a skeleton image to detect junction points using 8-neighbor crossing-number.

Scans the skeleton image and marks pixels as junctions where three or more branches meet, using the crossing-number method (counts 0→1 transitions in the 8-neighbor ring).

@param skel Binary skeleton image (nonzero = skeleton pixel) @param w Image width @param h Image height @return Junction mask with nonzero entries marking junction pixels

Definition at line 221 of file graph.cpp.

221 {
222 std::vector<uint8_t> junction_map(w * h, 0);
223
224 // 8-Neighbor Order (Clockwise)
225 // P9 P2 P3
226 // P8 P1 P4
227 // P7 P6 P5
228 int dx[] = {0, 1, 1, 1, 0, -1, -1, -1};
229 int dy[] = {-1, -1, 0, 1, 1, 1, 0, -1};
230
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)
234 continue;
235
236 // 1. Get Neighbors in Circular Order
237 int p[8];
238 for (int k = 0; k < 8; ++k) {
239 p[k] = getPixel(skel, w, h, x + dx[k], y + dy[k]) ? 1 : 0;
240 }
241
242 // 2. Count Transitions (0 -> 1)
243 // This is the Crossing Number / 2
244 int transitions = 0;
245 for (int k = 0; k < 8; ++k) {
246 if (p[k] == 0 && p[(k + 1) % 8] == 1)
247 transitions++;
248 }
249
250 // 3. Count Total Neighbors (for Endpoint check)
251 int neighbors = 0;
252 for (int k = 0; k < 8; ++k)
253 neighbors += p[k];
254
255 // 4. Classify
256 if (transitions >= 3) {
257 junction_map[y * w + x] = 1;
258 }
259 }
260 }
261 return junction_map;
262}
uint8_t getPixel(const std::vector< uint8_t > &img, int w, int h, int x, int y)
Definition graph.h:52

References getPixel().

+ Here is the call graph for this function:

◆ clear_unconnected_nodes()

void Graph::clear_unconnected_nodes ( )

Definition at line 99 of file graph.cpp.

99 {
100 std::vector<Node_ptr>& nodes {*m_nodes};
101
102 nodes.erase(
103 std::remove_if(
104 nodes.begin(), nodes.end(), [](const Node_ptr& n) { return n->area() == 0; }
105 ),
106 nodes.end()
107 );
108
109 hash_node_ids();
110}

◆ compute_contours()

void Graph::compute_contours ( )

Definition at line 264 of file graph.cpp.

264 {
265
266 /*
267 Shared-edge mode: build region boundaries on the crack grid so
268 neighbouring contours are exactly coincident along shared edges -- no
269 overlap band, no gaps
270 */
271
272 float eps = 0.25f;
273
274 std::vector<int32_t> labels(static_cast<size_t>(m_width) * m_height, -1);
275 for (const Node_ptr& n : get_nodes()) {
276 if (n->area() == 0)
277 continue;
278 for (auto& p : n->get_pixels())
279 labels[static_cast<size_t>(p.position.y) * m_width + p.position.x] = n->id();
280 }
281
282 auto loops = build_shared_loops(labels, m_width, m_height, eps);
283
284 for (const Node_ptr& n : get_nodes()) {
285 if (n->area() == 0)
286 continue;
287 n->clear_contour();
288 auto it = loops.find(n->id());
289 if (it == loops.end())
290 continue;
291 ImageLib::RGBPixel<uint8_t> c = n->color();
292 ImageLib::RGBAPixel<uint8_t> col {c.red, c.green, c.blue, 255};
293 for (std::vector<QuadBezier>& curve : it->second) {
294 std::vector<Point> anchors; // keep contours[] parallel to curves[]
295 anchors.reserve(curve.size() + 1);
296 for (const QuadBezier& q : curve)
297 anchors.push_back(q.p0);
298 if (!curve.empty())
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);
305 }
306 }
307}

◆ discover_edges()

void Graph::discover_edges ( const std::vector< int32_t > &  region_labels,
const int32_t  width,
const int32_t  height 
)

Definition at line 112 of file graph.cpp.

114 {
115 // Moore 8-connected neighbourhood
116 constexpr int8_t dirs[8][2] {{1, 0}, {-1, 0}, {0, 1}, {0, -1},
117 {1, 1}, {-1, -1}, {-1, 1}, {1, -1}};
118
119 int32_t rneigh[8];
120
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]};
125
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]};
129
130 if (nx >= 0 && nx < width && ny >= 0 && ny < height) {
131 rneigh[k] = region_labels[ny * width + nx];
132 } else {
133 rneigh[k] = rid; // ignore out-of-bounds
134 }
135 }
136
137 for (int32_t r : rneigh) {
138 if (r != rid) {
139 add_edge(rid, r);
140 }
141 }
142 }
143 }
144}

◆ get_nodes()

const std::vector< Node_ptr > & Graph::get_nodes ( ) const
inline

Definition at line 92 of file graph.h.

92 {
93 return *m_nodes;
94 }

◆ getPixel()

uint8_t Graph::getPixel ( const std::vector< uint8_t > &  img,
int  w,
int  h,
int  x,
int  y 
)
inlineprotected

@brief Safely retrieves a pixel value from a binary image with bounds checking.

@param img Binary image buffer @param w Image width @param h Image height @param x Pixel x-coordinate @param y Pixel y-coordinate @return Pixel value at (x, y), or 0 if out of bounds

Definition at line 52 of file graph.h.

52 {
53 if (x < 0 || x >= w || y < 0 || y >= h)
54 return 0; // Boundary check
55 return img[y * w + x];
56 }

Referenced by analyzeJunctions().

+ Here is the caller graph for this function:

◆ hash_node_ids()

void Graph::hash_node_ids ( void  )
protected

Definition at line 36 of file graph.cpp.

36 {
37 for (int32_t i {0}; i < m_nodes->size(); i++) {
38 const int32_t key {m_nodes->at(i)->id()};
39 m_node_ids[key] = i;
40 }
41}

◆ merge_nodes()

bool Graph::merge_nodes ( const Node_ptr &  node_to_keep,
const Node_ptr &  node_to_remove 
)

Definition at line 70 of file graph.cpp.

70 {
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())};
74
75 if (node1_it == end_node_ids || node2_it == end_node_ids) {
76 return false;
77 }
78
79 const int32_t idx_k {node1_it->second};
80 const int32_t idx_r {node2_it->second};
81
82 // transfer edges from node_to_remove to node_to_keep
83 for (Node_ptr n : m_nodes->at(idx_r)->edges()) {
84 if (n->id() != node_to_keep->id()) {
85 // prevents self referencing
86 n->remove_edge(node_to_remove);
87 n->add_edge(node_to_keep);
88 node_to_keep->add_edge(n);
89 }
90 }
91
92 node_to_keep->add_pixels(node_to_remove->get_pixels());
93
94 node_to_remove->clear_all();
95
96 return true;
97}

◆ merge_small_area_nodes()

void Graph::merge_small_area_nodes ( const int32_t  min_area,
const int32_t  min_thickness = 0 
)

Definition at line 399 of file graph.cpp.

399 {
400 // Keep merging while any pass still makes progress. Using "did this pass
401 // merge anything?" as the loop guard (instead of re-testing every node)
402 // also avoids spinning forever on a node that is too small/thin but has no
403 // valid neighbour to merge into.
404 bool merged_any = true;
405 while (merged_any) {
406 merged_any = false;
407
408 for (const Node_ptr& n : get_nodes()) {
409 if (n->area() == 0)
410 continue;
411
412 bool needs_merge = n->area() < static_cast<size_t>(min_area);
413 if (!needs_merge && min_thickness > 0) {
414 // too thin == no inscribed disk of radius min_thickness/2 fits.
415 needs_merge = 2.0f * max_inscribed_radius(n) < static_cast<float>(min_thickness);
416 }
417 if (!needs_merge)
418 continue;
419
420 ImageLib::RGBPixel<uint8_t> col = n->color();
421
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) {
426 float cdist = ImageLib::RGBPixel<uint8_t>::colorDistance(ne->color(), col);
427 float score = static_cast<float>(ne->area()) + 10.f * cdist;
428 if (score < best_score) {
429 best_score = score;
430 best_neighbor = ne;
431 }
432 }
433 }
434
435 // no valid neighbor found, skip this node
436 if (!best_neighbor) {
437 continue;
438 }
439
440 if (best_neighbor->area() >= n->area()) {
441 merge_nodes(best_neighbor, n);
442 } else {
443 merge_nodes(n, best_neighbor);
444 }
445 merged_any = true;
446 }
447
448 clear_unconnected_nodes();
449 }
450}

◆ process_overlapping_edges()

void Graph::process_overlapping_edges ( )
protected

Definition at line 146 of file graph.cpp.

146 {
147 // 1. Build the Global Label Map ONCE (0 = background, else = node->id())
148 std::vector<int32_t> label_map(m_width * m_height, 0);
149
150 for (const Node_ptr& n : get_nodes()) {
151 if (n->area() == 0)
152 continue;
153
154 for (auto& [_, p] : n->get_pixels()) {
155 label_map[p.y * m_width + p.x] = n->id();
156 }
157 }
158
159 constexpr int8_t dirs[8][2] {{1, 0}, {-1, 0}, {0, 1}, {0, -1},
160 {1, 1}, {-1, -1}, {-1, 1}, {1, -1}};
161
162 // 2. Iterate directly over the pixels of each node
163 for (const Node_ptr& n : get_nodes()) {
164 if (n->area() == 0)
165 continue;
166 int32_t val = n->id();
167
168 for (auto& [_, p] : n->get_pixels()) {
169 int x = p.x;
170 int y = p.y;
171
172 // Check 8 neighbors in the global map
173 for (int k = 0; k < 8; ++k) {
174 int nx = x + dirs[k][0];
175 int ny = y + dirs[k][1];
176
177 // Fast boundary check (replaces std::clamp)
178 if (nx < 0 || nx >= m_width || ny < 0 || ny >= m_height)
179 continue;
180
181 int32_t n_val = label_map[ny * m_width + nx];
182
183 // Is it a neighbor? AND have we not processed this pairing yet?
184 if (n_val != 0 && n_val != val && val < n_val) {
185 bool is_too_thin = false;
186
187 // Check around the neighbor pixel for a 3rd region (pinching)
188 for (int mk = 0; mk < 8; ++mk) {
189 int mx = nx + dirs[mk][0];
190 int my = ny + dirs[mk][1];
191
192 if (mx < 0 || mx >= m_width || my < 0 || my >= m_height)
193 continue;
194
195 int32_t m_val = label_map[my * m_width + mx];
196
197 if (m_val != 0 && m_val != val && m_val != n_val) {
198 is_too_thin = true;
199 break; // CRITICAL: Stop checking immediately once proven thin!
200 }
201 }
202
203 if (is_too_thin) {
204 // Give our pixel to the neighbor
205 Node_ptr neighbor_node =
206 m_nodes->at(m_node_ids[n_val]); // get_node_by_id(n_val); // Assuming
207 // you have this lookup
208 if (neighbor_node) {
209 neighbor_node->add_edge_pixel(XY {x, y});
210 }
211 } else {
212 // Take the neighbor's pixel
213 n->add_edge_pixel(XY {nx, ny});
214 }
215 }
216 }
217 }
218 }
219}
Definition node.h:38

◆ size()

const size_t Graph::size ( )
inline

Definition at line 97 of file graph.h.

97 {
98 return m_nodes->size();
99 }

Member Data Documentation

◆ m_height

int Graph::m_height
protected

Definition at line 35 of file graph.h.

◆ m_node_ids

std::unordered_map<int32_t, int32_t> Graph::m_node_ids
protected

Definition at line 37 of file graph.h.

◆ m_nodes

std::unique_ptr<std::vector<Node_ptr> > Graph::m_nodes
protected

Definition at line 36 of file graph.h.

◆ m_width

int Graph::m_width
protected

Definition at line 35 of file graph.h.


The documentation for this class was generated from the following files: