Gamp v0.0.7-36-g24b1eb6
Gamp: Graphics, Audio, Multimedia and Processing
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
CDTriangulator2D.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_CDTesselator2D_HPP_
12#define JAU_GAMP_GRAPH_TESS_CDTesselator2D_HPP_
13
14#include <jau/backtrace.hpp>
15#include <jau/cpp_lang_util.hpp>
16#include <jau/debug.hpp>
17#include <jau/int_types.hpp>
18#include <jau/io_util.hpp>
20#include <jau/string_util.hpp>
21
22#include <gamp/GampTypes.hpp>
23#include <gamp/graph/Graph.hpp>
28
30
31 using namespace jau::math;
32 using namespace jau::math::geom;
33 using namespace gamp::graph;
34
35 /** \addtogroup Gamp_Graph
36 *
37 * @{
38 */
39
40 /**
41 * Constrained Delaunay Triangulation
42 * implementation of a list of Outlines that define a set of
43 * Closed Regions with optional n holes.
44 */
45 class CDTriangulator2D { // : public Triangulator {
46 private:
48
49 bool complexShape;
50 size_t m_addedVerticeCount;
51 int32_t m_maxTriID;
52
53 public:
54 /** Constructor for a new Delaunay triangulator
55 */
56 constexpr CDTriangulator2D() noexcept
57 : complexShape(false) { reset(); }
58
59 constexpr void setComplexShape(bool complex) noexcept { complexShape = complex; }
60
61 constexpr void reset() noexcept {
62 m_maxTriID = 0;
63 m_addedVerticeCount = 0;
64 loops.clear();
65 }
66
67 constexpr size_t getAddedVerticeCount() const noexcept { return m_addedVerticeCount; }
68
70
71 void addCurve(TriangleRefList& sink, Outline& polyline, float sharpness) {
72 impl::LoopRef loop = getContainerLoop(polyline);
73
74 if( !loop ) {
75 // HEdge.BOUNDARY -> Winding.CCW
76 if( !FixedWindingRule ) {
77 const Winding winding = polyline.getWinding();
78 if( Winding::CCW != winding ) {
79 WARN_PRINT("CDT2.add.xx.BOUNDARY: !CCW but %s", to_string(winding).c_str());
80 // polyline.print(System.err);
81 polyline.setWinding(Winding::CCW); // FIXME: Too late?
82 }
83 }
85 impl::GraphOutlineRef innerPoly = extractBoundaryTriangles(sink, outline, false /* hole */, sharpness);
86 // vertices.addAll(polyline.getVertices());
87 if( innerPoly->graphPoints().size() >= 3 ) {
88 loop = impl::Loop::createBoundary(innerPoly, complexShape);
89 if( loop ) {
90 loops.push_back(loop);
91 }
92 } else if ( Graph::DEBUG_MODE ) {
93 /*
94 * Font FreeMono-Bold: ID 0 + 465: Glyph[id 465 'uni020F', advance 600, leftSideBearings 42, kerning[size 0, horiz true, cross true], shape true], OutlineShape@5e8a459[outlines 2, vertices 34]
95 Drop innerPoly ctrlpts < 3
96 - innerPo[vertices 2, ctrlpts 2] < 3
97 - outline[vertices 4, ctrlpts 4]
98 - Input[vertices 4]
99 *
100 * Font FreeSans-Regular: ID 0 + 409: Glyph[id 409 'Udieresiscaron', advance 720, leftSideBearings 80, kerning[size 0, horiz true, cross false], shape true], OutlineShape@5eb97ced[outlines 3, vertices 33]
101 Drop innerPoly ctrlpts < 3
102 - innerPo[vertices 1, ctrlpts 1] < 3
103 - outline[vertices 1, ctrlpts 1]
104 - Input[vertices 1]
105
106 * Stack:
107 at jogamp.graph.curve.tess.CDTriangulator2D.addCurve(CDTriangulator2D.java:97)
108 at com.jogamp.graph.curve.OutlineShape.triangulateImpl(OutlineShape.java:988)
109 at com.jogamp.graph.curve.OutlineShape.getTriangles(OutlineShape.java:1012)
110 at com.jogamp.graph.curve.Region.countOutlineShape(Region.java:503)
111 at com.jogamp.graph.ui.shapes.GlyphShape.<init>(GlyphShape.java:77)
112 */
113 jau::PLAIN_PRINT(true, "Drop innerPoly ctrlpts < 3");
114 // jau::PLAIN_PRINT(true, "- innerPo[vertices "+innerPoly.getOutline().getVertexCount()+", ctrlpts "+innerPoly.getGraphPoint().size()+"] < 3");
115 // jau::PLAIN_PRINT(true, "- outline[vertices "+outline.getOutline().getVertexCount()+", ctrlpts "+outline.getGraphPoint().size()+"]");
116 // jau::PLAIN_PRINT(true, "- Input[vertices "+polyline.getVertexCount()+"]");
117 jau::print_backtrace(true, 4);
118 }
119 } else {
120 // HEdge.HOLE -> Winding.CW, but Winding.CCW is also accepted!
121 // Winding.CW not required, handled in Loop.initFromPolyline(): polyline.setWinding(winding);
123 impl::GraphOutlineRef innerPoly = extractBoundaryTriangles(sink, outline, true /* hole */, sharpness);
124 // vertices.addAll(innerPoly.getVertices());
125 loop->addConstraintCurveHole(innerPoly);
126 }
127 }
128
130 size_t loopsSize = loops.size();
131 size_t size;
132 for(size_t i=0;i<loopsSize;i++) {
133 impl::LoopRef& loop = loops[i];
134 size_t numTries = 0;
135 size = loop->computeLoopSize();
136 while(!loop->isSimplex()){
137 TriangleRef tri;
138 bool delaunay;
139 if(numTries > size) {
140 tri = loop->cut(false);
141 delaunay = false;
142 } else {
143 tri = loop->cut(true);
144 delaunay = true;
145 }
146 numTries++;
147
148 if(tri) {
149 tri->id() = m_maxTriID++;
150 sink.push_back(tri);
151 // if (Graph::DEBUG_MODE ) {
152 // jau::PLAIN_PRINT(true, "CDTri.gen["+i+"].0: delaunay "+delaunay+", tries "+numTries+", size "+size+", "+tri); // FIXME
153 // }
154 numTries = 0;
155 size--;
156 }
157 if(numTries > size*2){
158 // if (Graph::DEBUG_MODE ) {
159 // jau::PLAIN_PRINT("CDTri.gen["+i+"].X: Triangulation not complete!"); // FIXME
160 // }
161 break;
162 }
163 (void)delaunay;
164 }
165 TriangleRef tri = loop->cut(true);
166 if(tri) {
167 tri->id() = m_maxTriID++;
168 sink.push_back(tri);
169 // if (Graph::DEBUG_MODE ) {
170 // jau::PLAIN_PRINT("CDTri.gen["+i+"].1: size "+size+"/"+loopsSize+", "+tri); // FIXME
171 // }
172 }
173 }
174 }
175
176 private:
177 impl::GraphOutlineRef extractBoundaryTriangles(TriangleRefList& sink, const impl::GraphOutlineRef& outline, bool hole, float sharpness) {
179 const impl::GraphVertexRefList& outVertices = outline->graphPoints();
180 size_t size = outVertices.size();
181 for(size_t i=0; i < size; i++) {
182 const impl::GraphVertexRef& gv1 = outVertices[i]; // currentVertex
183 const impl::GraphVertexRef& gv0 = outVertices[(i+size-1)%size]; // -1
184 const impl::GraphVertexRef& gv2 = outVertices[(i+1)%size]; // +1
185
186 if( !gv1->vertex().onCurve() ) {
187 Vertex v0 = gv0->vertex().clone(); // deep copy w/ a max id marker
188 Vertex v2 = gv2->vertex().clone();
189 Vertex v1 = gv1->vertex().clone();
190 m_addedVerticeCount += 3;
191 Triangle::tribit_t boundaryVertices;
192 boundaryVertices.put(0, true);
193 boundaryVertices.put(1, true);
194 boundaryVertices.put(2, true);
195
196 gv0->setBoundaryContained(true);
197 gv1->setBoundaryContained(true);
198 gv2->setBoundaryContained(true);
199
200 const bool holeLike = !is2DCCW(v0.coord(), v1.coord(), v2.coord());
201 if( hole || holeLike ) {
202 v0.texCoord() = Vec3f(0.0f, -0.1f, 0);
203 v2.texCoord() = Vec3f(1.0f, -0.1f, 0);
204 v1.texCoord() = Vec3f(0.5f, -sharpness-0.1f, 0);
205 innerOutline->addVertex(gv1);
206 } else {
207 v0.texCoord() = Vec3f(0.0f, 0.1f, 0);
208 v2.texCoord() = Vec3f(1.0f, 0.1f, 0);
209 v1.texCoord() = Vec3f(0.5f, sharpness+0.1f, 0);
210 }
211 TriangleRef t;
212 if( holeLike ) {
213 t = Triangle::create(v2, v1, v0, boundaryVertices);
214 } else {
215 t = Triangle::create(v0, v1, v2, boundaryVertices);
216 }
217 t->id() = m_maxTriID++;
218 sink.push_back(t);
219 // if (Graph::DEBUG_MODE ) {
220 // PLAIN_PRINT(true, "CDTri.ebt[%zu].0: hole %d %s, %s", i, (hole || holeLike), gv1->toString().c_str(), t->toString().c_str());
221 // }
222 } else {
223 if( !gv2->vertex().onCurve() || !gv0->vertex().onCurve() ) {
224 gv1->setBoundaryContained(true);
225 }
226 innerOutline->addVertex(gv1);
227 // if (Graph::DEBUG_MODE ) {
228 // PLAIN_PRINT(true, "CDTri.ebt[%zu].1: %s", i, gv1->toString().c_str());
229 // }
230 }
231 }
232 return innerOutline;
233 }
234
235 impl::LoopRef getContainerLoop(const Outline& polyline) noexcept {
236 size_t count = loops.size();
237 if( 0 < count ) {
238 const VertexList& vertices = polyline.vertices();
239 for(size_t i=0; i < count; ++i) {
240 impl::LoopRef loop = loops[i];
241 for(size_t j=0; j < vertices.size(); ++j) { // NOLINT
242 if( loop->checkInside( vertices[j] ) ) {
243 return loop;
244 }
245 }
246 }
247 }
248 return nullptr;
249 }
250 };
251
252 /**@}*/
253
254} // namespace gamp::graph::tess
255
256#endif /* JAU_GAMP_GRAPH_TESS_CDTesselator2D_HPP_ */
static bool DEBUG_MODE
Definition Graph.hpp:24
Define a single continuous stroke by control vertices.
Definition Outline.hpp:51
void setWinding(Winding enforce)
Sets Winding to this outline.
Definition Outline.hpp:180
Winding getWinding() const noexcept
Returns the cached or computed winding of this Outlines polyline using VectorUtil#area(ArrayList).
Definition Outline.hpp:159
jau::bitfield< 3 > tribit_t
static TriangleRef create(const Vertex &v1, const Vertex &v2, const Vertex &v3, const tribit_t &boundaryVertices) noexcept
constexpr Vertex clone() const noexcept
Definition PrimTypes.hpp:56
constexpr const Vec3f & texCoord() const noexcept
Definition PrimTypes.hpp:96
constexpr const Vec3f & coord() const noexcept
Definition PrimTypes.hpp:93
constexpr CDTriangulator2D() noexcept
Constructor for a new Delaunay triangulator.
void generate(TriangleRefList &sink)
constexpr void setComplexShape(bool complex) noexcept
constexpr size_t getAddedVerticeCount() const noexcept
void addCurve(TriangleRefList &sink, Outline &polyline, float sharpness)
static constexpr bool FixedWindingRule
Definition Loop.hpp:48
static LoopRef createBoundary(const GraphOutlineRef &polyline, bool isConvex)
Definition Loop.hpp:279
constexpr void put(size_t bitnum, bool v) noexcept
Definition bitfield.hpp:90
constexpr size_type size() const noexcept
Like std::vector::size().
Definition darray.hpp:1069
constexpr void push_back(const value_type &x)
Like std::vector::push_back(), copy.
Definition darray.hpp:1522
#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
jau::darray< Vertex, uint32_t > VertexList
std::shared_ptr< Triangle > TriangleRef
jau::darray< TriangleRef, uint32_t > TriangleRefList
constexpr bool is2DCCW(const Vec3f &a, const Vec3f &b, const Vec3f &c) noexcept
Check if points are in ccw order.
Definition geom3f2D.hpp:133
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.
@ CCW
Counter-Clockwise.
Definition geom.hpp:39
void PLAIN_PRINT(const bool printPrefix, const char *format,...) noexcept
Use for unconditional plain messages, prefix '[elapsed_time] ' if printPrefix == true.
Definition debug.cpp:258
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