11#ifndef JAU_GAMP_GRAPH_TESS_IMPL_LOOP_HPP_
12#define JAU_GAMP_GRAPH_TESS_IMPL_LOOP_HPP_
32 using namespace jau::math;
33 using namespace jau::math::geom;
34 using namespace gamp::graph;
35 using namespace gamp::graph::tess;
51 std::vector<HEdgeRef> m_hedges;
58 struct Private{
explicit Private() =
default; };
63 m_hedges.push_back(std::move(owner));
69 : m_hedges(), m_box(),
70 m_initialOutline(
std::move(polyline)),
71 m_isComplex(isComplex),
73 m_root(initFromPolyline(m_initialOutline, edgeType))
86 if(vertices.size() < 3) {
87 ERR_PRINT2(
"Graph: Loop.initFromPolyline: GraphOutline's vertices < 3: %zu", vertices.size() );
107 const size_t maxVertIdx = vertices.size() - 1;
108 for(
size_t index = 0; index <= maxVertIdx; ++index) {
110 m_box.
resize(v1->coord());
112 HEdgePtr edge = createHEdge(v1, edgeType);
115 lastEdge->setNext(edge);
116 edge->setPrev(lastEdge);
122 if(index == maxVertIdx ) {
123 edge->setNext(firstEdge);
124 firstEdge->setPrev(edge);
131 for(
size_t index = vertices.size(); index-- > 0;) {
133 m_box.resize(v1->coord());
135 HEdgePtr edge = createHEdge(v1, edgeType);
138 lastEdge->setNext(edge);
139 edge->setPrev(lastEdge);
145 edge->setNext(firstEdge);
146 firstEdge->setPrev(edge);
163 float minDistance = std::numeric_limits<float>::max();
165 if( initVertices.size() < 2 ) {
172 size_t initSz = initVertices.size();
174 for(
size_t i=1; i< initSz; ++i){
176 for(
size_t pos=0; pos<vertices.size(); ++pos) {
178 float distance = v0->coord().dist(cand->coord());
179 if(distance < minDistance){
182 if( !( vert == v0 || vert == v1 || vert == cand) ) {
183 inside =
isInCircle2D(v0->coord(), v1->coord(), cand->coord(), vert->coord());
191 minDistance = distance;
192 closestE = v0->findBoundEdge();
204 bool intersectsOutline(
const Vertex& a1,
const Vertex& a2,
const Vertex& b) {
207 size_t sz = vertices.size();
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 &&
217 if( *v0 != a2 && v1 != a2 &&
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();
235 ( m_isComplex && intersectsOutline(rootPoint, nextPoint, candPoint) ) ) {
242 while (e != candEdge){
244 const Vertex& ep = egp->vertex();
245 if(egp != rootGPoint &&
272 boundary.
put(0, rootT->getGraphPoint()->isBoundaryContained());
273 boundary.
put(1, rootT->getNext()->getGraphPoint()->isBoundaryContained());
274 boundary.
put(2, rootT->getNext()->getNext()->getGraphPoint()->isBoundaryContained());
290 return (m_root->getNext()->getNext()->getNext() == m_root);
296 Vertex& rootPoint = m_root->getGraphPoint()->vertex();
299 if( m_isComplex && intersectsOutline(rootPoint, nextPoint, candPoint) ) {
302 return Triangle::create(rootPoint, nextPoint, candPoint, checkVerticesBoundary(m_root));
307 m_root = m_root->getNext();
327 TriangleRef t = createTriangle(v1->vertex(), v2->vertex(), v3->vertex(), m_root);
340 WARN_PRINT(
"Graph: Loop.locateClosestVertex returns null; root valid? %d", (
nullptr!=m_root));
346 HEdgePtr v3Edge = v3->findBoundEdge();
364 if(!m_box.contains(v.
coord())){
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 ) ) {
381 }
while(current != m_root);
391 }
while(e != m_root);
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
constexpr HEdgePtr getPrev() noexcept
constexpr HEdgePtr getSibling() noexcept
static constexpr int BOUNDARY
static void connect(HEdgePtr first, HEdgePtr next) noexcept
constexpr HEdgePtr getNext() noexcept
static void makeSiblings(HEdgePtr first, HEdgePtr second) noexcept
static HEdgeRef create(const GraphVertexRef &vert, int type)
constexpr const GraphVertexRef & getGraphPoint() const noexcept
static constexpr int HOLE
static constexpr int INNER
TriangleRef cut(bool delaunay)
static constexpr bool FixedWindingRule
bool checkInside(const Vertex &v)
static LoopRef createBoundary(const GraphOutlineRef &polyline, bool isConvex)
Loop(Private, GraphOutlineRef polyline, int edgeType, bool isComplex)
const HEdgePtr & getHEdge() const noexcept
bool isSimplex() const noexcept
void addConstraintCurveHole(const GraphOutlineRef &polyline)
constexpr void put(size_t bitnum, bool v) noexcept
constexpr void push_back(const value_type &x)
Like std::vector::push_back(), copy.
Axis Aligned Bounding Box.
AABBox3f & resize(const AABBox3f &o) noexcept
Resize the aabbox3f to encapsulate another AABox.
#define ERR_PRINT2(...)
Use for unconditional error messages, prefix '[elapsed_time] Error @ FILE:LINE FUNC: '.
#define WARN_PRINT(...)
Use for unconditional warning messages, prefix '[elapsed_time] Warning @ FILE:LINE FUNC: '.
std::shared_ptr< Loop > LoopRef
std::shared_ptr< GraphOutline > GraphOutlineRef
std::vector< GraphVertexRef > GraphVertexRefList
std::shared_ptr< GraphVertex > GraphVertexRef
std::vector< LoopRef > LoopRefList
HEdge * HEdgePtr
<Unique HEdge reference (pointer) w/ ownership, help at caller site
std::unique_ptr< HEdge > HEdgeRef
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.
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
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).
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,...