2018-07-13 04:52:21 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2024-01-12 17:04:05 +00:00
|
|
|
* Modifications Copyright (C) 2018-2024 KiCad Developers
|
2018-07-13 04:52:21 +00:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*
|
|
|
|
* Based on Uniform Plane Subdivision algorithm from Lamot, Marko, and Borut Žalik.
|
|
|
|
* "A fast polygon triangulation algorithm based on uniform plane subdivision."
|
|
|
|
* Computers & graphics 27, no. 2 (2003): 239-253.
|
|
|
|
*
|
|
|
|
* Code derived from:
|
|
|
|
* K-3D which is Copyright (c) 2005-2006, Romain Behar, GPL-2, license above
|
|
|
|
* earcut which is Copyright (c) 2016, Mapbox, ISC
|
|
|
|
*
|
|
|
|
* ISC License:
|
|
|
|
* Permission to use, copy, modify, and/or distribute this software for any purpose
|
|
|
|
* with or without fee is hereby granted, provided that the above copyright notice
|
|
|
|
* and this permission notice appear in all copies.
|
|
|
|
*
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
|
|
|
* REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
|
|
* FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
|
|
|
* INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM LOSS
|
|
|
|
* OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER
|
|
|
|
* TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF
|
|
|
|
* THIS SOFTWARE.
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __POLYGON_TRIANGULATION_H
|
|
|
|
#define __POLYGON_TRIANGULATION_H
|
|
|
|
|
|
|
|
#include <algorithm>
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <deque>
|
2018-07-13 04:52:21 +00:00
|
|
|
#include <cmath>
|
|
|
|
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <clipper.hpp>
|
|
|
|
#include <geometry/shape_line_chain.h>
|
|
|
|
#include <geometry/shape_poly_set.h>
|
|
|
|
#include <math/box2.h>
|
|
|
|
#include <math/vector2d.h>
|
2018-09-12 10:19:11 +00:00
|
|
|
|
2024-02-12 23:19:01 +00:00
|
|
|
#define TRIANGULATE_TRACE "triangulate"
|
2024-01-12 17:09:52 +00:00
|
|
|
class POLYGON_TRIANGULATION
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
public:
|
2024-01-12 17:09:52 +00:00
|
|
|
POLYGON_TRIANGULATION( SHAPE_POLY_SET::TRIANGULATED_POLYGON& aResult ) :
|
2018-07-13 04:52:21 +00:00
|
|
|
m_result( aResult )
|
|
|
|
{};
|
|
|
|
|
2024-01-12 17:04:05 +00:00
|
|
|
bool TesselatePolygon( const SHAPE_LINE_CHAIN& aPoly,
|
|
|
|
SHAPE_POLY_SET::TRIANGULATED_POLYGON* aHintData )
|
2020-12-12 03:43:16 +00:00
|
|
|
{
|
|
|
|
m_bbox = aPoly.BBox();
|
|
|
|
m_result.Clear();
|
|
|
|
|
|
|
|
if( !m_bbox.GetWidth() || !m_bbox.GetHeight() )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
/// Place the polygon Vertices into a circular linked list
|
|
|
|
/// and check for lists that have only 0, 1 or 2 elements and
|
|
|
|
/// therefore cannot be polygons
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* firstVertex = createList( aPoly );
|
|
|
|
|
2020-12-12 03:43:16 +00:00
|
|
|
if( !firstVertex || firstVertex->prev == firstVertex->next )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
firstVertex->updateList();
|
|
|
|
|
2024-01-12 17:04:05 +00:00
|
|
|
if( aHintData )
|
|
|
|
{
|
|
|
|
m_result.Triangles() = aHintData->Triangles();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto retval = earcutList( firstVertex );
|
|
|
|
m_vertices.clear();
|
|
|
|
return retval;
|
|
|
|
}
|
2020-12-12 03:43:16 +00:00
|
|
|
}
|
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
private:
|
2024-01-12 17:09:52 +00:00
|
|
|
struct VERTEX
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX( size_t aIndex, double aX, double aY, POLYGON_TRIANGULATION* aParent ) :
|
2020-12-12 03:43:16 +00:00
|
|
|
i( aIndex ),
|
|
|
|
x( aX ),
|
|
|
|
y( aY ),
|
|
|
|
parent( aParent )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
}
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX& operator=( const VERTEX& ) = delete;
|
|
|
|
VERTEX& operator=( VERTEX&& ) = delete;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
2024-01-12 17:09:52 +00:00
|
|
|
bool operator==( const VERTEX& rhs ) const
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
return this->x == rhs.x && this->y == rhs.y;
|
|
|
|
}
|
2024-01-12 17:09:52 +00:00
|
|
|
bool operator!=( const VERTEX& rhs ) const { return !( *this == rhs ); }
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Split the referenced polygon between the reference point and
|
2018-07-13 04:52:21 +00:00
|
|
|
* vertex b, assuming they are in the same polygon. Notes that while we
|
|
|
|
* create a new vertex pointer for the linked list, we maintain the same
|
|
|
|
* vertex index value from the original polygon. In this way, we have
|
|
|
|
* two polygons that both share the same vertices.
|
|
|
|
*
|
2021-07-01 17:46:17 +00:00
|
|
|
* @return the newly created vertex in the polygon that does not include the
|
|
|
|
* reference vertex.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* split( VERTEX* b )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
parent->m_vertices.emplace_back( i, x, y, parent );
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* a2 = &parent->m_vertices.back();
|
2018-07-13 04:52:21 +00:00
|
|
|
parent->m_vertices.emplace_back( b->i, b->x, b->y, parent );
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* b2 = &parent->m_vertices.back();
|
|
|
|
VERTEX* an = next;
|
|
|
|
VERTEX* bp = b->prev;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
next = b;
|
|
|
|
b->prev = this;
|
|
|
|
|
|
|
|
a2->next = an;
|
|
|
|
an->prev = a2;
|
|
|
|
|
|
|
|
b2->next = a2;
|
|
|
|
a2->prev = b2;
|
|
|
|
|
|
|
|
bp->next = b2;
|
|
|
|
b2->prev = bp;
|
|
|
|
|
|
|
|
return b2;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Remove the node from the linked list and z-ordered linked list.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
|
|
|
void remove()
|
|
|
|
{
|
|
|
|
next->prev = prev;
|
|
|
|
prev->next = next;
|
|
|
|
|
|
|
|
if( prevZ )
|
|
|
|
prevZ->nextZ = nextZ;
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
if( nextZ )
|
|
|
|
nextZ->prevZ = prevZ;
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2021-07-18 14:06:48 +00:00
|
|
|
next = nullptr;
|
|
|
|
prev = nullptr;
|
|
|
|
nextZ = nullptr;
|
|
|
|
prevZ = nullptr;
|
2018-07-13 04:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void updateOrder()
|
|
|
|
{
|
|
|
|
if( !z )
|
|
|
|
z = parent->zOrder( x, y );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* After inserting or changing nodes, this function should be called to
|
2021-07-01 17:46:17 +00:00
|
|
|
* remove duplicate vertices and ensure z-ordering is correct.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
|
|
|
void updateList()
|
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* p = next;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
while( p != this )
|
|
|
|
{
|
|
|
|
/**
|
|
|
|
* Remove duplicates
|
|
|
|
*/
|
|
|
|
if( *p == *p->next )
|
|
|
|
{
|
|
|
|
p = p->prev;
|
|
|
|
p->next->remove();
|
|
|
|
|
|
|
|
if( p == p->next )
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
p->updateOrder();
|
|
|
|
p = p->next;
|
|
|
|
};
|
|
|
|
|
|
|
|
updateOrder();
|
|
|
|
zSort();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Sort all vertices in this vertex's list by their Morton code.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
|
|
|
void zSort()
|
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
std::deque<VERTEX*> queue;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
queue.push_back( this );
|
|
|
|
|
|
|
|
for( auto p = next; p && p != this; p = p->next )
|
|
|
|
queue.push_back( p );
|
|
|
|
|
2024-01-12 17:09:52 +00:00
|
|
|
std::sort( queue.begin(), queue.end(), []( const VERTEX* a, const VERTEX* b )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2023-02-13 19:18:04 +00:00
|
|
|
if( a->z != b->z )
|
|
|
|
return a->z < b->z;
|
|
|
|
|
|
|
|
if( a->x != b->x )
|
|
|
|
return a->x < b->x;
|
|
|
|
|
|
|
|
if( a->y != b->y )
|
|
|
|
return a->y < b->y;
|
|
|
|
|
|
|
|
return a->i < b->i;
|
2018-07-13 04:52:21 +00:00
|
|
|
} );
|
|
|
|
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* prev_elem = nullptr;
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
for( auto elem : queue )
|
|
|
|
{
|
|
|
|
if( prev_elem )
|
|
|
|
prev_elem->nextZ = elem;
|
|
|
|
|
|
|
|
elem->prevZ = prev_elem;
|
|
|
|
prev_elem = elem;
|
|
|
|
}
|
|
|
|
|
|
|
|
prev_elem->nextZ = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check to see if triangle surrounds our current vertex
|
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool inTriangle( const VERTEX& a, const VERTEX& b, const VERTEX& c )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
return ( c.x - x ) * ( a.y - y ) - ( a.x - x ) * ( c.y - y ) >= 0
|
|
|
|
&& ( a.x - x ) * ( b.y - y ) - ( b.x - x ) * ( a.y - y ) >= 0
|
|
|
|
&& ( b.x - x ) * ( c.y - y ) - ( c.x - x ) * ( b.y - y ) >= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
const size_t i;
|
|
|
|
const double x;
|
|
|
|
const double y;
|
2024-01-12 17:09:52 +00:00
|
|
|
POLYGON_TRIANGULATION* parent;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
// previous and next vertices nodes in a polygon ring
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* prev = nullptr;
|
|
|
|
VERTEX* next = nullptr;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
// z-order curve value
|
|
|
|
int32_t z = 0;
|
|
|
|
|
|
|
|
// previous and next nodes in z-order
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* prevZ = nullptr;
|
|
|
|
VERTEX* nextZ = nullptr;
|
2018-07-13 04:52:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Calculate the Morton code of the Vertex
|
|
|
|
* http://www.graphics.stanford.edu/~seander/bithacks.html#InterleaveBMN
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
int32_t zOrder( const double aX, const double aY ) const
|
|
|
|
{
|
2024-01-18 16:34:11 +00:00
|
|
|
int32_t x = static_cast<int32_t>( 32767.0 * ( aX - m_bbox.GetX() ) / m_bbox.GetWidth() );
|
|
|
|
int32_t y = static_cast<int32_t>( 32767.0 * ( aY - m_bbox.GetY() ) / m_bbox.GetHeight() );
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
x = ( x | ( x << 8 ) ) & 0x00FF00FF;
|
|
|
|
x = ( x | ( x << 4 ) ) & 0x0F0F0F0F;
|
|
|
|
x = ( x | ( x << 2 ) ) & 0x33333333;
|
|
|
|
x = ( x | ( x << 1 ) ) & 0x55555555;
|
|
|
|
|
|
|
|
y = ( y | ( y << 8 ) ) & 0x00FF00FF;
|
|
|
|
y = ( y | ( y << 4 ) ) & 0x0F0F0F0F;
|
|
|
|
y = ( y | ( y << 2 ) ) & 0x33333333;
|
|
|
|
y = ( y | ( y << 1 ) ) & 0x55555555;
|
|
|
|
|
|
|
|
return x | ( y << 1 );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Iterate through the list to remove NULL triangles if they exist.
|
|
|
|
*
|
2018-07-13 04:52:21 +00:00
|
|
|
* This should only be called as a last resort when tesselation fails
|
|
|
|
* as the NULL triangles are inserted as Steiner points to improve the
|
|
|
|
* triangulation regularity of polygons
|
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* removeNullTriangles( VERTEX* aStart )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* retval = nullptr;
|
|
|
|
VERTEX* p = aStart->next;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
while( p != aStart )
|
|
|
|
{
|
2024-01-18 19:05:45 +00:00
|
|
|
if( *p == *( p->next ) || area( p->prev, p, p->next ) == 0.0 )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
p = p->prev;
|
|
|
|
p->next->remove();
|
2019-01-18 15:13:50 +00:00
|
|
|
retval = aStart;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
if( p == p->next )
|
|
|
|
break;
|
|
|
|
}
|
2024-01-18 19:05:45 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
p = p->next;
|
|
|
|
};
|
|
|
|
|
|
|
|
// We needed an end point above that wouldn't be removed, so
|
|
|
|
// here we do the final check for this as a Steiner point
|
|
|
|
if( area( aStart->prev, aStart, aStart->next ) == 0.0 )
|
|
|
|
{
|
2019-01-18 15:13:50 +00:00
|
|
|
retval = p->next;
|
2018-07-13 04:52:21 +00:00
|
|
|
p->remove();
|
|
|
|
}
|
|
|
|
|
|
|
|
return retval;
|
|
|
|
}
|
|
|
|
|
2018-09-12 10:19:11 +00:00
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Take a Clipper path and converts it into a circular, doubly-linked list for triangulation.
|
2018-09-12 10:19:11 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* createList( const ClipperLib::Path& aPath )
|
2018-09-12 10:19:11 +00:00
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* tail = nullptr;
|
2018-12-17 13:34:56 +00:00
|
|
|
double sum = 0.0;
|
|
|
|
auto len = aPath.size();
|
2018-09-12 10:19:11 +00:00
|
|
|
|
2018-12-17 13:34:56 +00:00
|
|
|
// Check for winding order
|
|
|
|
for( size_t i = 0; i < len; i++ )
|
|
|
|
{
|
|
|
|
auto p1 = aPath.at( i );
|
|
|
|
auto p2 = aPath.at( ( i + 1 ) < len ? i + 1 : 0 );
|
|
|
|
|
2018-12-18 01:13:13 +00:00
|
|
|
sum += ( ( p2.X - p1.X ) * ( p2.Y + p1.Y ) );
|
2018-12-17 13:34:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if( sum <= 0.0 )
|
|
|
|
{
|
|
|
|
for( auto point : aPath )
|
|
|
|
tail = insertVertex( VECTOR2I( point.X, point.Y ), tail );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for( size_t i = 0; i < len; i++ )
|
|
|
|
{
|
2018-12-17 13:51:12 +00:00
|
|
|
auto p = aPath.at( len - i - 1 );
|
2018-12-17 13:34:56 +00:00
|
|
|
tail = insertVertex( VECTOR2I( p.X, p.Y ), tail );
|
|
|
|
}
|
|
|
|
}
|
2018-09-12 10:19:11 +00:00
|
|
|
|
|
|
|
if( tail && ( *tail == *tail->next ) )
|
|
|
|
{
|
|
|
|
tail->next->remove();
|
|
|
|
}
|
|
|
|
|
|
|
|
return tail;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Take a #SHAPE_LINE_CHAIN and links each point into a circular, doubly-linked list.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* createList( const SHAPE_LINE_CHAIN& points )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* tail = nullptr;
|
2018-12-17 13:34:56 +00:00
|
|
|
double sum = 0.0;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
2018-12-17 13:34:56 +00:00
|
|
|
// Check for winding order
|
2018-07-13 04:52:21 +00:00
|
|
|
for( int i = 0; i < points.PointCount(); i++ )
|
2018-12-17 13:34:56 +00:00
|
|
|
{
|
|
|
|
VECTOR2D p1 = points.CPoint( i );
|
|
|
|
VECTOR2D p2 = points.CPoint( i + 1 );
|
|
|
|
|
2018-12-18 01:13:13 +00:00
|
|
|
sum += ( ( p2.x - p1.x ) * ( p2.y + p1.y ) );
|
2018-12-17 13:34:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if( sum > 0.0 )
|
|
|
|
for( int i = points.PointCount() - 1; i >= 0; i--)
|
|
|
|
tail = insertVertex( points.CPoint( i ), tail );
|
|
|
|
else
|
|
|
|
for( int i = 0; i < points.PointCount(); i++ )
|
|
|
|
tail = insertVertex( points.CPoint( i ), tail );
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
if( tail && ( *tail == *tail->next ) )
|
|
|
|
{
|
|
|
|
tail->next->remove();
|
|
|
|
}
|
|
|
|
|
|
|
|
return tail;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Walk through a circular linked list starting at \a aPoint.
|
|
|
|
*
|
|
|
|
* For each point, test to see if the adjacent points form a triangle that is completely
|
|
|
|
* enclosed by the remaining polygon (an "ear" sticking off the polygon). If the three
|
|
|
|
* points form an ear, we log the ear's location and remove the center point from the
|
|
|
|
* linked list.
|
2018-07-13 04:52:21 +00:00
|
|
|
*
|
2021-07-01 17:46:17 +00:00
|
|
|
* This function can be called recursively in the case of difficult polygons. In cases
|
|
|
|
* where there is an intersection (not technically allowed by KiCad, but could exist in
|
|
|
|
* an edited file), we create a single triangle and remove both vertices before attempting
|
|
|
|
* to.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool earcutList( VERTEX* aPoint, int pass = 0 )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
if( !aPoint )
|
2018-09-12 10:19:11 +00:00
|
|
|
return true;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* stop = aPoint;
|
|
|
|
VERTEX* prev;
|
|
|
|
VERTEX* next;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
while( aPoint->prev != aPoint->next )
|
|
|
|
{
|
|
|
|
prev = aPoint->prev;
|
|
|
|
next = aPoint->next;
|
|
|
|
|
|
|
|
if( isEar( aPoint ) )
|
|
|
|
{
|
|
|
|
m_result.AddTriangle( prev->i, aPoint->i, next->i );
|
|
|
|
aPoint->remove();
|
|
|
|
|
|
|
|
// Skip one vertex as the triangle will account for the prev node
|
|
|
|
aPoint = next->next;
|
|
|
|
stop = next->next;
|
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* nextNext = next->next;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
2024-01-18 16:34:11 +00:00
|
|
|
if( *prev != *nextNext && intersects( prev, aPoint, next, nextNext ) &&
|
|
|
|
locallyInside( prev, nextNext ) &&
|
|
|
|
locallyInside( nextNext, prev ) )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-18 16:34:11 +00:00
|
|
|
m_result.AddTriangle( prev->i, aPoint->i, nextNext->i );
|
2018-07-13 04:52:21 +00:00
|
|
|
|
2024-01-18 16:34:11 +00:00
|
|
|
// remove two nodes involved
|
|
|
|
next->remove();
|
|
|
|
aPoint->remove();
|
2018-07-13 04:52:21 +00:00
|
|
|
|
2024-01-18 16:34:11 +00:00
|
|
|
aPoint = nextNext;
|
|
|
|
stop = nextNext;
|
2023-12-13 21:35:10 +00:00
|
|
|
|
2024-01-18 16:34:11 +00:00
|
|
|
continue;
|
|
|
|
}
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
aPoint = next;
|
|
|
|
|
2021-07-01 17:46:17 +00:00
|
|
|
/*
|
|
|
|
* We've searched the entire polygon for available ears and there are still
|
|
|
|
* un-sliced nodes remaining.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
|
|
|
if( aPoint == stop )
|
|
|
|
{
|
|
|
|
// First, try to remove the remaining steiner points
|
2019-01-18 15:13:50 +00:00
|
|
|
// If aPoint is a steiner, we need to re-assign both the start and stop points
|
|
|
|
if( auto newPoint = removeNullTriangles( aPoint ) )
|
|
|
|
{
|
|
|
|
aPoint = newPoint;
|
|
|
|
stop = newPoint;
|
2018-07-13 04:52:21 +00:00
|
|
|
continue;
|
2019-01-18 15:13:50 +00:00
|
|
|
}
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
// If we don't have any NULL triangles left, cut the polygon in two and try again
|
2024-01-18 19:05:45 +00:00
|
|
|
if( !splitPolygon( aPoint ) )
|
|
|
|
return false;
|
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-10-12 19:13:14 +00:00
|
|
|
// Check to see if we are left with only three points in the polygon
|
|
|
|
if( aPoint->next && aPoint->prev == aPoint->next->next )
|
|
|
|
{
|
|
|
|
// Three concave points will never be able to be triangulated because they were
|
|
|
|
// created by an intersecting polygon, so just drop them.
|
|
|
|
if( area( aPoint->prev, aPoint, aPoint->next ) >= 0 )
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-07-01 17:46:17 +00:00
|
|
|
/*
|
|
|
|
* At this point, our polygon should be fully tessellated.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2018-09-12 10:19:11 +00:00
|
|
|
return( aPoint->prev == aPoint->next );
|
2018-07-13 04:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Check whether the given vertex is in the middle of an ear.
|
|
|
|
*
|
|
|
|
* This works by walking forward and backward in zOrder to the limits of the minimal
|
|
|
|
* bounding box formed around the triangle, checking whether any points are located
|
|
|
|
* inside the given triangle.
|
2018-07-13 04:52:21 +00:00
|
|
|
*
|
2021-07-01 17:46:17 +00:00
|
|
|
* @return true if aEar is the apex point of a ear in the polygon.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool isEar( VERTEX* aEar ) const
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
const VERTEX* a = aEar->prev;
|
|
|
|
const VERTEX* b = aEar;
|
|
|
|
const VERTEX* c = aEar->next;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
// If the area >=0, then the three points for a concave sequence
|
|
|
|
// with b as the reflex point
|
|
|
|
if( area( a, b, c ) >= 0 )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// triangle bbox
|
|
|
|
const double minTX = std::min( a->x, std::min( b->x, c->x ) );
|
|
|
|
const double minTY = std::min( a->y, std::min( b->y, c->y ) );
|
|
|
|
const double maxTX = std::max( a->x, std::max( b->x, c->x ) );
|
|
|
|
const double maxTY = std::max( a->y, std::max( b->y, c->y ) );
|
|
|
|
|
|
|
|
// z-order range for the current triangle bounding box
|
|
|
|
const int32_t minZ = zOrder( minTX, minTY );
|
|
|
|
const int32_t maxZ = zOrder( maxTX, maxTY );
|
|
|
|
|
|
|
|
// first look for points inside the triangle in increasing z-order
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* p = aEar->nextZ;
|
2018-07-13 04:52:21 +00:00
|
|
|
|
|
|
|
while( p && p->z <= maxZ )
|
|
|
|
{
|
|
|
|
if( p != a && p != c
|
|
|
|
&& p->inTriangle( *a, *b, *c )
|
|
|
|
&& area( p->prev, p, p->next ) >= 0 )
|
|
|
|
return false;
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
p = p->nextZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
// then look for points in decreasing z-order
|
|
|
|
p = aEar->prevZ;
|
|
|
|
|
|
|
|
while( p && p->z >= minZ )
|
|
|
|
{
|
|
|
|
if( p != a && p != c
|
|
|
|
&& p->inTriangle( *a, *b, *c )
|
|
|
|
&& area( p->prev, p, p->next ) >= 0 )
|
|
|
|
return false;
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
p = p->prevZ;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If we cannot find an ear to slice in the current polygon list, we
|
|
|
|
* use this to split the polygon into two separate lists and slice them each
|
|
|
|
* independently. This is assured to generate at least one new ear if the
|
|
|
|
* split is successful
|
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool splitPolygon( VERTEX* start )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* origPoly = start;
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2024-02-12 23:19:01 +00:00
|
|
|
// Our first attempts to split the polygon will be at overlapping points.
|
|
|
|
// These are natural split points and we only need to switch the loop directions
|
|
|
|
// to generate two new loops. Since they are overlapping, we are do not
|
|
|
|
// need to create a new segment to disconnect the two loops.
|
|
|
|
do
|
|
|
|
{
|
|
|
|
VERTEX* nextZ = origPoly->nextZ;
|
|
|
|
|
|
|
|
if( nextZ && *nextZ == *origPoly )
|
|
|
|
{
|
|
|
|
std::swap( origPoly->next, nextZ->next );
|
|
|
|
origPoly->next->prev = origPoly;
|
|
|
|
nextZ->next->prev = nextZ;
|
|
|
|
|
|
|
|
origPoly->updateList();
|
|
|
|
nextZ->updateList();
|
|
|
|
return earcutList( origPoly ) && earcutList( nextZ );
|
|
|
|
}
|
|
|
|
|
|
|
|
VERTEX* prevZ = origPoly->prevZ;
|
|
|
|
|
|
|
|
if( prevZ && *prevZ == *origPoly )
|
|
|
|
{
|
|
|
|
std::swap( origPoly->next, prevZ->next );
|
|
|
|
origPoly->next->prev = origPoly;
|
|
|
|
prevZ->next->prev = prevZ;
|
|
|
|
|
|
|
|
origPoly->updateList();
|
|
|
|
prevZ->updateList();
|
|
|
|
return earcutList( origPoly ) && earcutList( prevZ );
|
|
|
|
}
|
|
|
|
|
|
|
|
origPoly = origPoly->next;
|
|
|
|
|
|
|
|
} while ( origPoly != start );
|
|
|
|
|
|
|
|
// If we've made it through the split algorithm and we still haven't found a
|
|
|
|
// set of overlapping points, we need to create a new segment to split the polygon
|
|
|
|
// into two separate polygons. We do this by finding the two vertices that form
|
|
|
|
// a valid line (does not cross the existing polygon)
|
2018-07-13 04:52:21 +00:00
|
|
|
do
|
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* marker = origPoly->next->next;
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
while( marker != origPoly->prev )
|
|
|
|
{
|
|
|
|
// Find a diagonal line that is wholly enclosed by the polygon interior
|
|
|
|
if( origPoly->i != marker->i && goodSplit( origPoly, marker ) )
|
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* newPoly = origPoly->split( marker );
|
2018-07-13 04:52:21 +00:00
|
|
|
|
2024-01-18 16:34:11 +00:00
|
|
|
origPoly->updateList();
|
|
|
|
newPoly->updateList();
|
2018-07-13 04:52:21 +00:00
|
|
|
|
2024-01-18 19:05:45 +00:00
|
|
|
return earcutList( origPoly ) && earcutList( newPoly );
|
2018-07-13 04:52:21 +00:00
|
|
|
}
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
marker = marker->next;
|
|
|
|
}
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
origPoly = origPoly->next;
|
|
|
|
} while( origPoly != start );
|
2024-01-18 19:05:45 +00:00
|
|
|
|
|
|
|
return false;
|
2018-07-13 04:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Check if a segment joining two vertices lies fully inside the polygon.
|
|
|
|
* To do this, we first ensure that the line isn't along the polygon edge.
|
|
|
|
* Next, we know that if the line doesn't intersect the polygon, then it is
|
2024-01-18 19:05:45 +00:00
|
|
|
* either fully inside or fully outside the polygon. Next, we ensure that
|
|
|
|
* the proposed split is inside the local area of the polygon at both ends
|
|
|
|
* and the midpoint. Finally, we check to split creates two new polygons,
|
|
|
|
* each with positive area.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool goodSplit( const VERTEX* a, const VERTEX* b ) const
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-18 19:05:45 +00:00
|
|
|
bool a_on_edge = ( a->nextZ && *a == *a->nextZ ) || ( a->prevZ && *a == *a->prevZ );
|
|
|
|
bool b_on_edge = ( b->nextZ && *b == *b->nextZ ) || ( b->prevZ && *b == *b->prevZ );
|
|
|
|
bool no_intersect = a->next->i != b->i && a->prev->i != b->i && !intersectsPolygon( a, b );
|
|
|
|
bool local_split = locallyInside( a, b ) && locallyInside( b, a ) && middleInside( a, b );
|
|
|
|
bool same_dir = area( a->prev, a, b->prev ) != 0.0 || area( a, b->prev, b ) != 0.0;
|
|
|
|
bool has_len = ( *a == *b ) && area( a->prev, a, a->next ) > 0 && area( b->prev, b, b->next ) > 0;
|
|
|
|
|
|
|
|
|
|
|
|
return no_intersect && local_split && ( same_dir || has_len ) && !a_on_edge && !b_on_edge;
|
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Return the twice the signed area of the triangle formed by vertices p, q, and r.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
double area( const VERTEX* p, const VERTEX* q, const VERTEX* r ) const
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
return ( q->y - p->y ) * ( r->x - q->x ) - ( q->x - p->x ) * ( r->y - q->y );
|
|
|
|
}
|
|
|
|
|
2024-01-18 19:05:45 +00:00
|
|
|
|
|
|
|
constexpr int sign( double aVal ) const
|
|
|
|
{
|
|
|
|
return ( aVal > 0 ) - ( aVal < 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If p, q, and r are collinear and r lies between p and q, then return true.
|
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
constexpr bool overlapping( const VERTEX* p, const VERTEX* q, const VERTEX* r ) const
|
2024-01-18 19:05:45 +00:00
|
|
|
{
|
|
|
|
return q->x <= std::max( p->x, r->x ) &&
|
|
|
|
q->x >= std::min( p->x, r->x ) &&
|
|
|
|
q->y <= std::max( p->y, r->y ) &&
|
|
|
|
q->y >= std::min( p->y, r->y );
|
|
|
|
}
|
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Check for intersection between two segments, end points included.
|
|
|
|
*
|
|
|
|
* @return true if p1-p2 intersects q1-q2.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool intersects( const VERTEX* p1, const VERTEX* q1, const VERTEX* p2, const VERTEX* q2 ) const
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-18 19:05:45 +00:00
|
|
|
int sign1 = sign( area( p1, q1, p2 ) );
|
|
|
|
int sign2 = sign( area( p1, q1, q2 ) );
|
|
|
|
int sign3 = sign( area( p2, q2, p1 ) );
|
|
|
|
int sign4 = sign( area( p2, q2, q1 ) );
|
|
|
|
|
|
|
|
if( sign1 != sign2 && sign3 != sign4 )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if( sign1 == 0 && overlapping( p1, p2, q1 ) )
|
2018-07-13 04:52:21 +00:00
|
|
|
return true;
|
|
|
|
|
2024-01-18 19:05:45 +00:00
|
|
|
if( sign2 == 0 && overlapping( p1, q2, q1 ) )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if( sign3 == 0 && overlapping( p2, p1, q2 ) )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if( sign4 == 0 && overlapping( p2, q1, q2 ) )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
|
|
|
|
return false;
|
2018-07-13 04:52:21 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Check whether the segment from vertex a -> vertex b crosses any of the segments
|
2018-07-13 04:52:21 +00:00
|
|
|
* of the polygon of which vertex a is a member.
|
2021-07-01 17:46:17 +00:00
|
|
|
*
|
|
|
|
* @return true if the segment intersects the edge of the polygon.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool intersectsPolygon( const VERTEX* a, const VERTEX* b ) const
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
const VERTEX* p = a->next;
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
do
|
|
|
|
{
|
|
|
|
if( p->i != a->i &&
|
|
|
|
p->next->i != a->i &&
|
|
|
|
p->i != b->i &&
|
|
|
|
p->next->i != b->i && intersects( p, p->next, a, b ) )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
p = p->next;
|
|
|
|
} while( p != a );
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Check whether the segment from vertex a -> vertex b is inside the polygon
|
|
|
|
* around the immediate area of vertex a.
|
|
|
|
*
|
|
|
|
* We don't define the exact area over which the segment is inside but it is guaranteed to
|
|
|
|
* be inside the polygon immediately adjacent to vertex a.
|
|
|
|
*
|
|
|
|
* @return true if the segment from a->b is inside a's polygon next to vertex a.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool locallyInside( const VERTEX* a, const VERTEX* b ) const
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
if( area( a->prev, a, a->next ) < 0 )
|
|
|
|
return area( a, b, a->next ) >= 0 && area( a, a->prev, b ) >= 0;
|
|
|
|
else
|
|
|
|
return area( a, b, a->prev ) < 0 || area( a, a->next, b ) < 0;
|
|
|
|
}
|
|
|
|
|
2024-01-18 19:05:45 +00:00
|
|
|
/**
|
|
|
|
* Check to see if the segment halfway point between a and b is inside the polygon
|
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
bool middleInside( const VERTEX* a, const VERTEX* b ) const
|
2024-01-18 19:05:45 +00:00
|
|
|
{
|
2024-01-12 17:09:52 +00:00
|
|
|
const VERTEX* p = a;
|
2024-01-18 19:05:45 +00:00
|
|
|
bool inside = false;
|
|
|
|
double px = ( a->x + b->x ) / 2;
|
|
|
|
double py = ( a->y + b->y ) / 2;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
if( ( ( p->y > py ) != ( p->next->y > py ) )
|
|
|
|
&& ( px < ( p->next->x - p->x ) * ( py - p->y ) / ( p->next->y - p->y ) + p->x ) )
|
|
|
|
inside = !inside;
|
|
|
|
|
|
|
|
p = p->next;
|
|
|
|
} while( p != a );
|
|
|
|
|
|
|
|
return inside;
|
|
|
|
}
|
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
/**
|
2021-07-01 17:46:17 +00:00
|
|
|
* Create an entry in the vertices lookup and optionally inserts the newly created vertex
|
|
|
|
* into an existing linked list.
|
|
|
|
*
|
|
|
|
* @return a pointer to the newly created vertex.
|
2018-07-13 04:52:21 +00:00
|
|
|
*/
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* insertVertex( const VECTOR2I& pt, VERTEX* last )
|
2018-07-13 04:52:21 +00:00
|
|
|
{
|
|
|
|
m_result.AddVertex( pt );
|
|
|
|
m_vertices.emplace_back( m_result.GetVertexCount() - 1, pt.x, pt.y, this );
|
|
|
|
|
2024-01-12 17:09:52 +00:00
|
|
|
VERTEX* p = &m_vertices.back();
|
2021-07-01 17:46:17 +00:00
|
|
|
|
2018-07-13 04:52:21 +00:00
|
|
|
if( !last )
|
|
|
|
{
|
|
|
|
p->prev = p;
|
|
|
|
p->next = p;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
p->next = last->next;
|
|
|
|
p->prev = last;
|
|
|
|
last->next->prev = p;
|
|
|
|
last->next = p;
|
|
|
|
}
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
2020-12-12 03:43:16 +00:00
|
|
|
private:
|
|
|
|
BOX2I m_bbox;
|
2024-01-12 17:09:52 +00:00
|
|
|
std::deque<VERTEX> m_vertices;
|
2020-12-12 03:43:16 +00:00
|
|
|
SHAPE_POLY_SET::TRIANGULATED_POLYGON& m_result;
|
2018-07-13 04:52:21 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif //__POLYGON_TRIANGULATION_H
|