KiCad PCB EDA Suite
Loading...
Searching...
No Matches
geometry_utils.cpp
Go to the documentation of this file.
1/*
2 * This program source code file is part of KiCad, a free EDA CAD application.
3 *
4 * Copyright (C) 2018 Jean-Pierre Charras, jp.charras at wanadoo.fr
5 * Copyright (C) 1992-2021 KiCad Developers, see AUTHORS.txt for contributors.
6 *
7 * This program is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU General Public License
9 * as published by the Free Software Foundation; either version 2
10 * of the License, or (at your option) any later version.
11 *
12 * This program is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 * GNU General Public License for more details.
16 *
17 * You should have received a copy of the GNU General Public License
18 * along with this program; if not, you may find one here:
19 * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
20 * or you may search the http://www.gnu.org website for the version 2 license,
21 * or you may write to the Free Software Foundation, Inc.,
22 * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
23 */
24
30#include <cstdint>
31#include <algorithm> // for max, min
32
34#include <math/util.h> // for KiROUND
35
36// To approximate a circle by segments, a minimal seg count is mandatory.
37// Note that this is rarely used as the maxError constraint will yield a higher
38// segment count on everything but very small circles. (Even a 0.125mm track
39// with a 0.01mm maximum deviation yields 11 segments.)
40#define MIN_SEGCOUNT_FOR_CIRCLE 8
41
42int GetArcToSegmentCount( int aRadius, int aErrorMax, const EDA_ANGLE& aArcAngle )
43{
44 // calculate the number of segments to approximate a circle by segments
45 // given the max distance between the middle of a segment and the circle
46
47 // avoid divide-by-zero
48 aRadius = std::max( 1, aRadius );
49 aErrorMax = std::max( 1, aErrorMax );
50
51 // error relative to the radius value:
52 double rel_error = (double)aErrorMax / aRadius;
53 // minimal arc increment in degrees:
54 double arc_increment = 180 / M_PI * acos( 1.0 - rel_error ) * 2;
55
56 // Ensure a minimal arc increment reasonable value for a circle
57 // (360.0 degrees). For very small radius values, this is mandatory.
58 arc_increment = std::min( 360.0/MIN_SEGCOUNT_FOR_CIRCLE, arc_increment );
59
60 int segCount = KiROUND( fabs( aArcAngle.AsDegrees() ) / arc_increment );
61
62 // Ensure at least two segments are used for algorithmic safety
63 return std::max( segCount, 2 );
64}
65
66
67int CircleToEndSegmentDeltaRadius( int aRadius, int aSegCount )
68{
69 // The minimal seg count is 3, otherwise we cannot calculate the result
70 // in practice, the min count is clamped to 8 in kicad
71 if( aSegCount <= 2 )
72 aSegCount = 3;
73
74 // The angle between the center of the segment and one end of the segment
75 // when the circle is approximated by aSegCount segments
76 double alpha = M_PI / aSegCount;
77
78 // aRadius is the radius of the circle tangent to the middle of each segment
79 // and aRadius/cos(aplha) is the radius of the circle defined by seg ends
80 int delta = KiROUND( std::abs( aRadius * ( 1 - 1/cos( alpha ) ) ) );
81
82 return delta;
83}
84
85// When creating polygons to create a clearance polygonal area, the polygon must
86// be same or bigger than the original shape.
87// Polygons are bigger if the original shape has arcs (round rectangles, ovals,
88// circles...). However, when building the solder mask layer modifying the shapes
89// when converting them to polygons is not acceptable (the modification can break
90// calculations).
91// So one can disable the shape expansion within a particular scope by allocating
92// a DISABLE_ARC_CORRECTION.
93
94static bool s_disable_arc_correction = false;
95
97{
99}
100
102{
104}
105
106int GetCircleToPolyCorrection( int aMaxError )
107{
108 // Push all the error to the outside by increasing the radius
109 return s_disable_arc_correction ? 0 : aMaxError;
110}
111
112
113/***
114 * Utility for the line clipping code, returns the boundary code of
115 * a point. Bit allocation is arbitrary
116 */
117inline int clipOutCode( const BOX2I *aClipBox, int x, int y )
118{
119 int code;
120
121 if( y < aClipBox->GetY() )
122 code = 2;
123 else if( y > aClipBox->GetBottom() )
124 code = 1;
125 else
126 code = 0;
127
128 if( x < aClipBox->GetX() )
129 code |= 4;
130 else if( x > aClipBox->GetRight() )
131 code |= 8;
132
133 return code;
134}
135
136
137bool ClipLine( const BOX2I *aClipBox, int &x1, int &y1, int &x2, int &y2 )
138{
139 // Stock Cohen-Sutherland algorithm; check *any* CG book for details
140 int outcode1 = clipOutCode( aClipBox, x1, y1 );
141 int outcode2 = clipOutCode( aClipBox, x2, y2 );
142
143 while( outcode1 || outcode2 )
144 {
145 // Fast reject
146 if( outcode1 & outcode2 )
147 return true;
148
149 // Choose a side to clip
150 int thisoutcode, x, y;
151
152 if( outcode1 )
153 thisoutcode = outcode1;
154 else
155 thisoutcode = outcode2;
156
157 /* One clip round
158 * Since we use the full range of 32 bit ints, the proportion
159 * computation has to be done in 64 bits to avoid horrible
160 * results */
161 if( thisoutcode & 1 ) // Clip the bottom
162 {
163 y = aClipBox->GetBottom();
164 x = x1 + (x2 - x1) * std::int64_t(y - y1) / (y2 - y1);
165 }
166 else if( thisoutcode & 2 ) // Clip the top
167 {
168 y = aClipBox->GetY();
169 x = x1 + ( x2 - x1 ) * std::int64_t( y - y1 ) / ( y2 - y1 );
170 }
171 else if( thisoutcode & 8 ) // Clip the right
172 {
173 x = aClipBox->GetRight();
174 y = y1 + ( y2 - y1 ) * std::int64_t( x - x1 ) / ( x2 - x1 );
175 }
176 else // if( thisoutcode & 4), obviously, clip the left
177 {
178 x = aClipBox->GetX();
179 y = y1 + ( y2 - y1 ) * std::int64_t( x - x1 ) / ( x2 - x1 );
180 }
181
182 // Put the result back and update the boundary code
183 // No ambiguity, otherwise it would have been a fast reject
184 if( thisoutcode == outcode1 )
185 {
186 x1 = x;
187 y1 = y;
188 outcode1 = clipOutCode( aClipBox, x1, y1 );
189 }
190 else
191 {
192 x2 = x;
193 y2 = y;
194 outcode2 = clipOutCode( aClipBox, x2, y2 );
195 }
196 }
197
198 return false;
199}
200
201
202bool KIGEOM::BoxHitTest( const VECTOR2I& aHitter, const BOX2I& aHittee, int aAccuracy )
203{
204 const BOX2I hittee = aHittee.GetInflated( aAccuracy );
205 return hittee.Contains( aHitter );
206}
207
208
209bool KIGEOM::BoxHitTest( const BOX2I& aHitter, const BOX2I& aHittee, bool aHitteeContained,
210 int aAccuracy )
211{
212 const BOX2I hitter = aHitter.GetInflated( aAccuracy );
213
214 if( aHitteeContained )
215 return hitter.Contains( aHittee );
216
217 return hitter.Intersects( aHittee );
218}
constexpr BOX2I KiROUND(const BOX2D &aBoxD)
Definition: box2.h:990
constexpr coord_type GetY() const
Definition: box2.h:208
constexpr coord_type GetX() const
Definition: box2.h:207
constexpr bool Contains(const Vec &aPoint) const
Definition: box2.h:168
constexpr BOX2< Vec > GetInflated(coord_type aDx, coord_type aDy) const
Get a new rectangle that is this one, inflated by aDx and aDy.
Definition: box2.h:638
constexpr coord_type GetRight() const
Definition: box2.h:217
constexpr bool Intersects(const BOX2< Vec > &aRect) const
Definition: box2.h:311
constexpr coord_type GetBottom() const
Definition: box2.h:222
double AsDegrees() const
Definition: eda_angle.h:113
int GetCircleToPolyCorrection(int aMaxError)
static bool s_disable_arc_correction
int CircleToEndSegmentDeltaRadius(int aRadius, int aSegCount)
int clipOutCode(const BOX2I *aClipBox, int x, int y)
bool ClipLine(const BOX2I *aClipBox, int &x1, int &y1, int &x2, int &y2)
Test if any part of a line falls within the bounds of a rectangle.
int GetArcToSegmentCount(int aRadius, int aErrorMax, const EDA_ANGLE &aArcAngle)
#define MIN_SEGCOUNT_FOR_CIRCLE
a few functions useful in geometry calculations.
bool BoxHitTest(const VECTOR2I &aHitPoint, const BOX2I &aHittee, int aAccuracy)
Perform a point-to-box hit test.
EDA_ANGLE abs(const EDA_ANGLE &aAngle)
Definition: eda_angle.h:390
constexpr int delta