Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
Loading...
Searching...
No Matches
Loop.hpp
Go to the documentation of this file.
1/*
2 * Author: Sven Gothel <sgothel@jausoft.com> (C++, Java) and Rami Santina (Java)
3 * Copyright Gothel Software e.K. and the authors
4 *
5 * SPDX-License-Identifier: MIT
6 *
7 * This Source Code Form is subject to the terms of the MIT License
8 * If a copy of the MIT was not distributed with this file,
9 * you can obtain one at https://opensource.org/license/mit/.
10 */
11#ifndef JAU_GAMP_GRAPH_TESS_IMPL_LOOP_HPP_
12#define JAU_GAMP_GRAPH_TESS_IMPL_LOOP_HPP_
13
14#include <limits>
15
16#include <jau/cpp_lang_util.hpp>
17#include <jau/debug.hpp>
18#include <jau/int_types.hpp>
21#include <jau/string_util.hpp>
22
23#include <gamp/GampTypes.hpp>
24#include <gamp/graph/Graph.hpp>
29
31
32 using namespace jau::math;
33 using namespace jau::math::geom;
34 using namespace gamp::graph;
35 using namespace gamp::graph::tess;
36
37 /** \addtogroup Gamp_GraphImpl
38 *
39 * @{
40 */
41
42 class Loop;
43 typedef std::shared_ptr<Loop> LoopRef;
44 typedef std::vector<LoopRef> LoopRefList;
45
46 class Loop {
47 public:
48 constexpr static bool FixedWindingRule = true;
49
50 private:
51 std::vector<HEdgeRef> m_hedges;
52 AABBox3f m_box;
53 GraphOutlineRef m_initialOutline;
54 bool m_isComplex;
55 GraphOutlineRefList m_outlines;
56 HEdgePtr m_root;
57
58 struct Private{ explicit Private() = default; };
59
60 HEdgePtr createHEdge(const GraphVertexRef& vert, int type) {
61 HEdgeRef owner = HEdge::create(vert, type);
62 HEdgePtr ptr = owner.get();
63 m_hedges.push_back(std::move(owner));
64 return ptr;
65 }
66
67 public:
68 Loop(Private, GraphOutlineRef polyline, int edgeType, bool isComplex)
69 : m_hedges(), m_box(),
70 m_initialOutline(std::move(polyline)),
71 m_isComplex(isComplex),
72 m_outlines(),
73 m_root(initFromPolyline(m_initialOutline, edgeType))
74 { }
75
76 private:
77 /**
78 * Create a connected list of half edges (loop)
79 * from the boundary profile
80 * @param edgeType either {@link HEdge#BOUNDARY} requiring {@link Winding#CCW} or {@link HEdge#HOLE} using {@link Winding#CW} or even {@link Winding#CCW}
81 */
82 HEdgePtr initFromPolyline(const GraphOutlineRef& outline, int edgeType) {
83 m_outlines.push_back(outline);
84 const GraphVertexRefList& vertices = outline->graphPoints();
85
86 if(vertices.size() < 3) {
87 ERR_PRINT2("Graph: Loop.initFromPolyline: GraphOutline's vertices < 3: %zu", vertices.size() );
88 if( Graph::DEBUG_MODE ) {
89 jau::print_backtrace(true /* skip_anon_frames */, 4 /* max_frames */);
90 }
91 return nullptr;
92 }
93 Winding edgeWinding = HEdge::BOUNDARY == edgeType ? Winding::CCW : Winding::CW;
94 Winding winding = FixedWindingRule ? edgeWinding : outline->outline().getWinding();
95
96 if( HEdge::BOUNDARY == edgeType && Winding::CCW != winding ) {
97 // XXXX
98 WARN_PRINT("Loop.init.xx.01: BOUNDARY req CCW but has %s", to_string(winding).c_str());
99 // outline->outline().print(stderr);
100 jau::print_backtrace(true /* skip_anon_frames */, 4 /* max_frames */);
101 }
102 HEdgePtr firstEdge = nullptr;
103 HEdgePtr lastEdge = nullptr;
104
105 if( winding == edgeWinding || HEdge::BOUNDARY == edgeType ) {
106 // Correct Winding or skipped CW -> CCW (no inversion possible here, too late)
107 const size_t maxVertIdx = vertices.size() - 1;
108 for(size_t index = 0; index <= maxVertIdx; ++index) {
109 GraphVertexRef v1 = vertices[index];
110 m_box.resize(v1->coord());
111
112 HEdgePtr edge = createHEdge(v1, edgeType);
113 v1->addEdge(edge);
114 if(lastEdge) {
115 lastEdge->setNext(edge);
116 edge->setPrev(lastEdge);
117 // jau::PLAIN_PRINT(true, "initFromPoly[%zu]: lastEdge %s -> new edge %s", index, lastEdge->toString().c_str(), edge->toString().c_str());
118 } else {
119 firstEdge = edge;
120 // jau::PLAIN_PRINT(true, "initFromPoly[%zu]: new firstEdge %s", index, firstEdge->toString().c_str());
121 }
122 if(index == maxVertIdx ) {
123 edge->setNext(firstEdge);
124 firstEdge->setPrev(edge);
125 // jau::PLAIN_PRINT(true, "initFromPoly[%zu]: last new edge %s -> firstEdge %s", index, edge->toString().c_str(), firstEdge->toString().c_str());
126 }
127 lastEdge = edge;
128 }
129 } else { // if( winding == Winding::CW ) {
130 // CCW <-> CW
131 for(size_t index = vertices.size(); index-- > 0;) {
132 GraphVertexRef v1 = vertices[index];
133 m_box.resize(v1->coord());
134
135 HEdgePtr edge = createHEdge(v1, edgeType);
136 v1->addEdge(edge);
137 if(lastEdge) {
138 lastEdge->setNext(edge);
139 edge->setPrev(lastEdge);
140 } else {
141 firstEdge = edge;
142 }
143
144 if (index == 0) {
145 edge->setNext(firstEdge);
146 firstEdge->setPrev(edge);
147 }
148 lastEdge = edge;
149 }
150 }
151 // jau::PLAIN_PRINT(true, "initFromPoly.XX: firstEdge %s", firstEdge->toString().c_str());
152 // firstEdge->printChain();
153 return firstEdge;
154 }
155
156 /**
157 * Locates the vertex and update the loops root
158 * to have (root + vertex) as closest pair
159 * @param polyline the control polyline to search for closestvertices in CW
160 * @return the vertex that is closest to the newly set root Hedge.
161 */
162 GraphVertexRef locateClosestVertex(const GraphOutlineRef& polyline) {
163 float minDistance = std::numeric_limits<float>::max();
164 const GraphVertexRefList& initVertices = m_initialOutline->graphPoints();
165 if( initVertices.size() < 2 ) {
166 return nullptr;
167 }
168 const GraphVertexRefList& vertices = polyline->graphPoints();
169 HEdgePtr closestE = nullptr;
170 GraphVertexRef closestV;
171
172 size_t initSz = initVertices.size();
173 GraphVertexRef v0 = initVertices[0];
174 for(size_t i=1; i< initSz; ++i){
175 GraphVertexRef v1 = initVertices[i];
176 for(size_t pos=0; pos<vertices.size(); ++pos) {
177 GraphVertexRef cand = vertices[pos];
178 float distance = v0->coord().dist(cand->coord());
179 if(distance < minDistance){
180 bool inside = false;
181 for (const GraphVertexRef& vert : vertices){
182 if( !( vert == v0 || vert == v1 || vert == cand) ) {
183 inside = isInCircle2D(v0->coord(), v1->coord(), cand->coord(), vert->coord());
184 if(inside){
185 break;
186 }
187 }
188 }
189 if(!inside){
190 closestV = cand;
191 minDistance = distance;
192 closestE = v0->findBoundEdge();
193 }
194 }
195 }
196 v0 = v1;
197 }
198 if(closestE){
199 m_root = closestE;
200 }
201 return closestV;
202 }
203
204 bool intersectsOutline(const Vertex& a1, const Vertex& a2, const Vertex& b) {
205 for(const GraphOutlineRef& outline : m_outlines) {
206 const GraphVertexRefList& vertices = outline->graphPoints();
207 size_t sz = vertices.size();
208 if( sz >= 2 ) {
209 const Vertex* v0 = &vertices[0]->vertex();
210 for(size_t i=1; i< sz; i++){
211 const Vertex& v1 = vertices[i]->vertex();
212 if( *v0 != b && v1 != b ) {
213 if( *v0 != a1 && v1 != a1 &&
214 testSeg2SegIntersection2D(a1.coord(), b.coord(), v0->coord(), v1.coord()) ) {
215 return true;
216 }
217 if( *v0 != a2 && v1 != a2 &&
218 testSeg2SegIntersection2D(a2.coord(), b.coord(), v0->coord(), v1.coord()) ) {
219 return true;
220 }
221 }
222 v0 = &v1;
223 }
224 }
225 }
226 return false;
227 }
228 HEdgePtr isValidNeighbor(HEdgePtr candEdge, bool delaunay) {
229 const GraphVertexRef& rootGPoint = m_root->getGraphPoint();
230 const GraphVertexRef& nextGPoint = m_root->getNext()->getGraphPoint();
231 const Vertex& rootPoint = rootGPoint->vertex();
232 const Vertex& nextPoint = nextGPoint->vertex();
233 const Vertex& candPoint = candEdge->getGraphPoint()->vertex();
234 if( !is2DCCW(rootPoint.coord(), nextPoint.coord(), candPoint.coord()) ||
235 ( m_isComplex && intersectsOutline(rootPoint, nextPoint, candPoint) ) ) {
236 return nullptr;
237 }
238 if( !delaunay ) {
239 return candEdge;
240 }
241 HEdgePtr e = candEdge->getNext();
242 while (e != candEdge){
243 const GraphVertexRef& egp = e->getGraphPoint();
244 const Vertex& ep = egp->vertex();
245 if(egp != rootGPoint &&
246 egp != nextGPoint &&
247 ep != candPoint )
248 {
249 if( isInCircle2D(rootPoint.coord(), nextPoint.coord(), candPoint.coord(), ep.coord()) ) {
250 return nullptr;
251 }
252 }
253 e = e->getNext();
254 }
255 return candEdge;
256 }
257
258 /** Create a triangle from the param vertices only if
259 * the triangle is valid. IE not outside region.
260 * @param v1 vertex 1
261 * @param v2 vertex 2
262 * @param v3 vertex 3
263 * @param root and edge of this triangle
264 * @return the triangle iff it satisfies, null otherwise
265 */
266 TriangleRef createTriangle(Vertex& v1, Vertex& v2, Vertex& v3, HEdgePtr rootT) {
267 return Triangle::create(v1, v2, v3, checkVerticesBoundary(rootT));
268 }
269
270 Triangle::tribit_t checkVerticesBoundary(HEdgePtr rootT) {
271 Triangle::tribit_t boundary;
272 boundary.put(0, rootT->getGraphPoint()->isBoundaryContained());
273 boundary.put(1, rootT->getNext()->getGraphPoint()->isBoundaryContained());
274 boundary.put(2, rootT->getNext()->getNext()->getGraphPoint()->isBoundaryContained());
275 return boundary;
276 }
277
278 public:
279 static LoopRef createBoundary(const GraphOutlineRef& polyline, bool isConvex) {
280 LoopRef res = std::make_shared<Loop>(Private(), polyline, HEdge::BOUNDARY, isConvex);
281 if( !res->m_root ) {
282 return nullptr;
283 }
284 return res;
285 }
286
287 const HEdgePtr& getHEdge() const noexcept { return m_root; }
288
289 bool isSimplex() const noexcept {
290 return (m_root->getNext()->getNext()->getNext() == m_root);
291 }
292
293 TriangleRef cut(bool delaunay) {
294 HEdgePtr next1 = m_root->getNext();
295 if( isSimplex() ){
296 Vertex& rootPoint = m_root->getGraphPoint()->vertex();
297 Vertex& nextPoint = next1->getGraphPoint()->vertex();
298 Vertex& candPoint = next1->getNext()->getGraphPoint()->vertex();
299 if( m_isComplex && intersectsOutline(rootPoint, nextPoint, candPoint) ) {
300 return nullptr;
301 }
302 return Triangle::create(rootPoint, nextPoint, candPoint, checkVerticesBoundary(m_root));
303 }
304 HEdgePtr prev = m_root->getPrev();
305 HEdgePtr next2 = isValidNeighbor(next1->getNext(), delaunay);
306 if(!next2) {
307 m_root = m_root->getNext();
308 return nullptr;
309 }
310 GraphVertexRef v1 = m_root->getGraphPoint();
311 GraphVertexRef v2 = next1->getGraphPoint();
312 GraphVertexRef v3 = next2->getGraphPoint();
313
314 HEdgePtr v3Edge = createHEdge(v3, HEdge::INNER);
315
316 HEdge::connect(v3Edge, m_root);
317 HEdge::connect(next1, v3Edge);
318
319 HEdgePtr v3EdgeSib = v3Edge->getSibling();
320 if(!v3EdgeSib){
321 v3EdgeSib = createHEdge(v3Edge->getNext()->getGraphPoint(), HEdge::INNER);
322 HEdge::makeSiblings(v3Edge, v3EdgeSib);
323 }
324 HEdge::connect(prev, v3EdgeSib);
325 HEdge::connect(v3EdgeSib, next2);
326
327 TriangleRef t = createTriangle(v1->vertex(), v2->vertex(), v3->vertex(), m_root);
328 m_root = next2;
329 return t;
330 }
331
333 // GraphOutline outline = new GraphOutline(polyline);
334 /**needed to generate vertex references.*/
335 if( !initFromPolyline(polyline, HEdge::HOLE) ) {
336 return;
337 }
338 GraphVertexRef v3 = locateClosestVertex(polyline);
339 if( !v3 ) {
340 WARN_PRINT("Graph: Loop.locateClosestVertex returns null; root valid? %d", (nullptr!=m_root));
341 if( Graph::DEBUG_MODE ) {
342 jau::print_backtrace(true /* skip_anon_frames */, 4 /* max_frames */);
343 }
344 return;
345 }
346 HEdgePtr v3Edge = v3->findBoundEdge();
347 HEdgePtr v3EdgeP = v3Edge->getPrev();
348 HEdgePtr crossEdge = createHEdge(m_root->getGraphPoint(), HEdge::INNER);
349
350 HEdge::connect(m_root->getPrev(), crossEdge);
351 HEdge::connect(crossEdge, v3Edge);
352
353 HEdgePtr crossEdgeSib = crossEdge->getSibling();
354 if(!crossEdgeSib) {
355 crossEdgeSib = createHEdge(crossEdge->getNext()->getGraphPoint(), HEdge::INNER);
356 HEdge::makeSiblings(crossEdge, crossEdgeSib);
357 }
358
359 HEdge::connect(v3EdgeP, crossEdgeSib);
360 HEdge::connect(crossEdgeSib, m_root);
361 }
362
363 bool checkInside(const Vertex& v) {
364 if(!m_box.contains(v.coord())){
365 return false;
366 }
367 bool inside = false;
368 HEdgePtr current = m_root;
369 HEdgePtr next = m_root->getNext();
370 const Vec3f& vc = v.coord();
371 do {
372 const Vec3f& v2c = current->getGraphPoint()->vertex().coord();
373 const Vec3f& v1c = next->getGraphPoint()->vertex().coord();
374
375 if ( ( (v1c.y > vc.y) != (v2c.y > vc.y) ) &&
376 ( vc.x < (v2c.x - v1c.x) * (vc.y - v1c.y) / (v2c.y - v1c.y) + v1c.x ) ) {
377 inside = !inside;
378 }
379 current = next;
380 next = current->getNext();
381 } while(current != m_root);
382 return inside;
383 }
384
386 size_t size = 0;
387 HEdgePtr e = m_root;
388 do{
389 size++;
390 e = e->getNext();
391 } while(e != m_root);
392 return size;
393 }
394 };
395
396
397 /**@}*/
398
399} // namespace gamp::graph::tess::impl
400
401#endif /* JAU_GAMP_GRAPH_TESS_IMPL_LOOP_HPP_ */
402
static bool DEBUG_MODE
Definition Graph.hpp:24
jau::bitfield< 3 > tribit_t
static TriangleRef create(const Vertex &v1, const Vertex &v2, const Vertex &v3, const tribit_t &boundaryVertices) noexcept
constexpr const Vec3f & coord() const noexcept
Definition PrimTypes.hpp:93
constexpr HEdgePtr getPrev() noexcept
Definition HEdge.hpp:81
constexpr HEdgePtr getSibling() noexcept
Definition HEdge.hpp:87
static constexpr int BOUNDARY
Definition HEdge.hpp:48
static void connect(HEdgePtr first, HEdgePtr next) noexcept
Definition HEdge.hpp:94
constexpr HEdgePtr getNext() noexcept
Definition HEdge.hpp:84
static void makeSiblings(HEdgePtr first, HEdgePtr second) noexcept
Definition HEdge.hpp:99
static HEdgeRef create(const GraphVertexRef &vert, int type)
Definition HEdge.hpp:72
constexpr const GraphVertexRef & getGraphPoint() const noexcept
Definition HEdge.hpp:78
static constexpr int HOLE
Definition HEdge.hpp:50
static constexpr int INNER
Definition HEdge.hpp:49
TriangleRef cut(bool delaunay)
Definition Loop.hpp:293
static constexpr bool FixedWindingRule
Definition Loop.hpp:48
bool checkInside(const Vertex &v)
Definition Loop.hpp:363
static LoopRef createBoundary(const GraphOutlineRef &polyline, bool isConvex)
Definition Loop.hpp:279
Loop(Private, GraphOutlineRef polyline, int edgeType, bool isComplex)
Definition Loop.hpp:68
const HEdgePtr & getHEdge() const noexcept
Definition Loop.hpp:287
bool isSimplex() const noexcept
Definition Loop.hpp:289
void addConstraintCurveHole(const GraphOutlineRef &polyline)
Definition Loop.hpp:332
constexpr void put(size_t bitnum, bool v) noexcept
Definition bitfield.hpp:90
constexpr void push_back(const value_type &x)
Like std::vector::push_back(), copy.
Definition darray.hpp:1522
value_type x
Definition vec3f.hpp:69
value_type y
Definition vec3f.hpp:70
Axis Aligned Bounding Box.
Definition aabbox3f.hpp:43
AABBox3f & resize(const AABBox3f &o) noexcept
Resize the aabbox3f to encapsulate another AABox.
Definition aabbox3f.hpp:228
#define ERR_PRINT2(...)
Use for unconditional error messages, prefix '[elapsed_time] Error @ FILE:LINE FUNC: '.
Definition debug.hpp:115
#define WARN_PRINT(...)
Use for unconditional warning messages, prefix '[elapsed_time] Warning @ FILE:LINE FUNC: '.
Definition debug.hpp:126
std::shared_ptr< Loop > LoopRef
Definition Loop.hpp:43
std::shared_ptr< GraphOutline > GraphOutlineRef
std::vector< GraphVertexRef > GraphVertexRefList
Definition HEdge.hpp:39
std::shared_ptr< GraphVertex > GraphVertexRef
Definition HEdge.hpp:38
std::vector< LoopRef > LoopRefList
Definition Loop.hpp:44
HEdge * HEdgePtr
<Unique HEdge reference (pointer) w/ ownership, help at caller site
Definition HEdge.hpp:43
std::unique_ptr< HEdge > HEdgeRef
Definition HEdge.hpp:42
jau::darray< GraphOutlineRef, uint32_t > GraphOutlineRefList
std::shared_ptr< Triangle > TriangleRef
constexpr bool is2DCCW(const Vec3f &a, const Vec3f &b, const Vec3f &c) noexcept
Check if points are in ccw order.
Definition geom3f2D.hpp:133
constexpr bool testSeg2SegIntersection2D(Vec2f *result, const Vec2f &p, const Vec2f &p2, const Vec2f &q, const Vec2f &q2, const bool do_collinear=false) noexcept
See p + t r = q + u s and its terse C# implementation
Definition geom2f.hpp:61
Vector3F< float > Vec3f
Definition vec3f.hpp:436
std::string to_string(const math_error_t v) noexcept
Returns std::string representation of math_error_t.
constexpr bool isInCircle2D(const Vec3f &a, const Vec3f &b, const Vec3f &c, const Vec3f &d) noexcept
Check if vertices in triangle circumcircle given d vertex, from paper by Guibas and Stolfi (1985).
Definition geom3f2D.hpp:117
@ CCW
Counter-Clockwise.
Definition geom.hpp:39
void print_backtrace(const bool skip_anon_frames, const jau::snsize_t max_frames=-1, const jau::snsize_t skip_frames=2) noexcept
Prints the de-mangled backtrace string separated by newline excluding this function to stderr,...
Definition debug.cpp:173
STL namespace.