2013-09-10 11:43:09 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
|
|
|
* Copyright (C) 2013 CERN
|
|
|
|
* @author Tomasz Wlostowski <tomasz.wlostowski@cern.ch>
|
2019-05-17 00:13:21 +00:00
|
|
|
* Copyright (C) 2013-2020 KiCad Developers, see AUTHORS.txt for contributors.
|
2013-09-10 11:43:09 +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
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifndef __SHAPE_LINE_CHAIN
|
|
|
|
#define __SHAPE_LINE_CHAIN
|
|
|
|
|
2020-01-07 17:12:59 +00:00
|
|
|
|
|
|
|
#include <algorithm> // for max
|
2013-09-10 11:43:09 +00:00
|
|
|
#include <sstream>
|
2019-03-24 13:54:56 +00:00
|
|
|
#include <vector>
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <wx/gdicmn.h> // for wxPoint
|
2013-09-10 11:43:09 +00:00
|
|
|
|
2017-11-01 11:14:16 +00:00
|
|
|
#include <core/optional.h>
|
2013-09-10 11:43:09 +00:00
|
|
|
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <clipper.hpp>
|
2013-09-10 11:43:09 +00:00
|
|
|
#include <geometry/seg.h>
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <geometry/shape.h>
|
2019-03-26 02:55:59 +00:00
|
|
|
#include <geometry/shape_arc.h>
|
2020-01-07 17:12:59 +00:00
|
|
|
#include <math/box2.h> // for BOX2I
|
|
|
|
#include <math/vector2d.h>
|
2019-01-31 11:53:01 +00:00
|
|
|
|
2013-09-10 11:43:09 +00:00
|
|
|
|
|
|
|
/**
|
2020-01-10 14:31:00 +00:00
|
|
|
* SHAPE_LINE_CHAIN
|
2013-09-10 11:43:09 +00:00
|
|
|
*
|
|
|
|
* Represents a polyline (an zero-thickness chain of connected line segments).
|
2013-10-14 18:40:36 +00:00
|
|
|
* I purposedly didn't name it "polyline" to avoid confusion with the existing CPolyLine
|
2020-01-10 14:31:00 +00:00
|
|
|
* in pcbnew.
|
2013-09-10 11:43:09 +00:00
|
|
|
*
|
|
|
|
* SHAPE_LINE_CHAIN class shall not be used for polygons!
|
|
|
|
*/
|
2013-10-14 18:40:36 +00:00
|
|
|
class SHAPE_LINE_CHAIN : public SHAPE
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
typedef std::vector<VECTOR2I>::iterator point_iter;
|
|
|
|
typedef std::vector<VECTOR2I>::const_iterator point_citer;
|
|
|
|
|
|
|
|
public:
|
|
|
|
/**
|
|
|
|
* Struct INTERSECTION
|
|
|
|
*
|
|
|
|
* Represents an intersection between two line segments
|
|
|
|
*/
|
|
|
|
struct INTERSECTION
|
|
|
|
{
|
|
|
|
/// segment belonging from the (this) argument of Intersect()
|
|
|
|
SEG our;
|
|
|
|
/// segment belonging from the aOther argument of Intersect()
|
|
|
|
SEG their;
|
|
|
|
/// point of intersection between our and their.
|
|
|
|
VECTOR2I p;
|
|
|
|
};
|
|
|
|
|
2020-04-14 23:11:01 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Class POINT_INSIDE_TRACKER
|
|
|
|
*
|
|
|
|
* A dynamic state checking if a point lies within polygon with a dynamically built outline (
|
|
|
|
* with each piece of the outline added by AddPolyline ()
|
|
|
|
*/
|
|
|
|
class POINT_INSIDE_TRACKER
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
POINT_INSIDE_TRACKER( const VECTOR2I& aPoint );
|
|
|
|
|
|
|
|
void AddPolyline( const SHAPE_LINE_CHAIN& aPolyline );
|
|
|
|
bool IsInside();
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
bool processVertex ( const VECTOR2I& ip, const VECTOR2I& ipNext );
|
|
|
|
|
|
|
|
VECTOR2I m_point;
|
|
|
|
VECTOR2I m_lastPoint;
|
|
|
|
VECTOR2I m_firstPoint;
|
|
|
|
bool m_finished;
|
|
|
|
int m_state;
|
|
|
|
int m_count;
|
|
|
|
};
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
typedef std::vector<INTERSECTION> INTERSECTIONS;
|
|
|
|
|
2019-03-24 13:54:56 +00:00
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Constructor
|
|
|
|
* Initializes an empty line chain.
|
|
|
|
*/
|
2020-01-17 14:19:27 +00:00
|
|
|
SHAPE_LINE_CHAIN() : SHAPE( SH_LINE_CHAIN ), m_closed( false ), m_width( 0 )
|
2013-10-14 18:40:36 +00:00
|
|
|
{}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Copy Constructor
|
|
|
|
*/
|
2019-03-26 02:55:59 +00:00
|
|
|
|
2020-01-17 14:19:27 +00:00
|
|
|
SHAPE_LINE_CHAIN( const SHAPE_LINE_CHAIN& aShape )
|
|
|
|
: SHAPE( SH_LINE_CHAIN ),
|
|
|
|
m_points( aShape.m_points ),
|
2019-03-26 02:55:59 +00:00
|
|
|
m_shapes( aShape.m_shapes ),
|
|
|
|
m_arcs( aShape.m_arcs ),
|
2020-01-17 14:19:27 +00:00
|
|
|
m_closed( aShape.m_closed ),
|
2019-05-17 00:13:21 +00:00
|
|
|
m_width( aShape.m_width ),
|
|
|
|
m_bbox( aShape.m_bbox )
|
2013-10-14 18:40:36 +00:00
|
|
|
{}
|
|
|
|
|
2020-02-19 17:11:07 +00:00
|
|
|
SHAPE_LINE_CHAIN( const std::vector<int>& aV);
|
|
|
|
|
2019-03-24 13:54:56 +00:00
|
|
|
SHAPE_LINE_CHAIN( const std::vector<wxPoint>& aV, bool aClosed = false )
|
2020-01-17 14:19:27 +00:00
|
|
|
: SHAPE( SH_LINE_CHAIN ), m_closed( aClosed ), m_width( 0 )
|
2019-08-06 04:23:57 +00:00
|
|
|
{
|
|
|
|
m_points.reserve( aV.size() );
|
|
|
|
|
|
|
|
for( auto pt : aV )
|
|
|
|
m_points.emplace_back( pt.x, pt.y );
|
2019-03-26 02:55:59 +00:00
|
|
|
|
|
|
|
m_shapes = std::vector<ssize_t>( aV.size(), ssize_t( SHAPE_IS_PT ) );
|
2019-08-06 04:23:57 +00:00
|
|
|
}
|
|
|
|
|
2019-03-24 13:54:56 +00:00
|
|
|
SHAPE_LINE_CHAIN( const std::vector<VECTOR2I>& aV, bool aClosed = false )
|
2020-01-17 14:19:27 +00:00
|
|
|
: SHAPE( SH_LINE_CHAIN ), m_closed( aClosed ), m_width( 0 )
|
2019-03-23 18:26:44 +00:00
|
|
|
{
|
|
|
|
m_points = aV;
|
2019-03-26 02:55:59 +00:00
|
|
|
m_shapes = std::vector<ssize_t>( aV.size(), ssize_t( SHAPE_IS_PT ) );
|
2019-03-23 18:26:44 +00:00
|
|
|
}
|
|
|
|
|
2019-05-17 00:13:21 +00:00
|
|
|
SHAPE_LINE_CHAIN( const SHAPE_ARC& aArc, bool aClosed = false )
|
|
|
|
: SHAPE( SH_LINE_CHAIN ),
|
|
|
|
m_closed( aClosed ),
|
|
|
|
m_width( 0 )
|
|
|
|
{
|
|
|
|
m_points = aArc.ConvertToPolyline().CPoints();
|
|
|
|
m_arcs.emplace_back( aArc );
|
|
|
|
m_shapes = std::vector<ssize_t>( m_points.size(), 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
SHAPE_LINE_CHAIN( const ClipperLib::Path& aPath ) :
|
|
|
|
SHAPE( SH_LINE_CHAIN ),
|
|
|
|
m_closed( true ),
|
|
|
|
m_width( 0 )
|
2018-09-12 10:19:11 +00:00
|
|
|
{
|
|
|
|
m_points.reserve( aPath.size() );
|
2019-03-26 02:55:59 +00:00
|
|
|
m_shapes = std::vector<ssize_t>( aPath.size(), ssize_t( SHAPE_IS_PT ) );
|
2018-09-12 10:19:11 +00:00
|
|
|
|
2019-01-30 09:12:10 +00:00
|
|
|
for( const auto& point : aPath )
|
2018-09-12 10:19:11 +00:00
|
|
|
m_points.emplace_back( point.X, point.Y );
|
|
|
|
}
|
|
|
|
|
2019-03-24 13:54:56 +00:00
|
|
|
virtual ~SHAPE_LINE_CHAIN()
|
2013-10-14 18:40:36 +00:00
|
|
|
{}
|
|
|
|
|
2020-01-08 01:49:11 +00:00
|
|
|
SHAPE_LINE_CHAIN& operator=(const SHAPE_LINE_CHAIN&) = default;
|
|
|
|
|
2016-09-24 18:53:15 +00:00
|
|
|
SHAPE* Clone() const override;
|
2014-05-14 09:45:01 +00:00
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function Clear()
|
|
|
|
* Removes all points from the line chain.
|
|
|
|
*/
|
|
|
|
void Clear()
|
|
|
|
{
|
|
|
|
m_points.clear();
|
2019-03-26 02:55:59 +00:00
|
|
|
m_arcs.clear();
|
|
|
|
m_shapes.clear();
|
2013-10-14 18:40:36 +00:00
|
|
|
m_closed = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function SetClosed()
|
|
|
|
*
|
|
|
|
* Marks the line chain as closed (i.e. with a segment connecting the last point with
|
|
|
|
* the first point).
|
|
|
|
* @param aClosed: whether the line chain is to be closed or not.
|
|
|
|
*/
|
|
|
|
void SetClosed( bool aClosed )
|
|
|
|
{
|
|
|
|
m_closed = aClosed;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function IsClosed()
|
|
|
|
*
|
|
|
|
* @return aClosed: true, when our line is closed.
|
|
|
|
*/
|
|
|
|
bool IsClosed() const
|
|
|
|
{
|
|
|
|
return m_closed;
|
|
|
|
}
|
|
|
|
|
2020-01-17 14:19:27 +00:00
|
|
|
/**
|
|
|
|
* Sets the width of all segments in the chain
|
|
|
|
* @param aWidth width in internal units
|
|
|
|
*/
|
|
|
|
void SetWidth( int aWidth )
|
|
|
|
{
|
|
|
|
m_width = aWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the current width of the segments in the chain
|
|
|
|
* @return width in internal units
|
|
|
|
*/
|
|
|
|
int Width() const
|
|
|
|
{
|
|
|
|
return m_width;
|
|
|
|
}
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function SegmentCount()
|
|
|
|
*
|
|
|
|
* Returns number of segments in this line chain.
|
|
|
|
* @return number of segments
|
|
|
|
*/
|
|
|
|
int SegmentCount() const
|
|
|
|
{
|
|
|
|
int c = m_points.size() - 1;
|
|
|
|
if( m_closed )
|
|
|
|
c++;
|
|
|
|
|
|
|
|
return std::max( 0, c );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function PointCount()
|
|
|
|
*
|
|
|
|
* Returns the number of points (vertices) in this line chain
|
|
|
|
* @return number of points
|
|
|
|
*/
|
|
|
|
int PointCount() const
|
|
|
|
{
|
|
|
|
return m_points.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Segment()
|
|
|
|
*
|
2017-04-17 09:17:13 +00:00
|
|
|
* Returns a copy of the aIndex-th segment in the line chain.
|
2013-10-14 18:40:36 +00:00
|
|
|
* @param aIndex: index of the segment in the line chain. Negative values are counted from
|
|
|
|
* the end (i.e. -1 means the last segment in the line chain)
|
2017-04-17 09:17:13 +00:00
|
|
|
* @return SEG - aIndex-th segment in the line chain
|
2013-10-14 18:40:36 +00:00
|
|
|
*/
|
|
|
|
SEG Segment( int aIndex )
|
|
|
|
{
|
|
|
|
if( aIndex < 0 )
|
|
|
|
aIndex += SegmentCount();
|
|
|
|
|
2013-11-01 12:56:20 +00:00
|
|
|
if( aIndex == (int)( m_points.size() - 1 ) && m_closed )
|
2013-10-14 18:40:36 +00:00
|
|
|
return SEG( m_points[aIndex], m_points[0], aIndex );
|
|
|
|
else
|
|
|
|
return SEG( m_points[aIndex], m_points[aIndex + 1], aIndex );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function CSegment()
|
|
|
|
*
|
2017-04-17 09:17:13 +00:00
|
|
|
* Returns a constant copy of the aIndex-th segment in the line chain.
|
2013-10-14 18:40:36 +00:00
|
|
|
* @param aIndex: index of the segment in the line chain. Negative values are counted from
|
|
|
|
* the end (i.e. -1 means the last segment in the line chain)
|
2017-04-17 09:17:13 +00:00
|
|
|
* @return const SEG - aIndex-th segment in the line chain
|
2013-10-14 18:40:36 +00:00
|
|
|
*/
|
|
|
|
const SEG CSegment( int aIndex ) const
|
|
|
|
{
|
|
|
|
if( aIndex < 0 )
|
|
|
|
aIndex += SegmentCount();
|
|
|
|
|
2013-11-01 12:56:20 +00:00
|
|
|
if( aIndex == (int)( m_points.size() - 1 ) && m_closed )
|
2013-10-14 18:40:36 +00:00
|
|
|
return SEG( const_cast<VECTOR2I&>( m_points[aIndex] ),
|
|
|
|
const_cast<VECTOR2I&>( m_points[0] ), aIndex );
|
|
|
|
else
|
|
|
|
return SEG( const_cast<VECTOR2I&>( m_points[aIndex] ),
|
|
|
|
const_cast<VECTOR2I&>( m_points[aIndex + 1] ), aIndex );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-03-23 18:26:44 +00:00
|
|
|
* Accessor Function to move a point to a specific location
|
|
|
|
* @param aIndex Index (wrapping) of the point to move
|
|
|
|
* @param aPos New absolute location of the point
|
2013-10-14 18:40:36 +00:00
|
|
|
*/
|
2019-03-23 18:26:44 +00:00
|
|
|
void SetPoint( int aIndex, const VECTOR2I& aPos )
|
2013-10-14 18:40:36 +00:00
|
|
|
{
|
|
|
|
if( aIndex < 0 )
|
|
|
|
aIndex += PointCount();
|
2019-03-23 18:26:44 +00:00
|
|
|
else if( aIndex >= PointCount() )
|
|
|
|
aIndex -= PointCount();
|
2013-10-14 18:40:36 +00:00
|
|
|
|
2019-03-23 18:26:44 +00:00
|
|
|
m_points[aIndex] = aPos;
|
2019-03-26 02:55:59 +00:00
|
|
|
|
|
|
|
if( m_shapes[aIndex] != SHAPE_IS_PT )
|
|
|
|
convertArc( m_shapes[aIndex] );
|
2013-10-14 18:40:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-03-23 18:26:44 +00:00
|
|
|
* Function Point()
|
2013-10-14 18:40:36 +00:00
|
|
|
*
|
|
|
|
* Returns a const reference to a given point in the line chain.
|
|
|
|
* @param aIndex index of the point
|
|
|
|
* @return const reference to the point
|
|
|
|
*/
|
|
|
|
const VECTOR2I& CPoint( int aIndex ) const
|
|
|
|
{
|
|
|
|
if( aIndex < 0 )
|
|
|
|
aIndex += PointCount();
|
2015-08-01 10:20:23 +00:00
|
|
|
else if( aIndex >= PointCount() )
|
2015-07-27 19:45:57 +00:00
|
|
|
aIndex -= PointCount();
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
return m_points[aIndex];
|
|
|
|
}
|
|
|
|
|
2018-08-07 11:13:23 +00:00
|
|
|
const std::vector<VECTOR2I>& CPoints() const
|
|
|
|
{
|
|
|
|
return m_points;
|
|
|
|
}
|
|
|
|
|
2018-02-19 10:20:52 +00:00
|
|
|
/**
|
|
|
|
* Returns the last point in the line chain.
|
|
|
|
*/
|
|
|
|
const VECTOR2I& CLastPoint() const
|
|
|
|
{
|
|
|
|
return m_points[PointCount() - 1];
|
|
|
|
}
|
|
|
|
|
2019-05-17 00:13:21 +00:00
|
|
|
/**
|
|
|
|
* @return the vector of stored arcs
|
|
|
|
*/
|
|
|
|
const std::vector<SHAPE_ARC>& CArcs() const
|
|
|
|
{
|
|
|
|
return m_arcs;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @return the vector of values indicating shape type and location
|
|
|
|
*/
|
|
|
|
const std::vector<ssize_t>& CShapes() const
|
|
|
|
{
|
|
|
|
return m_shapes;
|
|
|
|
}
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/// @copydoc SHAPE::BBox()
|
2016-09-24 18:53:15 +00:00
|
|
|
const BOX2I BBox( int aClearance = 0 ) const override
|
2013-10-14 18:40:36 +00:00
|
|
|
{
|
2016-09-01 16:09:06 +00:00
|
|
|
BOX2I bbox;
|
|
|
|
bbox.Compute( m_points );
|
2013-10-14 18:40:36 +00:00
|
|
|
|
2020-01-17 14:19:27 +00:00
|
|
|
if( aClearance != 0 || m_width != 0 )
|
|
|
|
bbox.Inflate( aClearance + m_width );
|
2015-07-02 14:09:43 +00:00
|
|
|
|
2016-09-01 16:09:06 +00:00
|
|
|
return bbox;
|
2013-10-14 18:40:36 +00:00
|
|
|
}
|
|
|
|
|
2019-07-05 22:45:57 +00:00
|
|
|
void GenerateBBoxCache()
|
|
|
|
{
|
|
|
|
m_bbox.Compute( m_points );
|
2020-01-17 14:19:27 +00:00
|
|
|
|
|
|
|
if( m_width != 0 )
|
|
|
|
m_bbox.Inflate( m_width );
|
2019-07-05 22:45:57 +00:00
|
|
|
}
|
|
|
|
|
2020-05-14 21:01:39 +00:00
|
|
|
const BOX2I BBoxFromCache() const
|
|
|
|
{
|
|
|
|
return m_bbox;
|
|
|
|
}
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function Collide()
|
|
|
|
*
|
|
|
|
* Checks if point aP lies closer to us than aClearance.
|
|
|
|
* @param aP the point to check for collisions with
|
|
|
|
* @param aClearance minimum distance that does not qualify as a collision.
|
|
|
|
* @return true, when a collision has been found
|
|
|
|
*/
|
2016-09-24 18:53:15 +00:00
|
|
|
bool Collide( const VECTOR2I& aP, int aClearance = 0 ) const override;
|
2013-10-14 18:40:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Collide()
|
|
|
|
*
|
|
|
|
* Checks if segment aSeg lies closer to us than aClearance.
|
|
|
|
* @param aSeg the segment to check for collisions with
|
|
|
|
* @param aClearance minimum distance that does not qualify as a collision.
|
|
|
|
* @return true, when a collision has been found
|
|
|
|
*/
|
2016-09-24 18:53:15 +00:00
|
|
|
bool Collide( const SEG& aSeg, int aClearance = 0 ) const override;
|
2013-10-14 18:40:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Distance()
|
|
|
|
*
|
|
|
|
* Computes the minimum distance between the line chain and a point aP.
|
|
|
|
* @param aP the point
|
|
|
|
* @return minimum distance.
|
|
|
|
*/
|
2017-10-19 21:14:36 +00:00
|
|
|
int Distance( const VECTOR2I& aP, bool aOutlineOnly = false ) const;
|
2013-10-14 18:40:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Reverse()
|
|
|
|
*
|
|
|
|
* Reverses point order in the line chain.
|
|
|
|
* @return line chain with reversed point order (original A-B-C-D: returned D-C-B-A)
|
|
|
|
*/
|
|
|
|
const SHAPE_LINE_CHAIN Reverse() const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Length()
|
|
|
|
*
|
|
|
|
* Returns length of the line chain in Euclidean metric.
|
|
|
|
* @return length of the line chain
|
|
|
|
*/
|
2019-09-03 23:38:58 +00:00
|
|
|
long long int Length() const;
|
2013-10-14 18:40:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Append()
|
|
|
|
*
|
|
|
|
* Appends a new point at the end of the line chain.
|
|
|
|
* @param aX is X coordinate of the new point
|
|
|
|
* @param aY is Y coordinate of the new point
|
2017-06-18 07:04:42 +00:00
|
|
|
* @param aAllowDuplication = true to append the new point
|
|
|
|
* even it is the same as the last entered point
|
|
|
|
* false (default) to skip it if it is the same as the last entered point
|
2013-10-14 18:40:36 +00:00
|
|
|
*/
|
2015-06-12 15:11:56 +00:00
|
|
|
void Append( int aX, int aY, bool aAllowDuplication = false )
|
2013-10-14 18:40:36 +00:00
|
|
|
{
|
|
|
|
VECTOR2I v( aX, aY );
|
2015-06-12 15:11:56 +00:00
|
|
|
Append( v, aAllowDuplication );
|
2013-10-14 18:40:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Append()
|
|
|
|
*
|
|
|
|
* Appends a new point at the end of the line chain.
|
|
|
|
* @param aP the new point
|
2017-06-18 07:04:42 +00:00
|
|
|
* @param aAllowDuplication = true to append the new point
|
|
|
|
* even it is the same as the last entered point
|
|
|
|
* false (default) to skip it if it is the same as the last entered point
|
2013-10-14 18:40:36 +00:00
|
|
|
*/
|
2015-06-12 15:11:56 +00:00
|
|
|
void Append( const VECTOR2I& aP, bool aAllowDuplication = false )
|
2013-10-14 18:40:36 +00:00
|
|
|
{
|
2018-10-09 21:50:20 +00:00
|
|
|
if( m_points.size() == 0 )
|
|
|
|
m_bbox = BOX2I( aP, VECTOR2I( 0, 0 ) );
|
|
|
|
|
2015-06-12 15:11:56 +00:00
|
|
|
if( m_points.size() == 0 || aAllowDuplication || CPoint( -1 ) != aP )
|
2018-10-09 21:50:20 +00:00
|
|
|
{
|
2013-10-14 18:40:36 +00:00
|
|
|
m_points.push_back( aP );
|
2019-03-26 02:55:59 +00:00
|
|
|
m_shapes.push_back( ssize_t( SHAPE_IS_PT ) );
|
2018-10-09 21:50:20 +00:00
|
|
|
m_bbox.Merge( aP );
|
|
|
|
}
|
2013-10-14 18:40:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Append()
|
|
|
|
*
|
|
|
|
* Appends another line chain at the end.
|
|
|
|
* @param aOtherLine the line chain to be appended.
|
|
|
|
*/
|
2019-03-26 02:55:59 +00:00
|
|
|
void Append( const SHAPE_LINE_CHAIN& aOtherLine );
|
2013-10-14 18:40:36 +00:00
|
|
|
|
2019-03-26 02:55:59 +00:00
|
|
|
void Append( const SHAPE_ARC& aArc );
|
2013-10-14 11:43:57 +00:00
|
|
|
|
2019-03-26 02:55:59 +00:00
|
|
|
void Insert( size_t aVertex, const VECTOR2I& aP );
|
2013-10-14 18:40:36 +00:00
|
|
|
|
2019-03-26 02:55:59 +00:00
|
|
|
void Insert( size_t aVertex, const SHAPE_ARC& aArc );
|
2014-05-14 09:45:01 +00:00
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function Replace()
|
|
|
|
*
|
|
|
|
* Replaces points with indices in range [start_index, end_index] with a single
|
|
|
|
* point aP.
|
|
|
|
* @param aStartIndex start of the point range to be replaced (inclusive)
|
|
|
|
* @param aEndIndex end of the point range to be replaced (inclusive)
|
|
|
|
* @param aP replacement point
|
|
|
|
*/
|
|
|
|
void Replace( int aStartIndex, int aEndIndex, const VECTOR2I& aP );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Replace()
|
|
|
|
*
|
|
|
|
* Replaces points with indices in range [start_index, end_index] with the points from
|
|
|
|
* line chain aLine.
|
|
|
|
* @param aStartIndex start of the point range to be replaced (inclusive)
|
|
|
|
* @param aEndIndex end of the point range to be replaced (inclusive)
|
|
|
|
* @param aLine replacement line chain.
|
|
|
|
*/
|
|
|
|
void Replace( int aStartIndex, int aEndIndex, const SHAPE_LINE_CHAIN& aLine );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Remove()
|
|
|
|
*
|
|
|
|
* Removes the range of points [start_index, end_index] from the line chain.
|
|
|
|
* @param aStartIndex start of the point range to be replaced (inclusive)
|
|
|
|
* @param aEndIndex end of the point range to be replaced (inclusive)
|
|
|
|
*/
|
|
|
|
void Remove( int aStartIndex, int aEndIndex );
|
|
|
|
|
2017-03-07 12:06:00 +00:00
|
|
|
/**
|
|
|
|
* Function Remove()
|
|
|
|
* removes the aIndex-th point from the line chain.
|
|
|
|
* @param aIndex is the index of the point to be removed.
|
|
|
|
*/
|
|
|
|
void Remove( int aIndex )
|
|
|
|
{
|
|
|
|
Remove( aIndex, aIndex );
|
|
|
|
}
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function Split()
|
|
|
|
*
|
|
|
|
* Inserts the point aP belonging to one of the our segments, splitting the adjacent
|
|
|
|
* segment in two.
|
|
|
|
* @param aP the point to be inserted
|
|
|
|
* @return index of the newly inserted point (or a negative value if aP does not lie on
|
|
|
|
* our line)
|
|
|
|
*/
|
|
|
|
int Split( const VECTOR2I& aP );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Find()
|
|
|
|
*
|
|
|
|
* Searches for point aP.
|
|
|
|
* @param aP the point to be looked for
|
|
|
|
* @return index of the correspoinding point in the line chain or negative when not found.
|
|
|
|
*/
|
2014-02-17 10:33:03 +00:00
|
|
|
int Find( const VECTOR2I& aP ) const;
|
2013-10-14 18:40:36 +00:00
|
|
|
|
2014-05-14 09:45:01 +00:00
|
|
|
/**
|
|
|
|
* Function FindSegment()
|
|
|
|
*
|
|
|
|
* Searches for segment containing point aP.
|
|
|
|
* @param aP the point to be looked for
|
|
|
|
* @return index of the correspoinding segment in the line chain or negative when not found.
|
|
|
|
*/
|
|
|
|
int FindSegment( const VECTOR2I& aP ) const;
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function Slice()
|
|
|
|
*
|
|
|
|
* Returns a subset of this line chain containing the [start_index, end_index] range of points.
|
|
|
|
* @param aStartIndex start of the point range to be returned (inclusive)
|
|
|
|
* @param aEndIndex end of the point range to be returned (inclusive)
|
|
|
|
* @return cut line chain.
|
|
|
|
*/
|
|
|
|
const SHAPE_LINE_CHAIN Slice( int aStartIndex, int aEndIndex = -1 ) const;
|
|
|
|
|
|
|
|
struct compareOriginDistance
|
|
|
|
{
|
2014-05-14 11:52:29 +00:00
|
|
|
compareOriginDistance( const VECTOR2I& aOrigin ):
|
2013-10-14 18:40:36 +00:00
|
|
|
m_origin( aOrigin )
|
|
|
|
{}
|
|
|
|
|
|
|
|
bool operator()( const INTERSECTION& aA, const INTERSECTION& aB )
|
2013-10-14 11:43:57 +00:00
|
|
|
{
|
2013-10-14 18:40:36 +00:00
|
|
|
return ( m_origin - aA.p ).EuclideanNorm() < ( m_origin - aB.p ).EuclideanNorm();
|
2013-10-14 11:43:57 +00:00
|
|
|
}
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
VECTOR2I m_origin;
|
|
|
|
};
|
|
|
|
|
2014-05-14 09:45:01 +00:00
|
|
|
bool Intersects( const SHAPE_LINE_CHAIN& aChain ) const;
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function Intersect()
|
|
|
|
*
|
|
|
|
* Finds all intersection points between our line chain and the segment aSeg.
|
|
|
|
* @param aSeg the segment chain to find intersections with
|
|
|
|
* @param aIp reference to a vector to store found intersections. Intersection points
|
|
|
|
* are sorted with increasing distances from point aSeg.a.
|
|
|
|
* @return number of intersections found
|
|
|
|
*/
|
|
|
|
int Intersect( const SEG& aSeg, INTERSECTIONS& aIp ) const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Intersect()
|
|
|
|
*
|
|
|
|
* Finds all intersection points between our line chain and the line chain aChain.
|
|
|
|
* @param aChain the line chain to find intersections with
|
|
|
|
* @param aIp reference to a vector to store found intersections. Intersection points
|
|
|
|
* are sorted with increasing path lengths from the starting point of aChain.
|
|
|
|
* @return number of intersections found
|
|
|
|
*/
|
|
|
|
int Intersect( const SHAPE_LINE_CHAIN& aChain, INTERSECTIONS& aIp ) const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function PathLength()
|
|
|
|
*
|
|
|
|
* Computes the walk path length from the beginning of the line chain and
|
|
|
|
* the point aP belonging to our line.
|
|
|
|
* @return: path length in Euclidean metric or negative if aP does not belong to
|
|
|
|
* the line chain.
|
|
|
|
*/
|
|
|
|
int PathLength( const VECTOR2I& aP ) const;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function PointInside()
|
|
|
|
*
|
2018-12-19 18:53:27 +00:00
|
|
|
* Checks if point aP lies inside a polygon (any type) defined by the line chain.
|
|
|
|
* For closed shapes only.
|
2019-05-05 10:33:34 +00:00
|
|
|
* @param aPt point to check
|
2019-07-05 22:45:57 +00:00
|
|
|
* @param aUseBBoxCache gives better peformance if the bounding boxe caches have been
|
|
|
|
* generated.
|
2013-10-14 18:40:36 +00:00
|
|
|
* @return true if the point is inside the shape (edge is not treated as being inside).
|
|
|
|
*/
|
2020-04-14 23:11:01 +00:00
|
|
|
bool PointInside( const VECTOR2I& aPt, int aAccuracy = 0, bool aUseBBoxCache = false ) const;
|
2020-02-19 17:11:07 +00:00
|
|
|
|
2020-04-14 23:11:01 +00:00
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function PointOnEdge()
|
|
|
|
*
|
|
|
|
* Checks if point aP lies on an edge or vertex of the line chain.
|
|
|
|
* @param aP point to check
|
|
|
|
* @return true if the point lies on the edge.
|
|
|
|
*/
|
2019-05-05 10:33:34 +00:00
|
|
|
bool PointOnEdge( const VECTOR2I& aP, int aAccuracy = 0 ) const;
|
2013-10-14 18:40:36 +00:00
|
|
|
|
2019-02-03 10:21:48 +00:00
|
|
|
/**
|
|
|
|
* Function EdgeContainingPoint()
|
|
|
|
*
|
|
|
|
* Checks if point aP lies on an edge or vertex of the line chain.
|
|
|
|
* @param aP point to check
|
|
|
|
* @return index of the first edge containing the point, otherwise negative
|
|
|
|
*/
|
2019-05-05 10:33:34 +00:00
|
|
|
int EdgeContainingPoint( const VECTOR2I& aP, int aAccuracy = 0 ) const;
|
2019-02-03 10:21:48 +00:00
|
|
|
|
2018-05-08 20:03:52 +00:00
|
|
|
/**
|
|
|
|
* Function CheckClearance()
|
|
|
|
*
|
|
|
|
* Checks if point aP is closer to (or on) an edge or vertex of the line chain.
|
|
|
|
* @param aP point to check
|
|
|
|
* @param aDist distance in internal units
|
|
|
|
* @return true if the point is equal to or closer than aDist to the line chain.
|
|
|
|
*/
|
|
|
|
bool CheckClearance( const VECTOR2I& aP, const int aDist) const;
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function SelfIntersecting()
|
|
|
|
*
|
|
|
|
* Checks if the line chain is self-intersecting.
|
|
|
|
* @return (optional) first found self-intersection point.
|
|
|
|
*/
|
2017-11-01 11:14:16 +00:00
|
|
|
const OPT<INTERSECTION> SelfIntersecting() const;
|
2013-10-14 18:40:36 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Function Simplify()
|
|
|
|
*
|
|
|
|
* Simplifies the line chain by removing colinear adjacent segments and duplicate vertices.
|
|
|
|
* @return reference to self.
|
|
|
|
*/
|
|
|
|
SHAPE_LINE_CHAIN& Simplify();
|
|
|
|
|
2019-03-26 02:55:59 +00:00
|
|
|
/**
|
|
|
|
* Converts an arc to only a point chain by removing the arc and references
|
|
|
|
*
|
|
|
|
* @param aArcIndex index of the arc to convert to points
|
|
|
|
*/
|
|
|
|
void convertArc( ssize_t aArcIndex );
|
|
|
|
|
2018-09-12 10:19:11 +00:00
|
|
|
/**
|
|
|
|
* Creates a new Clipper path from the SHAPE_LINE_CHAIN in a given orientation
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
ClipperLib::Path convertToClipper( bool aRequiredOrientation ) const;
|
|
|
|
|
2020-02-15 18:44:41 +00:00
|
|
|
/**
|
|
|
|
* Find the segment nearest the given point.
|
|
|
|
*
|
|
|
|
* @param aP point to compare with
|
|
|
|
* @return the index of the segment closest to the point
|
|
|
|
*/
|
|
|
|
int NearestSegment( const VECTOR2I& aP ) const;
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/**
|
|
|
|
* Function NearestPoint()
|
|
|
|
*
|
|
|
|
* Finds a point on the line chain that is closest to point aP.
|
|
|
|
* @return the nearest point.
|
|
|
|
*/
|
|
|
|
const VECTOR2I NearestPoint( const VECTOR2I& aP ) const;
|
|
|
|
|
2015-07-02 14:09:43 +00:00
|
|
|
/**
|
|
|
|
* Function NearestPoint()
|
|
|
|
*
|
|
|
|
* Finds a point on the line chain that is closest to the line defined
|
|
|
|
* by the points of segment aSeg, also returns the distance.
|
|
|
|
* @param aSeg Segment defining the line.
|
|
|
|
* @param dist reference receiving the distance to the nearest point.
|
|
|
|
* @return the nearest point.
|
|
|
|
*/
|
|
|
|
const VECTOR2I NearestPoint( const SEG& aSeg, int& dist ) const;
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/// @copydoc SHAPE::Format()
|
2016-09-24 18:53:15 +00:00
|
|
|
const std::string Format() const override;
|
2013-10-14 18:40:36 +00:00
|
|
|
|
2015-08-03 18:29:44 +00:00
|
|
|
/// @copydoc SHAPE::Parse()
|
2016-09-24 18:53:15 +00:00
|
|
|
bool Parse( std::stringstream& aStream ) override;
|
2015-08-03 18:29:44 +00:00
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
bool operator!=( const SHAPE_LINE_CHAIN& aRhs ) const
|
|
|
|
{
|
|
|
|
if( PointCount() != aRhs.PointCount() )
|
|
|
|
return true;
|
|
|
|
|
|
|
|
for( int i = 0; i < PointCount(); i++ )
|
2013-10-14 11:43:57 +00:00
|
|
|
{
|
2013-10-14 18:40:36 +00:00
|
|
|
if( CPoint( i ) != aRhs.CPoint( i ) )
|
2013-10-14 11:43:57 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2017-01-19 12:55:22 +00:00
|
|
|
bool CompareGeometry( const SHAPE_LINE_CHAIN& aOther ) const;
|
2014-05-14 09:45:01 +00:00
|
|
|
|
2016-09-24 18:53:15 +00:00
|
|
|
void Move( const VECTOR2I& aVector ) override
|
2014-11-14 18:18:31 +00:00
|
|
|
{
|
2019-03-26 02:55:59 +00:00
|
|
|
for( auto& pt : m_points )
|
|
|
|
pt += aVector;
|
|
|
|
|
|
|
|
for( auto& arc : m_arcs )
|
|
|
|
arc.Move( aVector );
|
2014-11-14 18:18:31 +00:00
|
|
|
}
|
|
|
|
|
2019-03-23 18:26:44 +00:00
|
|
|
/**
|
|
|
|
* Mirrors the line points about y or x (or both)
|
|
|
|
* @param aX If true, mirror about the y axis (flip X coordinate)
|
|
|
|
* @param aY If true, mirror about the x axis (flip Y coordinate)
|
|
|
|
* @param aRef sets the reference point about which to mirror
|
|
|
|
*/
|
2019-03-26 02:55:59 +00:00
|
|
|
void Mirror( bool aX = true, bool aY = false, const VECTOR2I& aRef = { 0, 0 } );
|
2019-03-23 18:26:44 +00:00
|
|
|
|
2017-12-22 09:54:47 +00:00
|
|
|
/**
|
|
|
|
* Function Rotate
|
|
|
|
* rotates all vertices by a given angle
|
|
|
|
* @param aCenter is the rotation center
|
|
|
|
* @param aAngle rotation angle in radians
|
|
|
|
*/
|
2019-03-23 18:26:44 +00:00
|
|
|
void Rotate( double aAngle, const VECTOR2I& aCenter = VECTOR2I( 0, 0 ) );
|
2017-12-22 09:54:47 +00:00
|
|
|
|
2016-09-24 18:53:15 +00:00
|
|
|
bool IsSolid() const override
|
2014-11-14 18:18:31 +00:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
2015-07-27 19:45:57 +00:00
|
|
|
|
2017-01-19 12:55:22 +00:00
|
|
|
const VECTOR2I PointAlong( int aPathLength ) const;
|
|
|
|
|
2017-10-19 21:14:36 +00:00
|
|
|
double Area() const;
|
|
|
|
|
2019-05-17 00:13:21 +00:00
|
|
|
size_t ArcCount() const
|
|
|
|
{
|
|
|
|
return m_arcs.size();
|
|
|
|
}
|
|
|
|
|
|
|
|
ssize_t ArcIndex( size_t aSegment ) const
|
|
|
|
{
|
|
|
|
if( aSegment >= m_shapes.size() )
|
|
|
|
return SHAPE_IS_PT;
|
|
|
|
|
|
|
|
return m_shapes[aSegment];
|
|
|
|
}
|
|
|
|
|
|
|
|
const SHAPE_ARC& Arc( size_t aArc ) const
|
|
|
|
{
|
|
|
|
return m_arcs[aArc];
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isArc( size_t aSegment ) const
|
|
|
|
{
|
|
|
|
return aSegment < m_shapes.size() && m_shapes[aSegment] != SHAPE_IS_PT;
|
|
|
|
}
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
private:
|
2019-05-17 00:13:21 +00:00
|
|
|
|
|
|
|
constexpr static ssize_t SHAPE_IS_PT = -1;
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/// array of vertices
|
|
|
|
std::vector<VECTOR2I> m_points;
|
2013-10-14 11:43:57 +00:00
|
|
|
|
2019-03-26 02:55:59 +00:00
|
|
|
/**
|
|
|
|
* Array of indices that refer to the index of the shape if the point
|
|
|
|
* is part of a larger shape, e.g. arc or spline.
|
|
|
|
* If the value is -1, the point is just a point
|
|
|
|
*/
|
|
|
|
std::vector<ssize_t> m_shapes;
|
|
|
|
|
|
|
|
std::vector<SHAPE_ARC> m_arcs;
|
|
|
|
|
2013-10-14 18:40:36 +00:00
|
|
|
/// is the line chain closed?
|
|
|
|
bool m_closed;
|
2018-10-09 21:50:20 +00:00
|
|
|
|
2020-01-17 14:19:27 +00:00
|
|
|
/// Width of the segments (for BBox calculations in RTree)
|
|
|
|
/// TODO Adjust usage of SHAPE_LINE_CHAIN to account for where we need a width and where not
|
|
|
|
/// Alternatively, we could split the class into a LINE_CHAIN (no width) and SHAPE_LINE_CHAIN that derives from
|
|
|
|
/// SHAPE as well that does have a width. Not sure yet on the correct path.
|
|
|
|
int m_width;
|
|
|
|
|
2018-10-09 21:50:20 +00:00
|
|
|
/// cached bounding box
|
|
|
|
BOX2I m_bbox;
|
2013-09-10 11:43:09 +00:00
|
|
|
};
|
|
|
|
|
2019-03-26 02:55:59 +00:00
|
|
|
|
2013-09-10 11:43:09 +00:00
|
|
|
#endif // __SHAPE_LINE_CHAIN
|