Rework Copper Sliver check

Adds QA checks to copper sliver tests.  Adds the following checks:
- Dot product between two arms (quickly avoids checks for >90°)
- Checks the sliver is convex (area test)
- Eliminates minor slivers with angles that are approximately 0 and ones
  with the opposite side width beneath a configurable level
- Updates Clipper2 to fix a couple of jagged edges on inflate
- Adds simplify during zone fill inflation to limit jaggies

Fixes https://gitlab.com/kicad/code/kicad/issues/14549

(cherry picked from commit f7f52d77e4)
This commit is contained in:
Seth Hillbrand 2023-04-25 10:26:03 -07:00
parent 600ac72a3d
commit 1a2c9011e6
20 changed files with 93308 additions and 186 deletions

View File

@ -1009,7 +1009,7 @@ public:
* #ROUND_ALL_CORNERS to round regardless of angles * #ROUND_ALL_CORNERS to round regardless of angles
*/ */
void Inflate( int aAmount, int aCircleSegCount, void Inflate( int aAmount, int aCircleSegCount,
CORNER_STRATEGY aCornerStrategy = ROUND_ALL_CORNERS ); CORNER_STRATEGY aCornerStrategy = ROUND_ALL_CORNERS, bool aSimplify = false );
void Deflate( int aAmount, int aCircleSegmentsCount, void Deflate( int aAmount, int aCircleSegmentsCount,
CORNER_STRATEGY aCornerStrategy = CHAMFER_ALL_CORNERS ) CORNER_STRATEGY aCornerStrategy = CHAMFER_ALL_CORNERS )
@ -1402,7 +1402,7 @@ private:
const std::vector<SHAPE_ARC>& aArcBuffer ); const std::vector<SHAPE_ARC>& aArcBuffer );
void inflate1( int aAmount, int aCircleSegCount, CORNER_STRATEGY aCornerStrategy ); void inflate1( int aAmount, int aCircleSegCount, CORNER_STRATEGY aCornerStrategy );
void inflate2( int aAmount, int aCircleSegCount, CORNER_STRATEGY aCornerStrategy ); void inflate2( int aAmount, int aCircleSegCount, CORNER_STRATEGY aCornerStrategy, bool aSimplify = false );
/** /**
* This is the engine to execute all polygon boolean transforms (AND, OR, ... and polygon * This is the engine to execute all polygon boolean transforms (AND, OR, ... and polygon

View File

@ -992,7 +992,8 @@ void SHAPE_POLY_SET::inflate1( int aAmount, int aCircleSegCount, CORNER_STRATEGY
} }
void SHAPE_POLY_SET::inflate2( int aAmount, int aCircleSegCount, CORNER_STRATEGY aCornerStrategy ) void SHAPE_POLY_SET::inflate2( int aAmount, int aCircleSegCount, CORNER_STRATEGY aCornerStrategy,
bool aSimplify )
{ {
using namespace Clipper2Lib; using namespace Clipper2Lib;
// A static table to avoid repetitive calculations of the coefficient // A static table to avoid repetitive calculations of the coefficient
@ -1072,17 +1073,35 @@ void SHAPE_POLY_SET::inflate2( int aAmount, int aCircleSegCount, CORNER_STRATEGY
c.MiterLimit( miterLimit ); c.MiterLimit( miterLimit );
PolyTree64 tree; PolyTree64 tree;
if( aSimplify )
{
Paths64 paths;
c.Execute( aAmount, paths );
Clipper2Lib::SimplifyPaths( paths, std::abs( aAmount ) * coeff, false );
Clipper64 c2;
c2.PreserveCollinear = false;
c2.ReverseSolution = false;
c2.AddSubject( paths );
c2.Execute(ClipType::Union, FillRule::Positive, tree);
}
else
{
c.Execute( aAmount, tree ); c.Execute( aAmount, tree );
}
importTree( tree, zValues, arcBuffer ); importTree( tree, zValues, arcBuffer );
tree.Clear(); tree.Clear();
} }
void SHAPE_POLY_SET::Inflate( int aAmount, int aCircleSegCount, CORNER_STRATEGY aCornerStrategy ) void SHAPE_POLY_SET::Inflate( int aAmount, int aCircleSegCount, CORNER_STRATEGY aCornerStrategy,
bool aSimplify )
{ {
if( ADVANCED_CFG::GetCfg().m_UseClipper2 ) if( ADVANCED_CFG::GetCfg().m_UseClipper2 )
inflate2( aAmount, aCircleSegCount, aCornerStrategy ); inflate2( aAmount, aCircleSegCount, aCornerStrategy, aSimplify );
else else
inflate1( aAmount, aCircleSegCount, aCornerStrategy ); inflate1( aAmount, aCircleSegCount, aCornerStrategy );
} }

View File

@ -84,9 +84,11 @@ bool DRC_TEST_PROVIDER_SLIVER_CHECKER::Run()
if( !reportPhase( _( "Running sliver detection on copper layers..." ) ) ) if( !reportPhase( _( "Running sliver detection on copper layers..." ) ) )
return false; // DRC cancelled return false; // DRC cancelled
int widthTolerance = pcbIUScale.mmToIU( ADVANCED_CFG::GetCfg().m_SliverWidthTolerance ); int64_t widthTolerance = pcbIUScale.mmToIU( ADVANCED_CFG::GetCfg().m_SliverWidthTolerance );
int64_t squared_width = widthTolerance * widthTolerance;
double angleTolerance = ADVANCED_CFG::GetCfg().m_SliverAngleTolerance; double angleTolerance = ADVANCED_CFG::GetCfg().m_SliverAngleTolerance;
int testLength = widthTolerance / ( 2 * sin( DEG2RAD( angleTolerance / 2 ) ) ); double cosangleTol = 2.0 * cos( DEG2RAD( angleTolerance ) );
LSET copperLayerSet = m_drcEngine->GetBoard()->GetEnabledLayers() & LSET::AllCuMask(); LSET copperLayerSet = m_drcEngine->GetBoard()->GetEnabledLayers() & LSET::AllCuMask();
LSEQ copperLayers = copperLayerSet.Seq(); LSEQ copperLayers = copperLayerSet.Seq();
int layerCount = copperLayers.size(); int layerCount = copperLayers.size();
@ -130,7 +132,6 @@ bool DRC_TEST_PROVIDER_SLIVER_CHECKER::Run()
if( !zone->GetIsRuleArea() ) if( !zone->GetIsRuleArea() )
{ {
fill = zone->GetFill( layer )->CloneDropTriangulation(); fill = zone->GetFill( layer )->CloneDropTriangulation();
fill.Unfracture( SHAPE_POLY_SET::PM_FAST );
poly.Append( fill ); poly.Append( fill );
// Report progress on board zones only. Everything else is // Report progress on board zones only. Everything else is
@ -141,7 +142,7 @@ bool DRC_TEST_PROVIDER_SLIVER_CHECKER::Run()
else else
{ {
item->TransformShapeToPolygon( poly, layer, 0, ARC_LOW_DEF, item->TransformShapeToPolygon( poly, layer, 0, ARC_LOW_DEF,
ERROR_OUTSIDE ); ERROR_INSIDE );
} }
if( m_drcEngine->IsCancelled() ) if( m_drcEngine->IsCancelled() )
@ -154,11 +155,7 @@ bool DRC_TEST_PROVIDER_SLIVER_CHECKER::Run()
if( m_drcEngine->IsCancelled() ) if( m_drcEngine->IsCancelled() )
return 0; return 0;
poly.Simplify( SHAPE_POLY_SET::PM_FAST ); poly.Simplify( SHAPE_POLY_SET::POLYGON_MODE::PM_FAST );
// Sharpen corners
poly.Deflate( widthTolerance / 2, ARC_LOW_DEF,
SHAPE_POLY_SET::ALLOW_ACUTE_CORNERS );
return 1; return 1;
}; };
@ -183,7 +180,6 @@ bool DRC_TEST_PROVIDER_SLIVER_CHECKER::Run()
} }
} }
for( int ii = 0; ii < layerCount; ++ii ) for( int ii = 0; ii < layerCount; ++ii )
{ {
PCB_LAYER_ID layer = copperLayers[ii]; PCB_LAYER_ID layer = copperLayers[ii];
@ -203,31 +199,78 @@ bool DRC_TEST_PROVIDER_SLIVER_CHECKER::Run()
{ {
const std::vector<VECTOR2I>& pts = poly.Outline( jj ).CPoints(); const std::vector<VECTOR2I>& pts = poly.Outline( jj ).CPoints();
int ptCount = pts.size(); int ptCount = pts.size();
int offset = 0;
for( int kk = 0; kk < ptCount; ++kk ) auto area = [&]( const VECTOR2I& p, const VECTOR2I& q, const VECTOR2I& r ) -> VECTOR2I::extended_type
{ {
return static_cast<VECTOR2I::extended_type>( q.y - p.y ) * ( r.x - q.x ) -
static_cast<VECTOR2I::extended_type>( q.x - p.x ) * ( r.y - q.y );
};
auto isLocallyInside = [&]( int aA, int aB ) -> bool
{
int prev = ( ptCount + aA - 1 ) % ptCount;
int next = ( aA + 1 ) % ptCount;
if( area( pts[prev], pts[aA], pts[next] ) < 0 )
return area( pts[aA], pts[aB], pts[next] ) >= 0 && area( pts[aA], pts[prev], pts[aB] ) >= 0;
else
return area( pts[aA], pts[aB], pts[prev] ) < 0 || area( pts[aA], pts[next], pts[aB] ) < 0;
};
if( ptCount <= 5 )
continue;
for( int kk = 0; kk < ptCount; kk += offset )
{
int prior_index = ( ptCount + kk - 1 ) % ptCount;
int next_index = ( kk + 1 ) % ptCount;
VECTOR2I pt = pts[ kk ]; VECTOR2I pt = pts[ kk ];
VECTOR2I ptPrior = pts[ ( ptCount + kk - 1 ) % ptCount ]; VECTOR2I ptPrior = pts[ prior_index ];
VECTOR2I vPrior = ( ptPrior - pt ); VECTOR2I vPrior = ( ptPrior - pt );
int forward_offset = 1;
if( std::abs( vPrior.x ) < min_len && std::abs( vPrior.y ) < min_len && ptCount > 5) offset = 1;
while( std::abs( vPrior.x ) < min_len && std::abs( vPrior.y ) < min_len
&& offset < ptCount )
{ {
ptPrior = pts[ ( ptCount + kk - 2 ) % ptCount ]; pt = pts[ ( kk + offset++ ) % ptCount ];
vPrior = ( ptPrior - pt ); vPrior = ( ptPrior - pt );
} }
VECTOR2I ptAfter = pts[ ( kk + 1 ) % ptCount ]; if( offset >= ptCount )
break;
VECTOR2I ptAfter = pts[ next_index ];
VECTOR2I vAfter = ( ptAfter - pt ); VECTOR2I vAfter = ( ptAfter - pt );
if( std::abs( vAfter.x ) < min_len && std::abs( vAfter.y ) < min_len && ptCount > 5 ) while( std::abs( vAfter.x ) < min_len && std::abs( vAfter.y ) < min_len
&& forward_offset < ptCount )
{ {
ptAfter = pts[ ( kk + 2 ) % ptCount ]; next_index = ( kk + forward_offset++ ) % ptCount;
ptAfter = pts[ next_index ];
vAfter = ( ptAfter - pt ); vAfter = ( ptAfter - pt );
} }
VECTOR2I vIncluded = vPrior.Resize( testLength ) - vAfter.Resize( testLength ); if( offset >= ptCount )
break;
if( vIncluded.SquaredEuclideanNorm() < SEG::Square( widthTolerance ) ) // Negative dot product means that the angle is > 90°
if( vPrior.Dot( vAfter ) <= 0 )
continue;
if( !isLocallyInside( prior_index, next_index ) )
continue;
VECTOR2I vIncluded = ptAfter - ptPrior;
double arm1 = vPrior.SquaredEuclideanNorm();
double arm2 = vAfter.SquaredEuclideanNorm();
double opp = vIncluded.SquaredEuclideanNorm();
double cos_ang = std::abs( ( opp - arm1 - arm2 ) / ( std::sqrt( arm1 ) * std::sqrt( arm2 ) ) );
if( cos_ang > cosangleTol && 2.0 - cos_ang > std::numeric_limits<float>::epsilon() && opp > squared_width )
{ {
std::shared_ptr<DRC_ITEM> drce = DRC_ITEM::Create( DRCE_COPPER_SLIVER ); std::shared_ptr<DRC_ITEM> drce = DRC_ITEM::Create( DRCE_COPPER_SLIVER );
drce->SetErrorMessage( drce->GetErrorText() + wxS( " " ) + layerDesc( layer ) ); drce->SetErrorMessage( drce->GetErrorText() + wxS( " " ) + layerDesc( layer ) );

View File

@ -1557,7 +1557,7 @@ bool ZONE_FILLER::fillCopperZone( const ZONE* aZone, PCB_LAYER_ID aLayer, PCB_LA
*/ */
if( half_min_width - epsilon > epsilon ) if( half_min_width - epsilon > epsilon )
aFillPolys.Inflate( half_min_width - epsilon, numSegs, cornerStrategy ); aFillPolys.Inflate( half_min_width - epsilon, numSegs, cornerStrategy, true );
DUMP_POLYS_TO_COPPER_LAYER( aFillPolys, In15_Cu, wxT( "after-reinflating" ) ); DUMP_POLYS_TO_COPPER_LAYER( aFillPolys, In15_Cu, wxT( "after-reinflating" ) );

View File

@ -0,0 +1,688 @@
(kicad_pcb (version 20221018) (generator pcbnew)
(general
(thickness 1.6)
)
(paper "A4")
(layers
(0 "F.Cu" signal)
(31 "B.Cu" signal)
(32 "B.Adhes" user "B.Adhesive")
(33 "F.Adhes" user "F.Adhesive")
(34 "B.Paste" user)
(35 "F.Paste" user)
(36 "B.SilkS" user "B.Silkscreen")
(37 "F.SilkS" user "F.Silkscreen")
(38 "B.Mask" user)
(39 "F.Mask" user)
(40 "Dwgs.User" user "User.Drawings")
(41 "Cmts.User" user "User.Comments")
(42 "Eco1.User" user "User.Eco1")
(43 "Eco2.User" user "User.Eco2")
(44 "Edge.Cuts" user)
(45 "Margin" user)
(46 "B.CrtYd" user "B.Courtyard")
(47 "F.CrtYd" user "F.Courtyard")
(48 "B.Fab" user)
(49 "F.Fab" user)
(50 "User.1" user)
(51 "User.2" user)
(52 "User.3" user)
(53 "User.4" user)
(54 "User.5" user)
(55 "User.6" user)
(56 "User.7" user)
(57 "User.8" user)
(58 "User.9" user)
)
(setup
(pad_to_mask_clearance 0)
(pcbplotparams
(layerselection 0x00010fc_ffffffff)
(plot_on_all_layers_selection 0x0000000_00000000)
(disableapertmacros false)
(usegerberextensions false)
(usegerberattributes true)
(usegerberadvancedattributes true)
(creategerberjobfile true)
(dashed_line_dash_ratio 12.000000)
(dashed_line_gap_ratio 3.000000)
(svgprecision 4)
(plotframeref false)
(viasonmask false)
(mode 1)
(useauxorigin false)
(hpglpennumber 1)
(hpglpenspeed 20)
(hpglpendiameter 15.000000)
(dxfpolygonmode true)
(dxfimperialunits true)
(dxfusepcbnewfont true)
(psnegative false)
(psa4output false)
(plotreference true)
(plotvalue true)
(plotinvisibletext false)
(sketchpadsonfab false)
(subtractmaskfromsilk false)
(outputformat 1)
(mirror false)
(drillshape 1)
(scaleselection 1)
(outputdirectory "")
)
)
(net 0 "")
(footprint "project:C_LOGO" (layer "F.Cu")
(tstamp 710f0618-ba48-4eda-a0d1-184be345cef5)
(at 71 88)
(descr "Test")
(fp_text reference "LOGO101_TOP0" (at 0 0) (layer "F.SilkS") hide
(effects (font (size 1.27 1.27) (thickness 0.15)))
(tstamp 27afe1ea-6789-4aaa-897d-4fc3c0d052d7)
)
(fp_text value "C_LOGO_SCH-28MM7MM" (at 1.1 1.8) (layer "F.Fab") hide
(effects (font (size 1.27 1.27) (thickness 0.15)))
(tstamp 1527b770-c52f-4142-a6cf-1015edd2e0cd)
)
(fp_poly
(pts
(xy 1.011801 -2.635614)
(xy 1.040231 -2.632063)
(xy 1.062153 -2.628414)
(xy 1.114484 -2.617964)
(xy 1.130888 -2.613832)
(xy 1.140922 -2.610495)
(xy 1.154322 -2.607146)
(xy 1.166231 -2.603186)
(xy 1.196375 -2.59312)
(xy 1.196376 -2.593121)
(xy 1.209841 -2.588644)
(xy 1.216539 -2.585294)
(xy 1.226616 -2.581944)
(xy 1.243244 -2.57363)
(xy 1.246795 -2.571847)
(xy 1.256793 -2.568515)
(xy 1.300868 -2.546511)
(xy 1.304229 -2.543124)
(xy 1.310218 -2.540129)
(xy 1.310219 -2.54013)
(xy 1.324241 -2.533145)
(xy 1.327619 -2.529766)
(xy 1.34107 -2.523016)
(xy 1.344392 -2.519692)
(xy 1.351093 -2.516343)
(xy 1.354415 -2.51302)
(xy 1.361141 -2.50967)
(xy 1.364571 -2.506239)
(xy 1.371272 -2.502864)
(xy 1.374568 -2.499567)
(xy 1.381225 -2.496239)
(xy 1.387949 -2.489539)
(xy 1.394693 -2.486168)
(xy 1.404767 -2.476093)
(xy 1.411468 -2.472743)
(xy 1.421544 -2.462666)
(xy 1.42827 -2.459291)
(xy 1.451716 -2.435843)
(xy 1.458418 -2.432493)
(xy 1.5157 -2.375211)
(xy 1.5157 -2.37186)
(xy 1.568935 -2.318626)
(xy 1.277572 -2.103275)
(xy 1.237039 -2.103275)
(xy 1.213193 -2.127121)
(xy 1.177702 -2.15939)
(xy 1.156024 -2.177974)
(xy 1.131596 -2.196284)
(xy 1.112639 -2.208907)
(xy 1.112603 -2.20893)
(xy 1.095077 -2.220621)
(xy 1.079064 -2.228617)
(xy 1.055567 -2.238702)
(xy 1.044024 -2.244473)
(xy 1.028315 -2.249723)
(xy 1.007333 -2.255711)
(xy 0.979896 -2.261798)
(xy 0.962911 -2.264625)
(xy 0.927023 -2.267908)
(xy 0.902462 -2.270975)
(xy 0.898629 -2.270975)
(xy 0.873702 -2.267865)
(xy 0.831115 -2.264567)
(xy 0.774901 -2.25521)
(xy 0.738461 -2.246083)
(xy 0.710951 -2.236928)
(xy 0.686099 -2.227594)
(xy 0.665009 -2.218565)
(xy 0.640725 -2.206423)
(xy 0.61994 -2.194559)
(xy 0.592377 -2.176176)
(xy 0.580657 -2.16738)
(xy 0.562834 -2.153143)
(xy 0.553203 -2.146698)
(xy 0.536833 -2.133073)
(xy 0.523284 -2.119524)
(xy 0.507957 -2.101136)
(xy 0.507981 -2.101164)
(xy 0.491532 -2.0814)
(xy 0.478893 -2.065608)
(xy 0.469291 -2.052813)
(xy 0.46081 -2.041505)
(xy 0.451464 -2.025906)
(xy 0.445795 -2.017413)
(xy 0.440513 -2.006848)
(xy 0.433788 -1.993373)
(xy 0.427038 -1.979923)
(xy 0.415241 -1.956286)
(xy 0.411872 -1.946253)
(xy 0.408546 -1.939575)
(xy 0.406309 -1.932864)
(xy 0.402971 -1.922824)
(xy 0.402959 -1.922789)
(xy 0.396247 -1.902653)
(xy 0.392861 -1.892543)
(xy 0.39286 -1.892543)
(xy 0.390144 -1.884395)
(xy 0.384201 -1.86058)
(xy 0.381082 -1.844958)
(xy 0.381079 -1.844944)
(xy 0.377886 -1.828979)
(xy 0.374763 -1.810219)
(xy 0.371679 -1.788771)
(xy 0.368625 -1.75207)
(xy 0.368625 -1.745348)
(xy 0.365275 -1.705095)
(xy 0.365275 -1.696539)
(xy 0.368625 -1.66634)
(xy 0.368625 -1.662488)
(xy 0.37169 -1.628717)
(xy 0.374904 -1.603174)
(xy 0.378004 -1.581453)
(xy 0.384198 -1.550435)
(xy 0.387257 -1.538202)
(xy 0.387262 -1.538181)
(xy 0.390612 -1.524756)
(xy 0.393481 -1.51328)
(xy 0.396247 -1.505022)
(xy 0.399609 -1.494936)
(xy 0.399621 -1.494901)
(xy 0.402971 -1.484826)
(xy 0.406309 -1.474811)
(xy 0.406321 -1.474776)
(xy 0.408544 -1.46809)
(xy 0.411869 -1.461441)
(xy 0.415245 -1.451361)
(xy 0.41861 -1.444632)
(xy 0.42197 -1.434551)
(xy 0.423755 -1.430969)
(xy 0.430388 -1.417702)
(xy 0.43378 -1.410944)
(xy 0.440513 -1.397477)
(xy 0.44388 -1.390719)
(xy 0.443879 -1.390718)
(xy 0.446917 -1.384642)
(xy 0.4592 -1.372361)
(xy 0.4592 -1.369011)
(xy 0.46255 -1.365661)
(xy 0.46255 -1.362311)
(xy 0.46925 -1.355611)
(xy 0.46925 -1.352206)
(xy 0.47595 -1.345481)
(xy 0.47595 -1.34224)
(xy 0.482675 -1.33554)
(xy 0.482675 -1.332099)
(xy 0.492725 -1.322024)
(xy 0.492725 -1.318711)
(xy 0.5028 -1.308636)
(xy 0.5028 -1.305264)
(xy 0.51955 -1.288483)
(xy 0.51955 -1.285186)
(xy 0.546141 -1.258593)
(xy 0.552843 -1.255243)
(xy 0.566267 -1.241818)
(xy 0.572968 -1.238468)
(xy 0.57964 -1.231794)
(xy 0.586409 -1.228423)
(xy 0.593134 -1.221671)
(xy 0.599793 -1.218343)
(xy 0.603115 -1.21502)
(xy 0.609841 -1.21167)
(xy 0.613218 -1.208292)
(xy 0.620003 -1.2049)
(xy 0.623329 -1.201549)
(xy 0.643393 -1.191518)
(xy 0.64672 -1.188189)
(xy 0.653044 -1.185005)
(xy 0.653043 -1.185004)
(xy 0.659543 -1.181754)
(xy 0.659544 -1.181755)
(xy 0.663125 -1.17997)
(xy 0.673172 -1.176622)
(xy 0.683077 -1.171688)
(xy 0.686681 -1.169885)
(xy 0.696697 -1.166522)
(xy 0.703374 -1.163196)
(xy 0.710086 -1.160959)
(xy 0.710051 -1.160971)
(xy 0.720126 -1.157621)
(xy 0.720161 -1.157609)
(xy 0.730211 -1.154259)
(xy 0.730176 -1.154271)
(xy 0.738485 -1.151507)
(xy 0.750087 -1.148586)
(xy 0.750086 -1.148585)
(xy 0.763406 -1.145262)
(xy 0.763427 -1.145257)
(xy 0.77566 -1.142198)
(xy 0.790495 -1.139236)
(xy 0.809201 -1.136122)
(xy 0.828628 -1.13335)
(xy 0.830717 -1.13335)
(xy 0.900176 -1.12704)
(xy 0.955872 -1.132917)
(xy 0.973704 -1.135886)
(xy 0.988539 -1.138848)
(xy 0.998926 -1.141445)
(xy 1.009003 -1.144796)
(xy 1.022343 -1.14813)
(xy 1.027236 -1.14977)
(xy 1.033984 -1.153144)
(xy 1.04406 -1.156494)
(xy 1.050759 -1.159844)
(xy 1.0608 -1.163182)
(xy 1.071023 -1.168313)
(xy 1.083907 -1.174743)
(xy 1.087284 -1.17812)
(xy 1.100635 -1.18477)
(xy 1.104032 -1.188168)
(xy 1.110816 -1.19156)
(xy 1.114192 -1.19491)
(xy 1.120806 -1.198217)
(xy 1.124103 -1.201514)
(xy 1.130847 -1.204911)
(xy 1.137625 -1.211664)
(xy 1.144282 -1.214992)
(xy 1.151009 -1.22172)
(xy 1.157717 -1.225061)
(xy 1.174467 -1.241836)
(xy 1.18121 -1.245195)
(xy 1.1977 -1.261685)
(xy 1.1977 -1.264927)
(xy 1.215629 -1.28299)
(xy 1.221175 -1.288535)
(xy 1.221175 -1.291886)
(xy 1.260514 -1.331225)
(xy 1.303838 -1.331225)
(xy 1.579425 -1.136032)
(xy 1.579425 -1.095061)
(xy 1.567181 -1.070481)
(xy 1.566 -1.069303)
(xy 1.566 -1.066017)
(xy 1.5526 -1.052592)
(xy 1.5526 -1.049214)
(xy 1.529125 -1.02574)
(xy 1.529125 -1.022381)
(xy 1.465111 -0.958404)
(xy 1.458417 -0.955057)
(xy 1.438347 -0.934986)
(xy 1.431616 -0.931595)
(xy 1.421489 -0.921493)
(xy 1.41486 -0.918178)
(xy 1.408134 -0.911427)
(xy 1.401365 -0.908055)
(xy 1.398043 -0.904732)
(xy 1.391342 -0.901381)
(xy 1.38467 -0.894709)
(xy 1.377944 -0.891334)
(xy 1.374568 -0.887957)
(xy 1.361083 -0.881214)
(xy 1.357708 -0.877865)
(xy 1.351093 -0.874557)
(xy 1.347797 -0.871261)
(xy 1.333836 -0.864229)
(xy 1.320832 -0.857739)
(xy 1.317458 -0.85439)
(xy 1.303718 -0.84752)
(xy 1.303719 -0.84752)
(xy 1.296919 -0.844095)
(xy 1.290094 -0.840695)
(xy 1.290093 -0.840695)
(xy 1.280282 -0.83579)
(xy 1.270201 -0.832429)
(xy 1.266619 -0.830645)
(xy 1.266618 -0.830645)
(xy 1.260118 -0.827395)
(xy 1.260119 -0.827395)
(xy 1.250162 -0.82238)
(xy 1.240015 -0.819006)
(xy 1.233316 -0.815656)
(xy 1.219851 -0.811179)
(xy 1.219886 -0.811191)
(xy 1.213208 -0.808965)
(xy 1.206541 -0.805631)
(xy 1.193097 -0.801128)
(xy 1.181179 -0.797155)
(xy 1.167754 -0.793805)
(xy 1.157704 -0.790455)
(xy 1.128981 -0.783288)
(xy 1.129002 -0.783293)
(xy 1.115602 -0.779943)
(xy 1.101048 -0.776284)
(xy 1.082278 -0.772536)
(xy 1.040231 -0.765537)
(xy 1.011783 -0.761984)
(xy 0.97514 -0.758325)
(xy 0.819648 -0.758325)
(xy 0.776301 -0.761933)
(xy 0.74802 -0.765465)
(xy 0.723537 -0.768958)
(xy 0.702047 -0.772536)
(xy 0.684448 -0.77605)
(xy 0.667638 -0.779432)
(xy 0.649788 -0.783002)
(xy 0.619921 -0.790455)
(xy 0.60987 -0.793805)
(xy 0.596446 -0.797155)
(xy 0.584528 -0.801128)
(xy 0.57627 -0.803893)
(xy 0.562928 -0.807229)
(xy 0.547659 -0.812306)
(xy 0.54096 -0.815656)
(xy 0.534274 -0.817879)
(xy 0.534239 -0.817891)
(xy 0.520782 -0.822376)
(xy 0.514054 -0.825752)
(xy 0.504043 -0.82909)
(xy 0.497362 -0.83243)
(xy 0.487272 -0.835785)
(xy 0.467129 -0.845869)
(xy 0.457104 -0.849211)
(xy 0.433784 -0.860856)
(xy 0.413094 -0.871221)
(xy 0.409761 -0.874553)
(xy 0.403623 -0.877612)
(xy 0.382932 -0.887957)
(xy 0.379556 -0.891332)
(xy 0.366157 -0.898032)
(xy 0.362834 -0.901354)
(xy 0.356109 -0.904705)
(xy 0.352731 -0.908082)
(xy 0.345947 -0.911475)
(xy 0.342624 -0.914822)
(xy 0.335984 -0.91813)
(xy 0.332607 -0.921507)
(xy 0.325907 -0.924857)
(xy 0.322584 -0.92818)
(xy 0.315816 -0.931552)
(xy 0.30909 -0.938303)
(xy 0.302432 -0.941632)
(xy 0.29576 -0.948304)
(xy 0.289006 -0.951669)
(xy 0.278956 -0.961743)
(xy 0.272242 -0.965089)
(xy 0.255492 -0.981863)
(xy 0.248784 -0.985205)
(xy 0.16465 -1.069339)
(xy 0.16465 -1.07269)
(xy 0.144525 -1.092814)
(xy 0.144525 -1.096192)
(xy 0.131125 -1.109617)
(xy 0.131125 -1.112939)
(xy 0.12105 -1.123014)
(xy 0.12105 -1.126364)
(xy 0.111 -1.136414)
(xy 0.111 -1.139764)
(xy 0.104275 -1.146489)
(xy 0.104275 -1.149839)
(xy 0.097575 -1.156539)
(xy 0.097575 -1.159944)
(xy 0.090875 -1.166669)
(xy 0.090875 -1.16988)
(xy 0.08415 -1.176555)
(xy 0.08415 -1.180014)
(xy 0.0808 -1.183364)
(xy 0.0808 -1.186739)
(xy 0.0741 -1.193439)
(xy 0.0741 -1.196789)
(xy 0.07075 -1.200139)
(xy 0.07075 -1.203598)
(xy 0.0674 -1.206973)
(xy 0.0674 -1.210189)
(xy 0.060675 -1.216914)
(xy 0.060675 -1.220264)
(xy 0.057325 -1.223614)
(xy 0.057325 -1.226989)
(xy 0.053975 -1.230339)
(xy 0.053975 -1.233689)
(xy 0.050625 -1.237039)
(xy 0.050625 -1.240389)
(xy 0.047275 -1.243739)
(xy 0.047275 -1.247089)
(xy 0.0439 -1.250464)
(xy 0.0439 -1.253814)
(xy 0.04055 -1.257164)
(xy 0.04055 -1.260514)
(xy 0.0372 -1.263864)
(xy 0.0372 -1.267323)
(xy 0.03385 -1.270698)
(xy 0.03385 -1.273939)
(xy 0.0305 -1.277289)
(xy 0.0305 -1.283989)
(xy 0.02715 -1.287339)
(xy 0.02715 -1.290714)
(xy 0.0238 -1.294064)
(xy 0.0238 -1.297305)
(xy 0.020425 -1.300655)
(xy 0.020425 -1.307573)
(xy 0.017075 -1.310948)
(xy 0.017075 -1.314189)
(xy 0.013725 -1.317539)
(xy 0.013725 -1.324239)
(xy 0.010375 -1.327589)
(xy 0.010375 -1.334314)
(xy 0.007025 -1.337664)
(xy 0.007025 -1.344364)
(xy 0.003675 -1.347714)
(xy 0.003675 -1.351064)
(xy 0.0003 -1.354439)
(xy 0.0003 -1.361139)
(xy -0.00305 -1.364489)
(xy -0.00305 -1.374564)
(xy -0.0064 -1.377914)
(xy -0.0064 -1.384614)
(xy -0.00975 -1.387964)
(xy -0.00975 -1.394689)
(xy -0.0131 -1.398039)
(xy -0.0131 -1.408089)
(xy -0.01645 -1.411439)
(xy -0.01645 -1.421405)
(xy -0.019825 -1.424755)
(xy -0.019825 -1.435023)
(xy -0.023175 -1.438398)
(xy -0.023175 -1.448339)
(xy -0.026525 -1.451689)
(xy -0.026525 -1.465114)
(xy -0.029875 -1.468464)
(xy -0.029875 -1.481889)
(xy -0.033225 -1.485239)
(xy -0.033225 -1.502014)
(xy -0.036575 -1.505364)
(xy -0.036575 -1.52538)
(xy -0.03995 -1.52873)
(xy -0.03995 -1.552314)
(xy -0.0433 -1.555664)
(xy -0.0433 -1.589214)
(xy -0.04665 -1.592564)
(xy -0.04665 -1.791502)
(xy -0.0433 -1.794877)
(xy -0.0433 -1.831752)
(xy -0.03995 -1.835127)
(xy -0.03995 -1.85882)
(xy -0.036575 -1.86217)
(xy -0.036575 -1.882186)
(xy -0.033225 -1.885536)
(xy -0.033225 -1.902311)
(xy -0.029875 -1.905661)
(xy -0.029875 -1.919086)
(xy -0.026525 -1.922436)
(xy -0.026525 -1.935727)
(xy -0.023175 -1.939102)
(xy -0.023175 -1.952611)
(xy -0.019825 -1.955961)
(xy -0.019825 -1.966145)
(xy -0.01645 -1.969495)
(xy -0.01645 -1.979327)
(xy -0.0131 -1.982702)
(xy -0.0131 -1.989511)
(xy -0.00975 -1.992861)
(xy -0.00975 -2.002936)
(xy -0.0064 -2.006286)
(xy -0.0064 -2.012986)
(xy -0.00305 -2.016336)
(xy -0.00305 -2.023061)
(xy 0.0003 -2.026411)
(xy 0.0003 -2.03657)
(xy 0.003675 -2.03992)
(xy 0.003675 -2.043052)
(xy 0.007025 -2.046427)
(xy 0.007025 -2.053236)
(xy 0.010375 -2.056586)
(xy 0.010375 -2.063177)
(xy 0.013725 -2.066552)
(xy 0.013725 -2.073361)
(xy 0.017075 -2.076711)
(xy 0.017075 -2.080061)
(xy 0.020425 -2.083411)
(xy 0.020425 -2.090245)
(xy 0.0238 -2.093595)
(xy 0.0238 -2.096836)
(xy 0.02715 -2.100186)
(xy 0.02715 -2.106911)
(xy 0.0305 -2.110261)
(xy 0.0305 -2.116961)
(xy 0.0372 -2.123661)
(xy 0.0372 -2.126902)
(xy 0.04055 -2.130277)
(xy 0.04055 -2.133736)
(xy 0.0439 -2.137086)
(xy 0.0439 -2.143895)
(xy 0.047275 -2.147245)
(xy 0.047275 -2.150511)
(xy 0.050625 -2.153861)
(xy 0.050625 -2.157211)
(xy 0.053975 -2.160561)
(xy 0.053975 -2.163911)
(xy 0.060675 -2.170611)
(xy 0.060675 -2.173986)
(xy 0.064025 -2.177336)
(xy 0.064025 -2.180795)
(xy 0.0674 -2.184145)
(xy 0.0674 -2.187277)
(xy 0.07075 -2.190652)
(xy 0.07075 -2.194111)
(xy 0.07745 -2.200811)
(xy 0.07745 -2.204161)
(xy 0.0808 -2.207511)
(xy 0.0808 -2.210752)
(xy 0.087525 -2.217527)
(xy 0.087525 -2.220936)
(xy 0.090875 -2.224286)
(xy 0.090875 -2.227581)
(xy 0.097575 -2.234306)
(xy 0.097575 -2.237711)
(xy 0.104275 -2.244411)
(xy 0.104275 -2.247761)
(xy 0.111 -2.254486)
(xy 0.111 -2.257836)
(xy 0.12105 -2.267886)
(xy 0.12105 -2.271236)
(xy 0.131125 -2.281311)
(xy 0.131125 -2.284661)
(xy 0.141175 -2.294711)
(xy 0.141175 -2.298086)
(xy 0.15795 -2.31486)
(xy 0.15795 -2.318211)
(xy 0.188125 -2.348385)
(xy 0.188125 -2.351736)
(xy 0.228659 -2.39227)
(xy 0.235343 -2.395599)
(xy 0.236287 -2.396547)
(xy 0.265532 -2.425793)
(xy 0.272271 -2.429162)
(xy 0.285629 -2.442553)
(xy 0.292399 -2.445938)
(xy 0.299125 -2.452639)
(xy 0.305739 -2.455946)
(xy 0.312466 -2.462698)
(xy 0.319234 -2.466069)
(xy 0.322557 -2.469393)
(xy 0.329257 -2.472743)
(xy 0.335984 -2.47947)
(xy 0.342624 -2.482777)
(xy 0.345947 -2.486125)
(xy 0.35275 -2.489526)
(xy 0.371769 -2.508475)
(xy 0.383957 -2.508475)
(xy 0.38628 -2.509641)
(xy 0.389632 -2.512993)
(xy 0.403623 -2.519988)
(xy 0.410381 -2.523355)
(xy 0.410381 -2.523354)
(xy 0.416881 -2.526604)
(xy 0.416881 -2.526605)
(xy 0.423681 -2.53003)
(xy 0.430506 -2.53343)
(xy 0.430506 -2.533429)
(xy 0.443873 -2.540113)
(xy 0.450631 -2.54348)
(xy 0.450631 -2.543479)
(xy 0.457131 -2.546729)
(xy 0.457131 -2.54673)
(xy 0.463931 -2.550155)
(xy 0.470756 -2.553555)
(xy 0.470756 -2.553554)
(xy 0.480568 -2.55846)
(xy 0.490648 -2.56182)
(xy 0.494231 -2.563605)
(xy 0.494231 -2.563604)
(xy 0.504043 -2.56851)
(xy 0.514054 -2.571847)
(xy 0.520782 -2.575224)
(xy 0.534239 -2.579709)
(xy 0.534274 -2.579721)
(xy 0.544349 -2.583071)
(xy 0.544314 -2.583059)
(xy 0.554364 -2.586409)
(xy 0.554399 -2.586421)
(xy 0.564474 -2.589771)
(xy 0.564439 -2.589759)
(xy 0.574418 -2.593085)
(xy 0.574418 -2.593086)
(xy 0.584528 -2.596472)
(xy 0.596446 -2.600445)
(xy 0.611744 -2.604262)
(xy 0.611723 -2.604257)
(xy 0.623279 -2.607146)
(xy 0.634492 -2.610874)
(xy 0.653089 -2.614588)
(xy 0.666491 -2.617964)
(xy 0.702047 -2.625064)
(xy 0.723537 -2.628642)
(xy 0.74802 -2.632135)
(xy 0.776286 -2.635665)
(xy 0.819632 -2.6393)
(xy 0.975159 -2.6393)
)
(stroke (width 0) (type default)) (fill solid) (layer "F.Cu") (tstamp b249901f-c2e9-4c38-852a-e2816130e93a))
)
(gr_rect (start 70 85) (end 74 89)
(stroke (width 0.1) (type default)) (fill none) (layer "Edge.Cuts") (tstamp 9d07ce53-5f7e-43a1-b557-4bcf5c8818ba))
)

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,537 @@
{
"board": {
"3dviewports": [],
"design_settings": {
"defaults": {
"board_outline_line_width": 0.09999999999999999,
"copper_line_width": 0.19999999999999998,
"copper_text_italic": false,
"copper_text_size_h": 1.5,
"copper_text_size_v": 1.5,
"copper_text_thickness": 0.3,
"copper_text_upright": false,
"courtyard_line_width": 0.049999999999999996,
"dimension_precision": 2,
"dimension_units": 2,
"dimensions": {
"arrow_length": 1270000,
"extension_offset": 500000,
"keep_text_aligned": true,
"suppress_zeroes": true,
"text_position": 1,
"units_format": 1
},
"fab_line_width": 0.09999999999999999,
"fab_text_italic": false,
"fab_text_size_h": 1.0,
"fab_text_size_v": 1.0,
"fab_text_thickness": 0.15,
"fab_text_upright": false,
"other_line_width": 0.09999999999999999,
"other_text_italic": false,
"other_text_size_h": 1.0,
"other_text_size_v": 1.0,
"other_text_thickness": 0.15,
"other_text_upright": false,
"pads": {
"drill": 0.0,
"height": 0.621,
"width": 0.95
},
"silk_line_width": 0.15,
"silk_text_italic": false,
"silk_text_size_h": 1.0,
"silk_text_size_v": 1.0,
"silk_text_thickness": 0.15,
"silk_text_upright": false,
"zones": {
"min_clearance": 0.16
}
},
"diff_pair_dimensions": [
{
"gap": 0.0,
"via_gap": 0.0,
"width": 0.0
},
{
"gap": 0.15,
"via_gap": 0.2,
"width": 0.2
}
],
"drc_exclusions": [],
"meta": {
"version": 2
},
"rule_severities": {
"annular_width": "error",
"clearance": "error",
"connection_width": "error",
"copper_edge_clearance": "error",
"copper_sliver": "error",
"courtyards_overlap": "warning",
"diff_pair_gap_out_of_range": "error",
"diff_pair_uncoupled_length_too_long": "error",
"drill_out_of_range": "error",
"duplicate_footprints": "warning",
"extra_footprint": "warning",
"footprint": "error",
"footprint_type_mismatch": "ignore",
"hole_clearance": "error",
"hole_near_hole": "error",
"invalid_outline": "error",
"isolated_copper": "warning",
"item_on_disabled_layer": "error",
"items_not_allowed": "error",
"length_out_of_range": "error",
"lib_footprint_issues": "warning",
"lib_footprint_mismatch": "ignore",
"malformed_courtyard": "ignore",
"microvia_drill_out_of_range": "error",
"missing_courtyard": "ignore",
"missing_footprint": "warning",
"net_conflict": "warning",
"npth_inside_courtyard": "ignore",
"padstack": "warning",
"pth_inside_courtyard": "ignore",
"shorting_items": "error",
"silk_edge_clearance": "ignore",
"silk_over_copper": "ignore",
"silk_overlap": "ignore",
"skew_out_of_range": "error",
"solder_mask_bridge": "ignore",
"starved_thermal": "error",
"text_height": "ignore",
"text_thickness": "ignore",
"through_hole_pad_without_hole": "error",
"too_many_vias": "error",
"track_dangling": "error",
"track_width": "error",
"tracks_crossing": "error",
"unconnected_items": "error",
"unresolved_variable": "error",
"via_dangling": "ignore",
"zones_intersect": "error"
},
"rules": {
"allow_blind_buried_vias": false,
"allow_microvias": false,
"max_error": 0.005,
"min_clearance": 0.16,
"min_connection": 0.0,
"min_copper_edge_clearance": 0.39999999999999997,
"min_hole_clearance": 0.254,
"min_hole_to_hole": 0.5,
"min_microvia_diameter": 0.39999999999999997,
"min_microvia_drill": 0.19999999999999998,
"min_resolved_spokes": 1,
"min_silk_clearance": 0.0,
"min_text_height": 0.6,
"min_text_thickness": 0.15,
"min_through_hole_diameter": 0.19999999999999998,
"min_track_width": 0.16,
"min_via_annular_width": 0.16,
"min_via_diameter": 0.39999999999999997,
"solder_mask_clearance": 0.0,
"solder_mask_min_width": 0.0,
"solder_mask_to_copper_clearance": 0.0,
"use_height_for_length_calcs": true
},
"teardrop_options": [
{
"td_allow_use_two_tracks": true,
"td_curve_segcount": 5,
"td_on_pad_in_zone": false,
"td_onpadsmd": true,
"td_onroundshapesonly": false,
"td_ontrackend": true,
"td_onviapad": true
}
],
"teardrop_parameters": [
{
"td_curve_segcount": 5,
"td_height_ratio": 1.0,
"td_length_ratio": 0.5,
"td_maxheight": 2.0,
"td_maxlen": 0.5,
"td_target_name": "td_round_shape",
"td_width_to_size_filter_ratio": 0.9
},
{
"td_curve_segcount": 5,
"td_height_ratio": 1.0,
"td_length_ratio": 0.5,
"td_maxheight": 2.0,
"td_maxlen": 0.5,
"td_target_name": "td_rect_shape",
"td_width_to_size_filter_ratio": 0.9
},
{
"td_curve_segcount": 5,
"td_height_ratio": 1.0,
"td_length_ratio": 0.5,
"td_maxheight": 2.0,
"td_maxlen": 0.5,
"td_target_name": "td_track_end",
"td_width_to_size_filter_ratio": 0.9
}
],
"track_widths": [
0.0,
0.2,
0.3,
0.4,
0.5,
0.6,
0.7,
0.8,
0.9,
1.0,
1.2,
1.5,
2.0,
2.5,
3.0,
4.0,
5.0
],
"via_dimensions": [
{
"diameter": 0.0,
"drill": 0.0
},
{
"diameter": 0.6,
"drill": 0.2
},
{
"diameter": 0.9,
"drill": 0.5
}
],
"zones_allow_external_fillets": false
},
"layer_presets": [],
"viewports": []
},
"boards": [],
"cvpcb": {
"equivalence_files": []
},
"erc": {
"erc_exclusions": [],
"meta": {
"version": 0
},
"pin_map": [
[
0,
0,
0,
0,
0,
0,
1,
0,
0,
0,
0,
2
],
[
0,
2,
0,
1,
0,
0,
1,
0,
2,
2,
2,
2
],
[
0,
0,
0,
0,
0,
0,
1,
0,
1,
0,
1,
2
],
[
0,
1,
0,
0,
0,
0,
1,
1,
2,
1,
1,
2
],
[
0,
0,
0,
0,
0,
0,
1,
0,
0,
0,
0,
2
],
[
0,
0,
0,
0,
0,
0,
0,
0,
0,
0,
0,
2
],
[
1,
1,
1,
1,
1,
0,
1,
1,
1,
1,
1,
2
],
[
0,
0,
0,
1,
0,
0,
1,
0,
0,
0,
0,
2
],
[
0,
2,
1,
2,
0,
0,
1,
0,
2,
2,
2,
2
],
[
0,
2,
0,
1,
0,
0,
1,
0,
2,
0,
0,
2
],
[
0,
2,
1,
1,
0,
0,
1,
0,
2,
0,
0,
2
],
[
2,
2,
2,
2,
2,
2,
2,
2,
2,
2,
2,
2
]
],
"rule_severities": {
"bus_definition_conflict": "error",
"bus_entry_needed": "error",
"bus_to_bus_conflict": "error",
"bus_to_net_conflict": "error",
"conflicting_netclasses": "error",
"different_unit_footprint": "error",
"different_unit_net": "error",
"duplicate_reference": "error",
"duplicate_sheet_names": "error",
"endpoint_off_grid": "warning",
"extra_units": "error",
"global_label_dangling": "warning",
"hier_label_mismatch": "error",
"label_dangling": "error",
"lib_symbol_issues": "warning",
"missing_bidi_pin": "warning",
"missing_input_pin": "warning",
"missing_power_pin": "error",
"missing_unit": "warning",
"multiple_net_names": "warning",
"net_not_bus_member": "warning",
"no_connect_connected": "warning",
"no_connect_dangling": "warning",
"pin_not_connected": "error",
"pin_not_driven": "error",
"pin_to_pin": "error",
"power_pin_not_driven": "error",
"similar_labels": "warning",
"simulation_model_issue": "ignore",
"unannotated": "error",
"unit_value_mismatch": "error",
"unresolved_variable": "error",
"wire_dangling": "error"
}
},
"libraries": {
"pinned_footprint_libs": [],
"pinned_symbol_libs": []
},
"meta": {
"filename": "W220921.01.kicad_pro",
"version": 1
},
"net_settings": {
"classes": [
{
"bus_width": 12,
"clearance": 0.15,
"diff_pair_gap": 0.15,
"diff_pair_via_gap": 0.25,
"diff_pair_width": 0.2,
"line_style": 0,
"microvia_diameter": 0.6,
"microvia_drill": 0.2,
"name": "Default",
"pcb_color": "rgba(0, 0, 0, 0.000)",
"schematic_color": "rgba(0, 0, 0, 0.000)",
"track_width": 0.2,
"via_diameter": 0.6,
"via_drill": 0.2,
"wire_width": 6
}
],
"meta": {
"version": 3
},
"net_colors": null,
"netclass_assignments": null,
"netclass_patterns": []
},
"pcbnew": {
"last_paths": {
"gencad": "",
"idf": "",
"netlist": "",
"specctra_dsn": "",
"step": "Assembly/Export STEP/W220921.01.step",
"vrml": ""
},
"page_layout_descr_file": "worksheets/A5_WiSER.kicad_wks"
},
"schematic": {
"annotate_start_num": 0,
"drawing": {
"dashed_lines_dash_length_ratio": 12.0,
"dashed_lines_gap_length_ratio": 3.0,
"default_line_thickness": 6.0,
"default_text_size": 40.0,
"field_names": [],
"intersheets_ref_own_page": false,
"intersheets_ref_prefix": "",
"intersheets_ref_short": false,
"intersheets_ref_show": false,
"intersheets_ref_suffix": "",
"junction_size_choice": 3,
"label_size_ratio": 0.375,
"pin_symbol_size": 25.0,
"text_offset_ratio": 0.15
},
"legacy_lib_dir": "",
"legacy_lib_list": [],
"meta": {
"version": 1
},
"net_format_name": "",
"ngspice": {
"fix_include_paths": true,
"fix_passive_vals": false,
"meta": {
"version": 0
},
"model_mode": 0,
"workbook_filename": ""
},
"page_layout_descr_file": "worksheets\\A4_WiSER.kicad_wks",
"plot_directory": "Documentation/",
"spice_adjust_passive_values": false,
"spice_current_sheet_as_root": false,
"spice_external_command": "spice \"%I\"",
"spice_model_current_sheet_as_root": true,
"spice_save_all_currents": false,
"spice_save_all_voltages": false,
"subpart_first_id": 65,
"subpart_id_separator": 0
},
"sheets": [
[
"3648648f-5acc-4713-a8ab-e21fc1644e2e",
""
]
],
"text_variables": {
"ProjectAuthor": "René Široký",
"ProjectDate": "21. 9. 2022",
"ProjectNumber": "W220921.01",
"ProjectTitle": "TMDSEMU110-U Debug Probe SWD Adapter",
"ProjectVariant": "Default"
}
}

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -0,0 +1,591 @@
{
"board": {
"3dviewports": [],
"design_settings": {
"defaults": {
"board_outline_line_width": 0.09999999999999999,
"copper_line_width": 0.19999999999999998,
"copper_text_italic": false,
"copper_text_size_h": 1.0,
"copper_text_size_v": 1.0,
"copper_text_thickness": 0.15,
"copper_text_upright": false,
"courtyard_line_width": 0.049999999999999996,
"dimension_precision": 4,
"dimension_units": 3,
"dimensions": {
"arrow_length": 1270000,
"extension_offset": 500000,
"keep_text_aligned": true,
"suppress_zeroes": false,
"text_position": 0,
"units_format": 1
},
"fab_line_width": 0.09999999999999999,
"fab_text_italic": false,
"fab_text_size_h": 1.0,
"fab_text_size_v": 1.0,
"fab_text_thickness": 0.15,
"fab_text_upright": false,
"other_line_width": 0.15,
"other_text_italic": false,
"other_text_size_h": 1.0,
"other_text_size_v": 1.0,
"other_text_thickness": 0.15,
"other_text_upright": false,
"pads": {
"drill": 2.0,
"height": 2.0,
"width": 2.0
},
"silk_line_width": 0.15,
"silk_text_italic": false,
"silk_text_size_h": 1.0,
"silk_text_size_v": 1.0,
"silk_text_thickness": 0.15,
"silk_text_upright": false,
"zones": {
"45_degree_only": false,
"min_clearance": 0.19999999999999998
}
},
"diff_pair_dimensions": [
{
"gap": 0.0,
"via_gap": 0.0,
"width": 0.0
}
],
"drc_exclusions": [],
"meta": {
"version": 2
},
"rule_severities": {
"annular_width": "error",
"clearance": "ignore",
"connection_width": "error",
"copper_edge_clearance": "error",
"copper_sliver": "error",
"courtyards_overlap": "error",
"diff_pair_gap_out_of_range": "error",
"diff_pair_uncoupled_length_too_long": "error",
"drill_out_of_range": "error",
"duplicate_footprints": "error",
"extra_footprint": "error",
"footprint": "error",
"footprint_type_mismatch": "error",
"hole_clearance": "error",
"hole_near_hole": "error",
"invalid_outline": "error",
"isolated_copper": "error",
"item_on_disabled_layer": "error",
"items_not_allowed": "error",
"length_out_of_range": "error",
"lib_footprint_issues": "ignore",
"lib_footprint_mismatch": "ignore",
"malformed_courtyard": "error",
"microvia_drill_out_of_range": "error",
"missing_courtyard": "error",
"missing_footprint": "error",
"net_conflict": "error",
"npth_inside_courtyard": "error",
"padstack": "error",
"pth_inside_courtyard": "error",
"shorting_items": "error",
"silk_edge_clearance": "error",
"silk_over_copper": "error",
"silk_overlap": "error",
"skew_out_of_range": "error",
"solder_mask_bridge": "error",
"starved_thermal": "error",
"text_height": "error",
"text_thickness": "error",
"through_hole_pad_without_hole": "error",
"too_many_vias": "error",
"track_dangling": "error",
"track_width": "error",
"tracks_crossing": "error",
"unconnected_items": "error",
"unresolved_variable": "error",
"via_dangling": "error",
"zones_intersect": "error"
},
"rules": {
"allow_blind_buried_vias": false,
"allow_microvias": false,
"max_error": 0.005,
"min_clearance": 0.19999999999999998,
"min_connection": 0.19999999999999998,
"min_copper_edge_clearance": 0.19999999999999998,
"min_hole_clearance": 0.19999999999999998,
"min_hole_to_hole": 0.19999999999999998,
"min_microvia_diameter": 0.19999999999999998,
"min_microvia_drill": 0.09999999999999999,
"min_resolved_spokes": 2,
"min_silk_clearance": 0.0,
"min_text_height": 0.7999999999999999,
"min_text_thickness": 0.12,
"min_through_hole_diameter": 0.19999999999999998,
"min_track_width": 0.19999999999999998,
"min_via_annular_width": 0.09999999999999999,
"min_via_diameter": 0.39999999999999997,
"solder_mask_clearance": 0.0,
"solder_mask_min_width": 0.0,
"solder_mask_to_copper_clearance": 0.0,
"use_height_for_length_calcs": true
},
"teardrop_options": [
{
"td_allow_use_two_tracks": true,
"td_curve_segcount": 5,
"td_on_pad_in_zone": false,
"td_onpadsmd": true,
"td_onroundshapesonly": false,
"td_ontrackend": false,
"td_onviapad": true
}
],
"teardrop_parameters": [
{
"td_curve_segcount": 0,
"td_height_ratio": 1.0,
"td_length_ratio": 0.5,
"td_maxheight": 2.0,
"td_maxlen": 1.0,
"td_target_name": "td_round_shape",
"td_width_to_size_filter_ratio": 0.9
},
{
"td_curve_segcount": 0,
"td_height_ratio": 1.0,
"td_length_ratio": 0.5,
"td_maxheight": 2.0,
"td_maxlen": 1.0,
"td_target_name": "td_rect_shape",
"td_width_to_size_filter_ratio": 0.9
},
{
"td_curve_segcount": 0,
"td_height_ratio": 1.0,
"td_length_ratio": 0.5,
"td_maxheight": 2.0,
"td_maxlen": 1.0,
"td_target_name": "td_track_end",
"td_width_to_size_filter_ratio": 0.9
}
],
"track_widths": [
0.0,
0.2,
0.3,
0.5,
0.8,
2.0
],
"via_dimensions": [
{
"diameter": 0.0,
"drill": 0.0
},
{
"diameter": 0.4,
"drill": 0.2
}
],
"zones_allow_external_fillets": true,
"zones_use_no_outline": true
},
"layer_presets": [],
"viewports": []
},
"boards": [],
"cvpcb": {
"equivalence_files": []
},
"erc": {
"erc_exclusions": [],
"meta": {
"version": 0
},
"pin_map": [
[
0,
0,
0,
0,
0,
0,
1,
0,
0,
0,
0,
2
],
[
0,
2,
0,
1,
0,
0,
1,
0,
2,
2,
2,
2
],
[
0,
0,
0,
0,
0,
0,
1,
0,
1,
0,
1,
2
],
[
0,
1,
0,
0,
0,
0,
1,
1,
2,
1,
1,
2
],
[
0,
0,
0,
0,
0,
0,
1,
0,
0,
0,
0,
2
],
[
0,
0,
0,
0,
0,
0,
0,
0,
0,
0,
0,
2
],
[
1,
1,
1,
1,
1,
0,
1,
1,
1,
1,
1,
2
],
[
0,
0,
0,
1,
0,
0,
1,
0,
0,
0,
0,
2
],
[
0,
2,
1,
2,
0,
0,
1,
0,
2,
2,
2,
2
],
[
0,
2,
0,
1,
0,
0,
1,
0,
2,
0,
0,
2
],
[
0,
2,
1,
1,
0,
0,
1,
0,
2,
0,
0,
2
],
[
2,
2,
2,
2,
2,
2,
2,
2,
2,
2,
2,
2
]
],
"rule_severities": {
"bus_definition_conflict": "error",
"bus_entry_needed": "error",
"bus_to_bus_conflict": "error",
"bus_to_net_conflict": "error",
"conflicting_netclasses": "error",
"different_unit_footprint": "error",
"different_unit_net": "error",
"duplicate_reference": "error",
"duplicate_sheet_names": "error",
"endpoint_off_grid": "error",
"extra_units": "error",
"global_label_dangling": "error",
"hier_label_mismatch": "error",
"label_dangling": "error",
"lib_symbol_issues": "error",
"missing_bidi_pin": "error",
"missing_input_pin": "error",
"missing_power_pin": "error",
"missing_unit": "error",
"multiple_net_names": "error",
"net_not_bus_member": "error",
"no_connect_connected": "error",
"no_connect_dangling": "error",
"pin_not_connected": "error",
"pin_not_driven": "error",
"pin_to_pin": "error",
"power_pin_not_driven": "error",
"similar_labels": "error",
"simulation_model_issue": "error",
"unannotated": "error",
"unit_value_mismatch": "error",
"unresolved_variable": "error",
"wire_dangling": "error"
}
},
"libraries": {
"pinned_footprint_libs": [],
"pinned_symbol_libs": []
},
"meta": {
"filename": "pulse.kicad_pro",
"version": 1
},
"net_settings": {
"classes": [
{
"bus_width": 12,
"clearance": 0.2,
"diff_pair_gap": 0.25,
"diff_pair_via_gap": 0.25,
"diff_pair_width": 0.2,
"line_style": 0,
"microvia_diameter": 0.3,
"microvia_drill": 0.1,
"name": "Default",
"pcb_color": "rgba(0, 0, 0, 0.000)",
"schematic_color": "rgba(0, 0, 0, 0.000)",
"track_width": 0.2,
"via_diameter": 0.4,
"via_drill": 0.2,
"wire_width": 6
}
],
"meta": {
"version": 3
},
"net_colors": null,
"netclass_assignments": null,
"netclass_patterns": []
},
"pcbnew": {
"last_paths": {
"gencad": "",
"idf": "",
"netlist": "",
"specctra_dsn": "",
"step": "",
"vrml": ""
},
"page_layout_descr_file": "${KICAD_USER_TEMPLATE_DIR}/V2_A4_Empty.kicad_wks"
},
"schematic": {
"annotate_start_num": 0,
"drawing": {
"dashed_lines_dash_length_ratio": 12.0,
"dashed_lines_gap_length_ratio": 3.0,
"default_line_thickness": 6.0,
"default_text_size": 50.0,
"field_names": [],
"intersheets_ref_own_page": false,
"intersheets_ref_prefix": "",
"intersheets_ref_short": false,
"intersheets_ref_show": false,
"intersheets_ref_suffix": "",
"junction_size_choice": 3,
"label_size_ratio": 0.375,
"pin_symbol_size": 25.0,
"text_offset_ratio": 0.1
},
"legacy_lib_dir": "",
"legacy_lib_list": [],
"meta": {
"version": 1
},
"net_format_name": "",
"ngspice": {
"fix_include_paths": true,
"fix_passive_vals": false,
"meta": {
"version": 0
},
"model_mode": 0,
"workbook_filename": ""
},
"page_layout_descr_file": "${KICAD_USER_TEMPLATE_DIR}/V2_A4.kicad_wks",
"plot_directory": "",
"spice_adjust_passive_values": false,
"spice_current_sheet_as_root": false,
"spice_external_command": "spice \"%I\"",
"spice_model_current_sheet_as_root": true,
"spice_save_all_currents": false,
"spice_save_all_voltages": false,
"subpart_first_id": 65,
"subpart_id_separator": 0
},
"sheets": [
[
"6c8448b4-b04d-47e1-934e-e40cbe27a7be",
""
],
[
"ecb8eef8-bb26-4217-a3d3-147bd9b06112",
"PWM Channel 1"
],
[
"c0032080-b223-4ef5-af6e-f10445519ca7",
"PWM Channel 2"
],
[
"72fc00ec-a037-43b2-8be3-f6274b0c3d34",
"PWM Channel 3"
],
[
"c5e21047-c711-42c5-833c-f87be56058f4",
"PWM Channel 4"
],
[
"a89ccf9d-6e1e-42ce-a365-b8a588b2a764",
"PWM Channel 5"
],
[
"2e65de88-75bc-4218-bada-fc991c11693d",
"PWM Channel 6"
],
[
"a178fee9-8654-4c6d-afdd-d0f611515c27",
"PWM Channel 7"
],
[
"c39b559a-eba7-416b-9b64-29e457c350f9",
"PWM Channel 8"
],
[
"f374a830-6190-4414-97f7-cd95a96fcb7b",
"PWM Channel 9"
],
[
"f929036c-a21a-4759-9a71-4bb346b616ac",
"PWM Channel 12"
],
[
"11be0dca-9aa7-44a3-a97c-59531fd35a86",
"PWM Channel 11"
],
[
"0c49b5ee-ddea-43ea-ab5a-6c52ceb8d790",
"PWM Channel 10"
],
[
"87daeacc-2e6b-4f3e-a4a8-59a3ffd747c2",
"PWM Channel 13"
],
[
"1b24bac2-35ae-4e4e-8bc5-d88b5d418adf",
"PWM Channel 16"
],
[
"02d7dc07-d88b-46a9-8c27-5f474f94bba5",
"PWM Channel 15"
],
[
"ef3f4e24-65f6-49ab-9666-2948510217b4",
"PWM Channel 14"
],
[
"ba748654-beb9-4861-aa35-c19e08caaf2e",
"Power +24V"
],
[
"c6db2f80-f80f-4104-8ca1-02f0ab85cfea",
"Link Plug"
],
[
"2a03ea79-269e-4abf-833c-000ba0ff13d7",
"Link Socket"
]
],
"text_variables": {
"Order-Number": "JLCJLCJLCJLC"
}
}

View File

@ -0,0 +1,115 @@
(kicad_pcb (version 20230410) (generator pcbnew)
(general
(thickness 1.6)
)
(paper "A4")
(layers
(0 "F.Cu" signal)
(31 "B.Cu" signal)
(32 "B.Adhes" user "B.Adhesive")
(33 "F.Adhes" user "F.Adhesive")
(34 "B.Paste" user)
(35 "F.Paste" user)
(36 "B.SilkS" user "B.Silkscreen")
(37 "F.SilkS" user "F.Silkscreen")
(38 "B.Mask" user)
(39 "F.Mask" user)
(40 "Dwgs.User" user "User.Drawings")
(41 "Cmts.User" user "User.Comments")
(42 "Eco1.User" user "User.Eco1")
(43 "Eco2.User" user "User.Eco2")
(44 "Edge.Cuts" user)
(45 "Margin" user)
(46 "B.CrtYd" user "B.Courtyard")
(47 "F.CrtYd" user "F.Courtyard")
(48 "B.Fab" user)
(49 "F.Fab" user)
(50 "User.1" user)
(51 "User.2" user)
(52 "User.3" user)
(53 "User.4" user)
(54 "User.5" user)
(55 "User.6" user)
(56 "User.7" user)
(57 "User.8" user)
(58 "User.9" user)
)
(setup
(pad_to_mask_clearance 0)
(pcbplotparams
(layerselection 0x00010fc_ffffffff)
(plot_on_all_layers_selection 0x0000000_00000000)
(disableapertmacros false)
(usegerberextensions false)
(usegerberattributes true)
(usegerberadvancedattributes true)
(creategerberjobfile true)
(dashed_line_dash_ratio 12.000000)
(dashed_line_gap_ratio 3.000000)
(svgprecision 4)
(plotframeref false)
(viasonmask false)
(mode 1)
(useauxorigin false)
(hpglpennumber 1)
(hpglpenspeed 20)
(hpglpendiameter 15.000000)
(dxfpolygonmode true)
(dxfimperialunits true)
(dxfusepcbnewfont true)
(psnegative false)
(psa4output false)
(plotreference true)
(plotvalue true)
(plotinvisibletext false)
(sketchpadsonfab false)
(subtractmaskfromsilk false)
(outputformat 1)
(mirror false)
(drillshape 1)
(scaleselection 1)
(outputdirectory "")
)
)
(net 0 "")
(gr_poly
(pts
(xy 101.9556 69.1388)
(xy 133.096 69.2404)
(xy 133.2484 71.1708)
(xy 102.0572 71.1708)
(xy 102.0572 70.2564)
(xy 132.4356 70.2056)
(xy 102.0064 70.0024)
)
(stroke (width 0) (type solid)) (fill solid) (layer "F.Cu") (tstamp 53555c8c-8370-4dd2-82cd-fb5fcf7d0670))
(gr_poly
(pts
(xy 102.2096 76.5556)
(xy 133.4008 76.6064)
(xy 133.4008 74.3204)
(xy 102.2096 74.0664)
(xy 102.2096 75.184)
(xy 132.1308 75.438)
(xy 102.2096 75.7936)
)
(stroke (width 0) (type solid)) (fill solid) (layer "F.Cu") (tstamp 9e84cf11-135c-4d2a-a849-6dc335827c5c))
(gr_poly
(pts
(xy 102.2096 72.7964)
(xy 133.0452 73.152)
(xy 102.2096 73.2028)
(xy 101.9048 73.1012)
(xy 101.9048 72.898)
(xy 102.0572 72.7964)
)
(stroke (width 0) (type solid)) (fill solid) (layer "F.Cu") (tstamp ceafb5a5-949b-47d2-af13-6d5cb962e056))
(gr_rect (start 99.06 67.7164) (end 136.144 77.1652)
(stroke (width 0.1) (type default)) (fill none) (layer "Edge.Cuts") (tstamp 2be69152-e65e-49cc-b325-c520ff486dc0))
)

View File

@ -47,6 +47,7 @@ set( QA_PCBNEW_SRCS
drc/test_drc_courtyard_overlap.cpp drc/test_drc_courtyard_overlap.cpp
drc/test_drc_regressions.cpp drc/test_drc_regressions.cpp
drc/test_drc_copper_conn.cpp drc/test_drc_copper_conn.cpp
drc/test_drc_copper_sliver.cpp
drc/test_solder_mask_bridging.cpp drc/test_solder_mask_bridging.cpp
plugins/altium/test_altium_rule_transformer.cpp plugins/altium/test_altium_rule_transformer.cpp

View File

@ -0,0 +1,106 @@
/*
* This program source code file is part of KiCad, a free EDA CAD application.
*
* Copyright (C) 2023 KiCad Developers, see AUTHORS.txt for contributors.
*
* 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
*/
#include <qa_utils/wx_utils/unit_test_utils.h>
#include <pcbnew_utils/board_test_utils.h>
#include <board.h>
#include <board_design_settings.h>
#include <pad.h>
#include <pcb_track.h>
#include <pcb_marker.h>
#include <footprint.h>
#include <drc/drc_item.h>
#include <settings/settings_manager.h>
struct DRC_REGRESSION_TEST_FIXTURE
{
DRC_REGRESSION_TEST_FIXTURE() :
m_settingsManager( true /* headless */ )
{ }
SETTINGS_MANAGER m_settingsManager;
std::unique_ptr<BOARD> m_board;
};
BOOST_FIXTURE_TEST_CASE( DRCCopperSliver, DRC_REGRESSION_TEST_FIXTURE )
{
// Check for minimum copper connection errors
std::vector<std::pair<wxString, int>> tests =
{
{ "sliver", 1 },
{ "issue14449", 0 },
{ "issue14549", 0 },
{ "issue14549_2", 0 },
{ "issue14559", 0 }
};
for( const std::pair<wxString, int>& test : tests )
{
KI_TEST::LoadBoard( m_settingsManager, test.first, m_board );
KI_TEST::FillZones( m_board.get() );
std::vector<DRC_ITEM> violations;
BOARD_DESIGN_SETTINGS& bds = m_board->GetDesignSettings();
// Disable DRC tests not useful or not handled in this testcase
for( int ii = DRCE_FIRST; ii <= DRCE_LAST; ++ii )
bds.m_DRCSeverities[ ii ] = SEVERITY::RPT_SEVERITY_IGNORE;
// Ensure that our desired error is fired
bds.m_DRCSeverities[ DRCE_COPPER_SLIVER ] = SEVERITY::RPT_SEVERITY_ERROR;
bds.m_DRCEngine->SetViolationHandler(
[&]( const std::shared_ptr<DRC_ITEM>& aItem, VECTOR2I aPos, int aLayer )
{
if( bds.GetSeverity( aItem->GetErrorCode() ) == SEVERITY::RPT_SEVERITY_ERROR )
violations.push_back( *aItem );
} );
bds.m_DRCEngine->RunTests( EDA_UNITS::MILLIMETRES, true, false );
if( violations.size() == test.second )
{
BOOST_CHECK_EQUAL( 1, 1 ); // quiet "did not check any assertions" warning
BOOST_TEST_MESSAGE( wxString::Format( "DRC copper sliver: %s, passed", test.first ) );
}
else
{
UNITS_PROVIDER unitsProvider( pcbIUScale, EDA_UNITS::INCHES );
std::map<KIID, EDA_ITEM*> itemMap;
m_board->FillItemMap( itemMap );
for( const DRC_ITEM& item : violations )
{
BOOST_TEST_MESSAGE( item.ShowReport( &unitsProvider, RPT_SEVERITY_ERROR,
itemMap ) );
}
BOOST_ERROR( wxString::Format( "DRC copper sliver: %s, failed (violations found %d expected %d)",
test.first, (int)violations.size(), test.second ) );
}
}
}

View File

@ -1,6 +1,6 @@
/******************************************************************************* /*******************************************************************************
* Author : Angus Johnson * * Author : Angus Johnson *
* Date : 22 March 2023 * * Date : 8 April 2023 *
* Website : http://www.angusj.com * * Website : http://www.angusj.com *
* Copyright : Angus Johnson 2010-2023 * * Copyright : Angus Johnson 2010-2023 *
* Purpose : Core Clipper Library structures and functions * * Purpose : Core Clipper Library structures and functions *
@ -50,7 +50,9 @@ namespace Clipper2Lib
const int non_pair_error_i = 4; // non-fatal const int non_pair_error_i = 4; // non-fatal
const int range_error_i = 64; const int range_error_i = 64;
#ifndef PI
static const double PI = 3.141592653589793238; static const double PI = 3.141592653589793238;
#endif
static const int64_t MAX_COORD = INT64_MAX >> 2; static const int64_t MAX_COORD = INT64_MAX >> 2;
static const int64_t MIN_COORD = -MAX_COORD; static const int64_t MIN_COORD = -MAX_COORD;
static const int64_t INVALID = INT64_MAX; static const int64_t INVALID = INT64_MAX;
@ -558,38 +560,22 @@ namespace Clipper2Lib
} }
template<typename T> template<typename T>
inline Path<T> StripDuplicates(const Path<T>& path, bool is_closed_path) inline void StripDuplicates( Path<T>& path, bool is_closed_path)
{ {
if (path.size() == 0) return Path<T>(); //https://stackoverflow.com/questions/1041620/whats-the-most-efficient-way-to-erase-duplicates-and-sort-a-vector#:~:text=Let%27s%20compare%20three%20approaches%3A
Path<T> result; path.erase(std::unique(path.begin(), path.end()),path.end());
result.reserve(path.size()); if (is_closed_path)
typename Path<T>::const_iterator path_iter = path.cbegin(); while (path.size() > 1 && path.back() == path.front()) path.pop_back();
Point<T> first_pt = *path_iter++, last_pt = first_pt;
result.push_back(first_pt);
for (; path_iter != path.cend(); ++path_iter)
{
if (*path_iter != last_pt)
{
last_pt = *path_iter;
result.push_back(last_pt);
}
}
if (!is_closed_path) return result;
while (result.size() > 1 && result.back() == first_pt) result.pop_back();
return result;
} }
template<typename T> template<typename T>
inline Paths<T> StripDuplicates(const Paths<T>& paths, bool is_closed_path) inline void StripDuplicates( Paths<T>& paths, bool is_closed_path)
{ {
Paths<T> result; for (typename Paths<T>::iterator paths_citer = paths.begin();
result.reserve(paths.size()); paths_citer != paths.end(); ++paths_citer)
for (typename Paths<T>::const_iterator paths_citer = paths.cbegin();
paths_citer != paths.cend(); ++paths_citer)
{ {
result.push_back(StripDuplicates(*paths_citer, is_closed_path)); StripDuplicates(*paths_citer, is_closed_path);
} }
return result;
} }
// Miscellaneous ------------------------------------------------------------ // Miscellaneous ------------------------------------------------------------

View File

@ -1,6 +1,6 @@
/******************************************************************************* /*******************************************************************************
* Author : Angus Johnson * * Author : Angus Johnson *
* Date : 2 March 2023 * * Date : 22 April 2023 *
* Website : http://www.angusj.com * * Website : http://www.angusj.com *
* Copyright : Angus Johnson 2010-2023 * * Copyright : Angus Johnson 2010-2023 *
* Purpose : This is the main polygon clipping module * * Purpose : This is the main polygon clipping module *
@ -10,7 +10,7 @@
#ifndef CLIPPER_ENGINE_H #ifndef CLIPPER_ENGINE_H
#define CLIPPER_ENGINE_H #define CLIPPER_ENGINE_H
constexpr auto CLIPPER2_VERSION = "1.2.1"; constexpr auto CLIPPER2_VERSION = "1.2.2";
#include <cstdlib> #include <cstdlib>
#include <iostream> #include <iostream>
@ -260,8 +260,8 @@ namespace Clipper2Lib {
bool ExecuteInternal(ClipType ct, FillRule ft, bool use_polytrees); bool ExecuteInternal(ClipType ct, FillRule ft, bool use_polytrees);
void CleanCollinear(OutRec* outrec); void CleanCollinear(OutRec* outrec);
bool CheckBounds(OutRec* outrec); bool CheckBounds(OutRec* outrec);
bool CheckSplitOwner(OutRec* outrec);
void RecursiveCheckOwners(OutRec* outrec, PolyPath* polypath); void RecursiveCheckOwners(OutRec* outrec, PolyPath* polypath);
void DeepCheckOwners(OutRec* outrec, PolyPath* polypath);
#ifdef USINGZ #ifdef USINGZ
ZCallback64 zCallback_ = nullptr; ZCallback64 zCallback_ = nullptr;
void SetZ(const Active& e1, const Active& e2, Point64& pt); void SetZ(const Active& e1, const Active& e2, Point64& pt);

View File

@ -1,6 +1,6 @@
/******************************************************************************* /*******************************************************************************
* Author : Angus Johnson * * Author : Angus Johnson *
* Date : 22 March 2023 * * Date : 23 March 2023 *
* Website : http://www.angusj.com * * Website : http://www.angusj.com *
* Copyright : Angus Johnson 2010-2023 * * Copyright : Angus Johnson 2010-2023 *
* Purpose : This module exports the Clipper2 Library (ie DLL/so) * * Purpose : This module exports the Clipper2 Library (ie DLL/so) *
@ -157,14 +157,14 @@ EXTERN_DLL_EXPORT CPathsD InflatePathsD(const CPathsD paths,
int precision = 2, double miter_limit = 2.0, int precision = 2, double miter_limit = 2.0,
double arc_tolerance = 0.0, bool reverse_solution = false); double arc_tolerance = 0.0, bool reverse_solution = false);
// RectClip & RectClipLines: // ExecuteRectClip & ExecuteRectClipLines:
EXTERN_DLL_EXPORT CPaths64 RectClip64(const CRect64& rect, EXTERN_DLL_EXPORT CPaths64 ExecuteRectClip64(const CRect64& rect,
const CPaths64 paths, bool convex_only = false); const CPaths64 paths, bool convex_only = false);
EXTERN_DLL_EXPORT CPathsD RectClipD(const CRectD& rect, EXTERN_DLL_EXPORT CPathsD ExecuteRectClipD(const CRectD& rect,
const CPathsD paths, int precision = 2, bool convex_only = false); const CPathsD paths, int precision = 2, bool convex_only = false);
EXTERN_DLL_EXPORT CPaths64 RectClipLines64(const CRect64& rect, EXTERN_DLL_EXPORT CPaths64 ExecuteRectClipLines64(const CRect64& rect,
const CPaths64 paths); const CPaths64 paths);
EXTERN_DLL_EXPORT CPathsD RectClipLinesD(const CRectD& rect, EXTERN_DLL_EXPORT CPathsD ExecuteRectClipLinesD(const CRectD& rect,
const CPathsD paths, int precision = 2); const CPathsD paths, int precision = 2);
////////////////////////////////////////////////////// //////////////////////////////////////////////////////
@ -381,7 +381,7 @@ EXTERN_DLL_EXPORT CPathsD InflatePathsD(const CPathsD paths,
return CreateCPathsD(result, 1/scale); return CreateCPathsD(result, 1/scale);
} }
EXTERN_DLL_EXPORT CPaths64 RectClip64(const CRect64& rect, EXTERN_DLL_EXPORT CPaths64 ExecuteRectClip64(const CRect64& rect,
const CPaths64 paths, bool convex_only) const CPaths64 paths, bool convex_only)
{ {
if (CRectIsEmpty(rect) || !paths) return nullptr; if (CRectIsEmpty(rect) || !paths) return nullptr;
@ -392,7 +392,7 @@ EXTERN_DLL_EXPORT CPaths64 RectClip64(const CRect64& rect,
return CreateCPaths64(result); return CreateCPaths64(result);
} }
EXTERN_DLL_EXPORT CPathsD RectClipD(const CRectD& rect, EXTERN_DLL_EXPORT CPathsD ExecuteRectClipD(const CRectD& rect,
const CPathsD paths, int precision, bool convex_only) const CPathsD paths, int precision, bool convex_only)
{ {
if (CRectIsEmpty(rect) || !paths) return nullptr; if (CRectIsEmpty(rect) || !paths) return nullptr;
@ -407,7 +407,7 @@ EXTERN_DLL_EXPORT CPathsD RectClipD(const CRectD& rect,
return CreateCPathsD(result, 1/scale); return CreateCPathsD(result, 1/scale);
} }
EXTERN_DLL_EXPORT CPaths64 RectClipLines64(const CRect64& rect, EXTERN_DLL_EXPORT CPaths64 ExecuteRectClipLines64(const CRect64& rect,
const CPaths64 paths) const CPaths64 paths)
{ {
if (CRectIsEmpty(rect) || !paths) return nullptr; if (CRectIsEmpty(rect) || !paths) return nullptr;
@ -418,7 +418,7 @@ EXTERN_DLL_EXPORT CPaths64 RectClipLines64(const CRect64& rect,
return CreateCPaths64(result); return CreateCPaths64(result);
} }
EXTERN_DLL_EXPORT CPathsD RectClipLinesD(const CRectD& rect, EXTERN_DLL_EXPORT CPathsD ExecuteRectClipLinesD(const CRectD& rect,
const CPathsD paths, int precision) const CPathsD paths, int precision)
{ {
if (CRectIsEmpty(rect) || !paths) return nullptr; if (CRectIsEmpty(rect) || !paths) return nullptr;

View File

@ -1,6 +1,6 @@
/******************************************************************************* /*******************************************************************************
* Author : Angus Johnson * * Author : Angus Johnson *
* Date : 9 February 2023 * * Date : 21 April 2023 *
* Website : http://www.angusj.com * * Website : http://www.angusj.com *
* Copyright : Angus Johnson 2010-2023 * * Copyright : Angus Johnson 2010-2023 *
* Purpose : This module provides a simple interface to the Clipper Library * * Purpose : This module provides a simple interface to the Clipper Library *
@ -198,23 +198,23 @@ namespace Clipper2Lib {
} }
inline Paths64 ExecuteRectClip(const Rect64& rect, inline Paths64 ExecuteRectClip(const Rect64& rect,
const Paths64& paths, bool convex_only = false) const Paths64& paths, bool convex_only)
{ {
if (rect.IsEmpty() || paths.empty()) return Paths64(); if (rect.IsEmpty() || paths.empty()) return Paths64();
class RectClip rc(rect); RectClip rc(rect);
return rc.Execute(paths, convex_only); return rc.Execute(paths, convex_only);
} }
inline Paths64 ExecuteRectClip(const Rect64& rect, inline Paths64 ExecuteRectClip(const Rect64& rect,
const Path64& path, bool convex_only = false) const Path64& path, bool convex_only)
{ {
if (rect.IsEmpty() || path.empty()) return Paths64(); if (rect.IsEmpty() || path.empty()) return Paths64();
class RectClip rc(rect); RectClip rc(rect);
return rc.Execute(Paths64{ path }, convex_only); return rc.Execute(Paths64{ path }, convex_only);
} }
inline PathsD ExecuteRectClip(const RectD& rect, inline PathsD ExecuteRectClip(const RectD& rect,
const PathsD& paths, bool convex_only = false, int precision = 2) const PathsD& paths, bool convex_only, int precision = 2)
{ {
if (rect.IsEmpty() || paths.empty()) return PathsD(); if (rect.IsEmpty() || paths.empty()) return PathsD();
int error_code = 0; int error_code = 0;
@ -222,7 +222,7 @@ namespace Clipper2Lib {
if (error_code) return PathsD(); if (error_code) return PathsD();
const double scale = std::pow(10, precision); const double scale = std::pow(10, precision);
Rect64 r = ScaleRect<int64_t, double>(rect, scale); Rect64 r = ScaleRect<int64_t, double>(rect, scale);
class RectClip rc(r); RectClip rc(r);
Paths64 pp = ScalePaths<int64_t, double>(paths, scale, error_code); Paths64 pp = ScalePaths<int64_t, double>(paths, scale, error_code);
if (error_code) return PathsD(); // ie: error_code result is lost if (error_code) return PathsD(); // ie: error_code result is lost
return ScalePaths<double, int64_t>( return ScalePaths<double, int64_t>(
@ -230,7 +230,7 @@ namespace Clipper2Lib {
} }
inline PathsD ExecuteRectClip(const RectD& rect, inline PathsD ExecuteRectClip(const RectD& rect,
const PathD& path, bool convex_only = false, int precision = 2) const PathD& path, bool convex_only, int precision = 2)
{ {
return ExecuteRectClip(rect, PathsD{ path }, convex_only, precision); return ExecuteRectClip(rect, PathsD{ path }, convex_only, precision);
} }
@ -238,7 +238,7 @@ namespace Clipper2Lib {
inline Paths64 ExecuteRectClipLines(const Rect64& rect, const Paths64& lines) inline Paths64 ExecuteRectClipLines(const Rect64& rect, const Paths64& lines)
{ {
if (rect.IsEmpty() || lines.empty()) return Paths64(); if (rect.IsEmpty() || lines.empty()) return Paths64();
class RectClipLines rcl(rect); RectClipLines rcl(rect);
return rcl.Execute(lines); return rcl.Execute(lines);
} }
@ -247,11 +247,6 @@ namespace Clipper2Lib {
return ExecuteRectClipLines(rect, Paths64{ line }); return ExecuteRectClipLines(rect, Paths64{ line });
} }
inline PathsD ExecuteRectClipLines(const RectD& rect, const PathD& line, int precision = 2)
{
return ExecuteRectClip(rect, PathsD{ line }, precision);
}
inline PathsD ExecuteRectClipLines(const RectD& rect, const PathsD& lines, int precision = 2) inline PathsD ExecuteRectClipLines(const RectD& rect, const PathsD& lines, int precision = 2)
{ {
if (rect.IsEmpty() || lines.empty()) return PathsD(); if (rect.IsEmpty() || lines.empty()) return PathsD();
@ -260,13 +255,18 @@ namespace Clipper2Lib {
if (error_code) return PathsD(); if (error_code) return PathsD();
const double scale = std::pow(10, precision); const double scale = std::pow(10, precision);
Rect64 r = ScaleRect<int64_t, double>(rect, scale); Rect64 r = ScaleRect<int64_t, double>(rect, scale);
class RectClipLines rcl(r); RectClipLines rcl(r);
Paths64 p = ScalePaths<int64_t, double>(lines, scale, error_code); Paths64 p = ScalePaths<int64_t, double>(lines, scale, error_code);
if (error_code) return PathsD(); if (error_code) return PathsD();
p = rcl.Execute(p); p = rcl.Execute(p);
return ScalePaths<double, int64_t>(p, 1 / scale, error_code); return ScalePaths<double, int64_t>(p, 1 / scale, error_code);
} }
inline PathsD ExecuteRectClipLines(const RectD& rect, const PathD& line, int precision = 2)
{
return ExecuteRectClipLines(rect, PathsD{ line }, precision);
}
namespace details namespace details
{ {
@ -370,6 +370,7 @@ namespace Clipper2Lib {
inline std::ostream& operator<< (std::ostream& os, const PolyTree64& pp) inline std::ostream& operator<< (std::ostream& os, const PolyTree64& pp)
{ {
os << std::endl << "Polytree root" << std::endl;
PolyPath64List::const_iterator it = pp.begin(); PolyPath64List::const_iterator it = pp.begin();
for (; it < pp.end() - 1; ++it) for (; it < pp.end() - 1; ++it)
details::OutlinePolyPath64(os, **it, " ", false); details::OutlinePolyPath64(os, **it, " ", false);
@ -645,7 +646,7 @@ namespace Clipper2Lib {
template <typename T> template <typename T>
inline Path<T> SimplifyPath(const Path<T> path, inline Path<T> SimplifyPath(const Path<T> path,
double epsilon, bool isOpenPath = false) double epsilon, bool isClosedPath = true)
{ {
const size_t len = path.size(), high = len -1; const size_t len = path.size(), high = len -1;
const double epsSqr = Sqr(epsilon); const double epsSqr = Sqr(epsilon);
@ -654,16 +655,16 @@ namespace Clipper2Lib {
std::vector<bool> flags(len); std::vector<bool> flags(len);
std::vector<double> distSqr(len); std::vector<double> distSqr(len);
size_t prior = high, curr = 0, start, next, prior2, next2; size_t prior = high, curr = 0, start, next, prior2, next2;
if (isOpenPath) if (isClosedPath)
{
distSqr[0] = MAX_DBL;
distSqr[high] = MAX_DBL;
}
else
{ {
distSqr[0] = PerpendicDistFromLineSqrd(path[0], path[high], path[1]); distSqr[0] = PerpendicDistFromLineSqrd(path[0], path[high], path[1]);
distSqr[high] = PerpendicDistFromLineSqrd(path[high], path[0], path[high - 1]); distSqr[high] = PerpendicDistFromLineSqrd(path[high], path[0], path[high - 1]);
} }
else
{
distSqr[0] = MAX_DBL;
distSqr[high] = MAX_DBL;
}
for (size_t i = 1; i < high; ++i) for (size_t i = 1; i < high; ++i)
distSqr[i] = PerpendicDistFromLineSqrd(path[i], path[i - 1], path[i + 1]); distSqr[i] = PerpendicDistFromLineSqrd(path[i], path[i - 1], path[i + 1]);
@ -689,7 +690,7 @@ namespace Clipper2Lib {
next = GetNext(next, high, flags); next = GetNext(next, high, flags);
next2 = GetNext(next, high, flags); next2 = GetNext(next, high, flags);
distSqr[curr] = PerpendicDistFromLineSqrd(path[curr], path[prior], path[next]); distSqr[curr] = PerpendicDistFromLineSqrd(path[curr], path[prior], path[next]);
if (next != high || !isOpenPath) if (next != high || isClosedPath)
distSqr[next] = PerpendicDistFromLineSqrd(path[next], path[curr], path[next2]); distSqr[next] = PerpendicDistFromLineSqrd(path[next], path[curr], path[next2]);
curr = next; curr = next;
} }
@ -700,7 +701,7 @@ namespace Clipper2Lib {
next = GetNext(next, high, flags); next = GetNext(next, high, flags);
prior2 = GetPrior(prior, high, flags); prior2 = GetPrior(prior, high, flags);
distSqr[curr] = PerpendicDistFromLineSqrd(path[curr], path[prior], path[next]); distSqr[curr] = PerpendicDistFromLineSqrd(path[curr], path[prior], path[next]);
if (prior != 0 || !isOpenPath) if (prior != 0 || isClosedPath)
distSqr[prior] = PerpendicDistFromLineSqrd(path[prior], path[prior2], path[curr]); distSqr[prior] = PerpendicDistFromLineSqrd(path[prior], path[prior2], path[curr]);
} }
} }
@ -713,12 +714,12 @@ namespace Clipper2Lib {
template <typename T> template <typename T>
inline Paths<T> SimplifyPaths(const Paths<T> paths, inline Paths<T> SimplifyPaths(const Paths<T> paths,
double epsilon, bool isOpenPath = false) double epsilon, bool isClosedPath = true)
{ {
Paths<T> result; Paths<T> result;
result.reserve(paths.size()); result.reserve(paths.size());
for (const auto& path : paths) for (const auto& path : paths)
result.push_back(SimplifyPath(path, epsilon, isOpenPath)); result.push_back(SimplifyPath(path, epsilon, isClosedPath));
return result; return result;
} }

View File

@ -1,6 +1,6 @@
/******************************************************************************* /*******************************************************************************
* Author : Angus Johnson * * Author : Angus Johnson *
* Date : 19 March 2023 * * Date : 23 April 2023 *
* Website : http://www.angusj.com * * Website : http://www.angusj.com *
* Copyright : Angus Johnson 2010-2023 * * Copyright : Angus Johnson 2010-2023 *
* Purpose : This is the main polygon clipping module * * Purpose : This is the main polygon clipping module *
@ -594,6 +594,9 @@ namespace Clipper2Lib {
//for each path create a circular double linked list of vertices //for each path create a circular double linked list of vertices
Vertex* v0 = v, * curr_v = v, * prev_v = nullptr; Vertex* v0 = v, * curr_v = v, * prev_v = nullptr;
if (path.empty())
continue;
v->prev = nullptr; v->prev = nullptr;
int cnt = 0; int cnt = 0;
for (const Point64& pt : path) for (const Point64& pt : path)
@ -1533,7 +1536,7 @@ namespace Clipper2Lib {
InsertScanline(e->top.y); InsertScanline(e->top.y);
CheckJoinLeft(*e, e->bot); CheckJoinLeft(*e, e->bot);
CheckJoinRight(*e, e->bot); CheckJoinRight(*e, e->bot, true); // (#500)
} }
Active* FindEdgeWithMatchingLocMin(Active* e) Active* FindEdgeWithMatchingLocMin(Active* e)
@ -2124,8 +2127,12 @@ namespace Clipper2Lib {
else if (Path1InsidePath2(or1->pts, or2->pts)) else if (Path1InsidePath2(or1->pts, or2->pts))
SetOwner(or1, or2); SetOwner(or1, or2);
else else
{
if (!or1->splits) or1->splits = new OutRecList();
or1->splits->push_back(or2); //(#498)
or2->owner = or1; or2->owner = or1;
} }
}
else else
or2->owner = or1; or2->owner = or1;
@ -2636,9 +2643,9 @@ namespace Clipper2Lib {
{ {
Active* prev = e.prev_in_ael; Active* prev = e.prev_in_ael;
if (IsOpen(e) || !IsHotEdge(e) || !prev || if (IsOpen(e) || !IsHotEdge(e) || !prev ||
IsOpen(*prev) || !IsHotEdge(*prev) || IsOpen(*prev) || !IsHotEdge(*prev)) return;
pt.y < e.top.y + 2 || pt.y < prev->top.y + 2) // avoid trivial joins if ((pt.y < e.top.y + 2 || pt.y < prev->top.y + 2) &&
return; ((e.bot.y > pt.y) || (prev->bot.y > pt.y))) return; // avoid trivial joins
if (check_curr_x) if (check_curr_x)
{ {
@ -2662,9 +2669,9 @@ namespace Clipper2Lib {
{ {
Active* next = e.next_in_ael; Active* next = e.next_in_ael;
if (IsOpen(e) || !IsHotEdge(e) || if (IsOpen(e) || !IsHotEdge(e) ||
!next || IsOpen(*next) || !IsHotEdge(*next) || !next || IsOpen(*next) || !IsHotEdge(*next)) return;
pt.y < e.top.y +2 || pt.y < next->top.y +2) // avoids trivial joins if ((pt.y < e.top.y +2 || pt.y < next->top.y +2) &&
return; ((e.bot.y > pt.y) || (next->bot.y > pt.y))) return; // avoid trivial joins
if (check_curr_x) if (check_curr_x)
{ {
@ -2679,6 +2686,7 @@ namespace Clipper2Lib {
JoinOutrecPaths(e, *next); JoinOutrecPaths(e, *next);
else else
JoinOutrecPaths(*next, e); JoinOutrecPaths(*next, e);
e.join_with = JoinWith::Right; e.join_with = JoinWith::Right;
next->join_with = JoinWith::Left; next->join_with = JoinWith::Left;
} }
@ -2755,6 +2763,23 @@ namespace Clipper2Lib {
return true; return true;
} }
bool ClipperBase::CheckSplitOwner(OutRec* outrec)
{
for (auto s : *outrec->owner->splits)
{
OutRec* split = GetRealOutRec(s);
if (split && split != outrec &&
split != outrec->owner && CheckBounds(split) &&
split->bounds.Contains(outrec->bounds) &&
Path1InsidePath2(outrec->pts, split->pts))
{
outrec->owner = split; //found in split
return true;
}
}
return false;
}
void ClipperBase::RecursiveCheckOwners(OutRec* outrec, PolyPath* polypath) void ClipperBase::RecursiveCheckOwners(OutRec* outrec, PolyPath* polypath)
{ {
// pre-condition: outrec will have valid bounds // pre-condition: outrec will have valid bounds
@ -2762,52 +2787,25 @@ namespace Clipper2Lib {
if (outrec->polypath || outrec->bounds.IsEmpty()) return; if (outrec->polypath || outrec->bounds.IsEmpty()) return;
while (outrec->owner &&
(!outrec->owner->pts || !CheckBounds(outrec->owner)))
outrec->owner = outrec->owner->owner;
if (outrec->owner && !outrec->owner->polypath)
RecursiveCheckOwners(outrec->owner, polypath);
while (outrec->owner) while (outrec->owner)
if (outrec->owner->bounds.Contains(outrec->bounds) && {
Path1InsidePath2(outrec->pts, outrec->owner->pts)) if (outrec->owner->splits && CheckSplitOwner(outrec)) break;
break; // found - owner contain outrec! if (outrec->owner->pts && CheckBounds(outrec->owner) &&
else outrec->owner->bounds.Contains(outrec->bounds) &&
Path1InsidePath2(outrec->pts, outrec->owner->pts)) break;
outrec->owner = outrec->owner->owner; outrec->owner = outrec->owner->owner;
}
if (outrec->owner) if (outrec->owner)
{
if (!outrec->owner->polypath)
RecursiveCheckOwners(outrec->owner, polypath);
outrec->polypath = outrec->owner->polypath->AddChild(outrec->path); outrec->polypath = outrec->owner->polypath->AddChild(outrec->path);
}
else else
outrec->polypath = polypath->AddChild(outrec->path); outrec->polypath = polypath->AddChild(outrec->path);
} }
void ClipperBase::DeepCheckOwners(OutRec* outrec, PolyPath* polypath)
{
RecursiveCheckOwners(outrec, polypath);
while (outrec->owner && outrec->owner->splits)
{
OutRec* split = nullptr;
for (auto s : *outrec->owner->splits)
{
split = GetRealOutRec(s);
if (split && split != outrec &&
split != outrec->owner && CheckBounds(split) &&
split->bounds.Contains(outrec->bounds) &&
Path1InsidePath2(outrec->pts, split->pts))
{
RecursiveCheckOwners(split, polypath);
outrec->owner = split; //found in split
break; // inner 'for' loop
}
else
split = nullptr;
}
if (!split) break;
}
}
void Clipper64::BuildPaths64(Paths64& solutionClosed, Paths64* solutionOpen) void Clipper64::BuildPaths64(Paths64& solutionClosed, Paths64* solutionOpen)
{ {
solutionClosed.resize(0); solutionClosed.resize(0);
@ -2866,7 +2864,7 @@ namespace Clipper2Lib {
} }
if (CheckBounds(outrec)) if (CheckBounds(outrec))
DeepCheckOwners(outrec, &polytree); RecursiveCheckOwners(outrec, &polytree);
} }
} }
@ -2969,7 +2967,7 @@ namespace Clipper2Lib {
} }
if (CheckBounds(outrec)) if (CheckBounds(outrec))
DeepCheckOwners(outrec, &polytree); RecursiveCheckOwners(outrec, &polytree);
} }
} }

View File

@ -1,6 +1,6 @@
/******************************************************************************* /*******************************************************************************
* Author : Angus Johnson * * Author : Angus Johnson *
* Date : 22 March 2023 * * Date : 8 April 2023 *
* Website : http://www.angusj.com * * Website : http://www.angusj.com *
* Copyright : Angus Johnson 2010-2023 * * Copyright : Angus Johnson 2010-2023 *
* Purpose : Path Offset (Inflate/Shrink) * * Purpose : Path Offset (Inflate/Shrink) *
@ -302,13 +302,7 @@ void ClipperOffset::OffsetPoint(Group& group, Path64& path, size_t j, size_t& k)
if (sin_a > 1.0) sin_a = 1.0; if (sin_a > 1.0) sin_a = 1.0;
else if (sin_a < -1.0) sin_a = -1.0; else if (sin_a < -1.0) sin_a = -1.0;
if (cos_a > 0.99) // almost straight - less than 8 degrees if (cos_a > -0.99 && (sin_a * group_delta_ < 0))
{
group.path.push_back(GetPerpendic(path[j], norms[k], group_delta_));
if (cos_a < 0.9998) // greater than 1 degree (#424)
group.path.push_back(GetPerpendic(path[j], norms[j], group_delta_)); // (#418)
}
else if (cos_a > -0.99 && (sin_a * group_delta_ < 0))
{ {
// is concave // is concave
group.path.push_back(GetPerpendic(path[j], norms[k], group_delta_)); group.path.push_back(GetPerpendic(path[j], norms[k], group_delta_));
@ -317,20 +311,20 @@ void ClipperOffset::OffsetPoint(Group& group, Path64& path, size_t j, size_t& k)
group.path.push_back(path[j]); // (#405) group.path.push_back(path[j]); // (#405)
group.path.push_back(GetPerpendic(path[j], norms[j], group_delta_)); group.path.push_back(GetPerpendic(path[j], norms[j], group_delta_));
} }
else if (join_type_ == JoinType::Round)
DoRound(group, path, j, k, std::atan2(sin_a, cos_a));
else if (join_type_ == JoinType::Miter) else if (join_type_ == JoinType::Miter)
{ {
// miter unless the angle is so acute the miter would exceeds ML // miter unless the angle is so acute the miter would exceeds ML
if (cos_a > temp_lim_ - 1) DoMiter(group, path, j, k, cos_a); if (cos_a > temp_lim_ - 1) DoMiter(group, path, j, k, cos_a);
else DoSquare(group, path, j, k); else DoSquare(group, path, j, k);
} }
// don't bother squaring angles that deviate < ~20 degrees because else if (cos_a > 0.9998)
// squaring will be indistinguishable from mitering and just be a lot slower // almost straight - less than 1 degree (#424)
else if (cos_a > 0.9)
DoMiter(group, path, j, k, cos_a); DoMiter(group, path, j, k, cos_a);
else else if (cos_a > 0.99 || join_type_ == JoinType::Square)
//angle less than 8 degrees or squared joins
DoSquare(group, path, j, k); DoSquare(group, path, j, k);
else
DoRound(group, path, j, k, std::atan2(sin_a, cos_a));
k = j; k = j;
} }
@ -481,10 +475,11 @@ void ClipperOffset::DoGroupOffset(Group& group)
bool is_joined = bool is_joined =
(end_type_ == EndType::Polygon) || (end_type_ == EndType::Polygon) ||
(end_type_ == EndType::Joined); (end_type_ == EndType::Joined);
Paths64::const_iterator path_iter; Paths64::iterator path_iter;
for(path_iter = group.paths_in.cbegin(); path_iter != group.paths_in.cend(); ++path_iter) for(path_iter = group.paths_in.begin(); path_iter != group.paths_in.end(); ++path_iter)
{ {
Path64 path = StripDuplicates(*path_iter, is_joined); auto path = *path_iter;
StripDuplicates(path, is_joined);
Path64::size_type cnt = path.size(); Path64::size_type cnt = path.size();
if (cnt == 0 || ((cnt < 3) && group.end_type == EndType::Polygon)) if (cnt == 0 || ((cnt < 3) && group.end_type == EndType::Polygon))
continue; continue;

View File

@ -1,6 +1,6 @@
/******************************************************************************* /*******************************************************************************
* Author : Angus Johnson * * Author : Angus Johnson *
* Date : 14 February 2023 * * Date : 22 April 2023 *
* Website : http://www.angusj.com * * Website : http://www.angusj.com *
* Copyright : Angus Johnson 2010-2023 * * Copyright : Angus Johnson 2010-2023 *
* Purpose : FAST rectangular clipping * * Purpose : FAST rectangular clipping *
@ -619,7 +619,7 @@ namespace Clipper2Lib {
p1 = cw[i]; p1 = cw[i];
if (!p1 || p1->next == p1->prev) if (!p1 || p1->next == p1->prev)
{ {
cw[i++]->edge = nullptr; cw[i++] = nullptr;
j = 0; j = 0;
continue; continue;
} }
@ -819,7 +819,7 @@ namespace Clipper2Lib {
Paths64 result; Paths64 result;
if (rect_.IsEmpty()) return result; if (rect_.IsEmpty()) return result;
for (const auto& path : paths) for (const Path64& path : paths)
{ {
if (path.size() < 3) continue; if (path.size() < 3) continue;
path_bounds_ = GetBounds(path); path_bounds_ = GetBounds(path);
@ -867,9 +867,7 @@ namespace Clipper2Lib {
for (const auto& path : paths) for (const auto& path : paths)
{ {
if (path.size() < 2) continue;
Rect64 pathrec = GetBounds(path); Rect64 pathrec = GetBounds(path);
if (!rect_.Intersects(pathrec)) continue; if (!rect_.Intersects(pathrec)) continue;
ExecuteInternal(path); ExecuteInternal(path);