Zone fill fixes for divots and hatching + more debug layers.
Fixes https://gitlab.com/kicad/code/kicad/issues/6203
This commit is contained in:
parent
0ef480dc84
commit
fc6fc88219
|
@ -1162,7 +1162,8 @@ void ZONE_CONTAINER::GetInteractingZones( PCB_LAYER_ID aLayer,
|
||||||
|
|
||||||
|
|
||||||
bool ZONE_CONTAINER::BuildSmoothedPoly( SHAPE_POLY_SET& aSmoothedPoly, PCB_LAYER_ID aLayer,
|
bool ZONE_CONTAINER::BuildSmoothedPoly( SHAPE_POLY_SET& aSmoothedPoly, PCB_LAYER_ID aLayer,
|
||||||
SHAPE_POLY_SET* aBoardOutline ) const
|
SHAPE_POLY_SET* aBoardOutline,
|
||||||
|
SHAPE_POLY_SET* aSmoothedPolyWithApron ) const
|
||||||
{
|
{
|
||||||
if( GetNumCorners() <= 2 ) // malformed zone. polygon calculations will not like it ...
|
if( GetNumCorners() <= 2 ) // malformed zone. polygon calculations will not like it ...
|
||||||
return false;
|
return false;
|
||||||
|
@ -1230,6 +1231,14 @@ bool ZONE_CONTAINER::BuildSmoothedPoly( SHAPE_POLY_SET& aSmoothedPoly, PCB_LAYER
|
||||||
|
|
||||||
smooth( aSmoothedPoly );
|
smooth( aSmoothedPoly );
|
||||||
|
|
||||||
|
if( aSmoothedPolyWithApron )
|
||||||
|
{
|
||||||
|
SHAPE_POLY_SET bufferedExtents = *maxExtents;
|
||||||
|
bufferedExtents.Inflate( m_ZoneMinThickness, 8 );
|
||||||
|
*aSmoothedPolyWithApron = aSmoothedPoly;
|
||||||
|
aSmoothedPolyWithApron->BooleanIntersection( bufferedExtents, SHAPE_POLY_SET::PM_FAST );
|
||||||
|
}
|
||||||
|
|
||||||
aSmoothedPoly.BooleanIntersection( *maxExtents, SHAPE_POLY_SET::PM_FAST );
|
aSmoothedPoly.BooleanIntersection( *maxExtents, SHAPE_POLY_SET::PM_FAST );
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
|
|
|
@ -672,7 +672,8 @@ public:
|
||||||
* Function GetSmoothedPoly
|
* Function GetSmoothedPoly
|
||||||
*/
|
*/
|
||||||
bool BuildSmoothedPoly( SHAPE_POLY_SET& aSmoothedPoly, PCB_LAYER_ID aLayer,
|
bool BuildSmoothedPoly( SHAPE_POLY_SET& aSmoothedPoly, PCB_LAYER_ID aLayer,
|
||||||
SHAPE_POLY_SET* aBoardOutline ) const;
|
SHAPE_POLY_SET* aBoardOutline,
|
||||||
|
SHAPE_POLY_SET* aSmoothedPolyWithApron = nullptr ) const;
|
||||||
|
|
||||||
void SetCornerSmoothingType( int aType ) { m_cornerSmoothingType = aType; };
|
void SetCornerSmoothingType( int aType ) { m_cornerSmoothingType = aType; };
|
||||||
|
|
||||||
|
|
|
@ -138,6 +138,9 @@ void ZONE_FILLER_TOOL::FillAllZones( wxWindow* aCaller, PROGRESS_REPORTER* aRepo
|
||||||
commit.Revert();
|
commit.Revert();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if( filler.IsDebug() )
|
||||||
|
getEditFrame<PCB_EDIT_FRAME>()->UpdateUserInterface();
|
||||||
|
|
||||||
canvas()->Refresh();
|
canvas()->Refresh();
|
||||||
|
|
||||||
// wxWidgets has keyboard focus issues after the progress reporter. Re-setting the focus
|
// wxWidgets has keyboard focus issues after the progress reporter. Re-setting the focus
|
||||||
|
|
|
@ -1005,18 +1005,18 @@ void ZONE_FILLER::subtractHigherPriorityZones( const ZONE_CONTAINER* aZone, PCB_
|
||||||
|
|
||||||
|
|
||||||
#define DUMP_POLYS_TO_COPPER_LAYER( a, b, c ) \
|
#define DUMP_POLYS_TO_COPPER_LAYER( a, b, c ) \
|
||||||
{ if( m_debugZoneFiller && dumpLayer == b ) \
|
{ if( m_debugZoneFiller && aDebugLayer == b ) \
|
||||||
{ \
|
{ \
|
||||||
m_board->SetLayerName( b, c ); \
|
m_board->SetLayerName( b, c ); \
|
||||||
SHAPE_POLY_SET d = a; \
|
SHAPE_POLY_SET d = a; \
|
||||||
d.Simplify( SHAPE_POLY_SET::PM_STRICTLY_SIMPLE ); \
|
d.Simplify( SHAPE_POLY_SET::PM_STRICTLY_SIMPLE ); \
|
||||||
d.Fracture( SHAPE_POLY_SET::PM_STRICTLY_SIMPLE ); \
|
d.Fracture( SHAPE_POLY_SET::PM_STRICTLY_SIMPLE ); \
|
||||||
aRawPolys = d; \
|
aRawPolys = d; \
|
||||||
aFinalPolys = d; \
|
return false; \
|
||||||
return; \
|
|
||||||
} \
|
} \
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* 1 - Creates the main zone outline using a correction to shrink the resulting area by
|
* 1 - Creates the main zone outline using a correction to shrink the resulting area by
|
||||||
* m_ZoneMinThickness / 2. The result is areas with a margin of m_ZoneMinThickness / 2
|
* m_ZoneMinThickness / 2. The result is areas with a margin of m_ZoneMinThickness / 2
|
||||||
|
@ -1028,16 +1028,12 @@ void ZONE_FILLER::subtractHigherPriorityZones( const ZONE_CONTAINER* aZone, PCB_
|
||||||
* 5 - Removes unconnected copper islands, deleting any affected spokes
|
* 5 - Removes unconnected copper islands, deleting any affected spokes
|
||||||
* 6 - Adds in the remaining spokes
|
* 6 - Adds in the remaining spokes
|
||||||
*/
|
*/
|
||||||
void ZONE_FILLER::computeRawFilledArea( const ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
bool ZONE_FILLER::computeRawFilledArea( const ZONE_CONTAINER* aZone,
|
||||||
|
PCB_LAYER_ID aLayer, PCB_LAYER_ID aDebugLayer,
|
||||||
const SHAPE_POLY_SET& aSmoothedOutline,
|
const SHAPE_POLY_SET& aSmoothedOutline,
|
||||||
SHAPE_POLY_SET& aRawPolys,
|
const SHAPE_POLY_SET& aMaxExtents,
|
||||||
SHAPE_POLY_SET& aFinalPolys )
|
SHAPE_POLY_SET& aRawPolys )
|
||||||
{
|
{
|
||||||
PCB_LAYER_ID dumpLayer = aLayer;
|
|
||||||
|
|
||||||
if( m_debugZoneFiller && LSET::InternalCuMask().Contains( aLayer ) )
|
|
||||||
aLayer = F_Cu;
|
|
||||||
|
|
||||||
m_maxError = m_board->GetDesignSettings().m_MaxError;
|
m_maxError = m_board->GetDesignSettings().m_MaxError;
|
||||||
|
|
||||||
// Features which are min_width should survive pruning; features that are *less* than
|
// Features which are min_width should survive pruning; features that are *less* than
|
||||||
|
@ -1067,24 +1063,24 @@ void ZONE_FILLER::computeRawFilledArea( const ZONE_CONTAINER* aZone, PCB_LAYER_I
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In1_Cu, "smoothed-outline" );
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In1_Cu, "smoothed-outline" );
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
knockoutThermalReliefs( aZone, aLayer, aRawPolys );
|
knockoutThermalReliefs( aZone, aLayer, aRawPolys );
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In2_Cu, "minus-thermal-reliefs" );
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In2_Cu, "minus-thermal-reliefs" );
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
buildCopperItemClearances( aZone, aLayer, clearanceHoles );
|
buildCopperItemClearances( aZone, aLayer, clearanceHoles );
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( clearanceHoles, In3_Cu, "clearance-holes" );
|
DUMP_POLYS_TO_COPPER_LAYER( clearanceHoles, In3_Cu, "clearance-holes" );
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
buildThermalSpokes( aZone, aLayer, thermalSpokes );
|
buildThermalSpokes( aZone, aLayer, thermalSpokes );
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
// Create a temporary zone that we can hit-test spoke-ends against. It's only temporary
|
// Create a temporary zone that we can hit-test spoke-ends against. It's only temporary
|
||||||
// because the "real" subtract-clearance-holes has to be done after the spokes are added.
|
// because the "real" subtract-clearance-holes has to be done after the spokes are added.
|
||||||
|
@ -1104,7 +1100,7 @@ void ZONE_FILLER::computeRawFilledArea( const ZONE_CONTAINER* aZone, PCB_LAYER_I
|
||||||
}
|
}
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
// Spoke-end-testing is hugely expensive so we generate cached bounding-boxes to speed
|
// Spoke-end-testing is hugely expensive so we generate cached bounding-boxes to speed
|
||||||
// things up a bit.
|
// things up a bit.
|
||||||
|
@ -1130,7 +1126,7 @@ void ZONE_FILLER::computeRawFilledArea( const ZONE_CONTAINER* aZone, PCB_LAYER_I
|
||||||
if( interval++ > 400 )
|
if( interval++ > 400 )
|
||||||
{
|
{
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
interval = 0;
|
interval = 0;
|
||||||
}
|
}
|
||||||
|
@ -1152,7 +1148,7 @@ void ZONE_FILLER::computeRawFilledArea( const ZONE_CONTAINER* aZone, PCB_LAYER_I
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( debugSpokes, In7_Cu, "spokes" );
|
DUMP_POLYS_TO_COPPER_LAYER( debugSpokes, In7_Cu, "spokes" );
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
aRawPolys.BooleanSubtract( clearanceHoles, SHAPE_POLY_SET::PM_FAST );
|
aRawPolys.BooleanSubtract( clearanceHoles, SHAPE_POLY_SET::PM_FAST );
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In8_Cu, "after-spoke-trimming" );
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In8_Cu, "after-spoke-trimming" );
|
||||||
|
@ -1164,16 +1160,17 @@ void ZONE_FILLER::computeRawFilledArea( const ZONE_CONTAINER* aZone, PCB_LAYER_I
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In9_Cu, "deflated" );
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In9_Cu, "deflated" );
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
// Now remove the non filled areas due to the hatch pattern
|
// Now remove the non filled areas due to the hatch pattern
|
||||||
if( aZone->GetFillMode() == ZONE_FILL_MODE::HATCH_PATTERN )
|
if( aZone->GetFillMode() == ZONE_FILL_MODE::HATCH_PATTERN )
|
||||||
addHatchFillTypeOnZone( aZone, aLayer, aRawPolys );
|
{
|
||||||
|
if( !addHatchFillTypeOnZone( aZone, aLayer, aDebugLayer, aRawPolys ) )
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In10_Cu, "after-hatching" );
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
// Re-inflate after pruning of areas that don't meet minimum-width criteria
|
// Re-inflate after pruning of areas that don't meet minimum-width criteria
|
||||||
if( aZone->GetFilledPolysUseThickness() )
|
if( aZone->GetFilledPolysUseThickness() )
|
||||||
|
@ -1186,21 +1183,21 @@ void ZONE_FILLER::computeRawFilledArea( const ZONE_CONTAINER* aZone, PCB_LAYER_I
|
||||||
aRawPolys.Inflate( half_min_width - epsilon, numSegs, cornerStrategy );
|
aRawPolys.Inflate( half_min_width - epsilon, numSegs, cornerStrategy );
|
||||||
}
|
}
|
||||||
|
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In11_Cu, "after-reinflating" );
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In15_Cu, "after-reinflating" );
|
||||||
|
|
||||||
// Ensure additive changes (thermal stubs and particularly inflating acute corners) do not
|
// Ensure additive changes (thermal stubs and particularly inflating acute corners) do not
|
||||||
// add copper outside the zone boundary or inside the clearance holes
|
// add copper outside the zone boundary or inside the clearance holes
|
||||||
aRawPolys.BooleanIntersection( aSmoothedOutline, SHAPE_POLY_SET::PM_FAST );
|
aRawPolys.BooleanIntersection( aMaxExtents, SHAPE_POLY_SET::PM_FAST );
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In12_Cu, "after-trim-to-outline" );
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In16_Cu, "after-trim-to-outline" );
|
||||||
aRawPolys.BooleanSubtract( clearanceHoles, SHAPE_POLY_SET::PM_FAST );
|
aRawPolys.BooleanSubtract( clearanceHoles, SHAPE_POLY_SET::PM_FAST );
|
||||||
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In13_Cu, "after-trim-to-clearance-holes" );
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In17_Cu, "after-trim-to-clearance-holes" );
|
||||||
|
|
||||||
// Lastly give any same-net but higher-priority zones control over their own area.
|
// Lastly give any same-net but higher-priority zones control over their own area.
|
||||||
subtractHigherPriorityZones( aZone, aLayer, aRawPolys );
|
subtractHigherPriorityZones( aZone, aLayer, aRawPolys );
|
||||||
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In18_Cu, "minus-higher-priority-zones" );
|
||||||
|
|
||||||
aRawPolys.Fracture( SHAPE_POLY_SET::PM_FAST );
|
aRawPolys.Fracture( SHAPE_POLY_SET::PM_FAST );
|
||||||
|
return true;
|
||||||
aFinalPolys = aRawPolys;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1213,14 +1210,22 @@ bool ZONE_FILLER::fillSingleZone( ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
||||||
SHAPE_POLY_SET& aRawPolys, SHAPE_POLY_SET& aFinalPolys )
|
SHAPE_POLY_SET& aRawPolys, SHAPE_POLY_SET& aFinalPolys )
|
||||||
{
|
{
|
||||||
SHAPE_POLY_SET* boardOutline = m_brdOutlinesValid ? &m_boardOutline : nullptr;
|
SHAPE_POLY_SET* boardOutline = m_brdOutlinesValid ? &m_boardOutline : nullptr;
|
||||||
|
SHAPE_POLY_SET maxExtents;
|
||||||
SHAPE_POLY_SET smoothedPoly;
|
SHAPE_POLY_SET smoothedPoly;
|
||||||
|
PCB_LAYER_ID debugLayer = UNDEFINED_LAYER;
|
||||||
|
|
||||||
|
if( m_debugZoneFiller && LSET::InternalCuMask().Contains( aLayer ) )
|
||||||
|
{
|
||||||
|
debugLayer = aLayer;
|
||||||
|
aLayer = F_Cu;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* convert outlines + holes to outlines without holes (adding extra segments if necessary)
|
* convert outlines + holes to outlines without holes (adding extra segments if necessary)
|
||||||
* m_Poly data is expected normalized, i.e. NormalizeAreaOutlines was used after building
|
* m_Poly data is expected normalized, i.e. NormalizeAreaOutlines was used after building
|
||||||
* this zone
|
* this zone
|
||||||
*/
|
*/
|
||||||
if ( !aZone->BuildSmoothedPoly( smoothedPoly, aLayer, boardOutline ) )
|
if ( !aZone->BuildSmoothedPoly( maxExtents, aLayer, boardOutline, &smoothedPoly ) )
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
if( m_progressReporter && m_progressReporter->IsCancelled() )
|
||||||
|
@ -1228,7 +1233,10 @@ bool ZONE_FILLER::fillSingleZone( ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
||||||
|
|
||||||
if( aZone->IsOnCopperLayer() )
|
if( aZone->IsOnCopperLayer() )
|
||||||
{
|
{
|
||||||
computeRawFilledArea( aZone, aLayer, smoothedPoly, aRawPolys, aFinalPolys );
|
if( computeRawFilledArea( aZone, aLayer, debugLayer, smoothedPoly, maxExtents, aRawPolys ) )
|
||||||
|
aZone->SetNeedRefill( false );
|
||||||
|
|
||||||
|
aFinalPolys = aRawPolys;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
|
@ -1243,7 +1251,7 @@ bool ZONE_FILLER::fillSingleZone( ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
||||||
|
|
||||||
// Remove the non filled areas due to the hatch pattern
|
// Remove the non filled areas due to the hatch pattern
|
||||||
if( aZone->GetFillMode() == ZONE_FILL_MODE::HATCH_PATTERN )
|
if( aZone->GetFillMode() == ZONE_FILL_MODE::HATCH_PATTERN )
|
||||||
addHatchFillTypeOnZone( aZone, aLayer, smoothedPoly );
|
addHatchFillTypeOnZone( aZone, aLayer, debugLayer, smoothedPoly );
|
||||||
|
|
||||||
// Re-inflate after pruning of areas that don't meet minimum-width criteria
|
// Re-inflate after pruning of areas that don't meet minimum-width criteria
|
||||||
if( aZone->GetFilledPolysUseThickness() )
|
if( aZone->GetFilledPolysUseThickness() )
|
||||||
|
@ -1252,15 +1260,17 @@ bool ZONE_FILLER::fillSingleZone( ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
||||||
// inflate the zone by half_min_width
|
// inflate the zone by half_min_width
|
||||||
}
|
}
|
||||||
else if( half_min_width - epsilon > epsilon )
|
else if( half_min_width - epsilon > epsilon )
|
||||||
|
{
|
||||||
smoothedPoly.Deflate( -( half_min_width - epsilon ), numSegs );
|
smoothedPoly.Deflate( -( half_min_width - epsilon ), numSegs );
|
||||||
|
}
|
||||||
|
|
||||||
aRawPolys = smoothedPoly;
|
aRawPolys = smoothedPoly;
|
||||||
aFinalPolys = smoothedPoly;
|
aFinalPolys = smoothedPoly;
|
||||||
|
|
||||||
aFinalPolys.Fracture( SHAPE_POLY_SET::PM_STRICTLY_SIMPLE );
|
aFinalPolys.Fracture( SHAPE_POLY_SET::PM_STRICTLY_SIMPLE );
|
||||||
|
aZone->SetNeedRefill( false );
|
||||||
}
|
}
|
||||||
|
|
||||||
aZone->SetNeedRefill( false );
|
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1384,8 +1394,8 @@ void ZONE_FILLER::buildThermalSpokes( const ZONE_CONTAINER* aZone, PCB_LAYER_ID
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void ZONE_FILLER::addHatchFillTypeOnZone( const ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
bool ZONE_FILLER::addHatchFillTypeOnZone( const ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
||||||
SHAPE_POLY_SET& aRawPolys )
|
PCB_LAYER_ID aDebugLayer, SHAPE_POLY_SET& aRawPolys )
|
||||||
{
|
{
|
||||||
// Build grid:
|
// Build grid:
|
||||||
|
|
||||||
|
@ -1511,22 +1521,29 @@ void ZONE_FILLER::addHatchFillTypeOnZone( const ZONE_CONTAINER* aZone, PCB_LAYER
|
||||||
|
|
||||||
holes.Move( bbox.GetPosition() );
|
holes.Move( bbox.GetPosition() );
|
||||||
|
|
||||||
// We must buffer holes by at least aZone->GetMinThickness() to guarantee that thermal
|
|
||||||
// reliefs can be built (and to give the zone a solid outline). However, it looks more
|
|
||||||
// visually consistent if the buffer width is the same as the hatch width.
|
|
||||||
int outline_margin = KiROUND( aZone->GetMinThickness() * 1.1 );
|
|
||||||
|
|
||||||
if( aZone->GetHatchBorderAlgorithm() )
|
|
||||||
outline_margin = std::max( outline_margin, aZone->GetHatchThickness() );
|
|
||||||
|
|
||||||
if( outline_margin > linethickness / 2 )
|
|
||||||
filledPolys.Deflate( outline_margin - linethickness / 2, 16 );
|
|
||||||
|
|
||||||
holes.BooleanIntersection( filledPolys, SHAPE_POLY_SET::PM_FAST );
|
|
||||||
|
|
||||||
if( orientation != 0.0 )
|
if( orientation != 0.0 )
|
||||||
holes.Rotate( -M_PI/180.0 * orientation, VECTOR2I( 0,0 ) );
|
holes.Rotate( -M_PI/180.0 * orientation, VECTOR2I( 0,0 ) );
|
||||||
|
|
||||||
|
DUMP_POLYS_TO_COPPER_LAYER( holes, In10_Cu, "hatch-holes" );
|
||||||
|
|
||||||
|
int outline_margin = aZone->GetMinThickness() * 1.1;
|
||||||
|
|
||||||
|
// Using GetHatchThickness() can look more consistent than GetMinThickness().
|
||||||
|
if( aZone->GetHatchBorderAlgorithm() && aZone->GetHatchThickness() > outline_margin )
|
||||||
|
outline_margin = aZone->GetHatchThickness();
|
||||||
|
|
||||||
|
// The fill has already been deflated to ensure GetMinThickness() so we just have to
|
||||||
|
// account for anything beyond that.
|
||||||
|
SHAPE_POLY_SET deflatedFilledPolys = aRawPolys;
|
||||||
|
deflatedFilledPolys.Deflate( outline_margin - aZone->GetMinThickness(), 16 );
|
||||||
|
holes.BooleanIntersection( deflatedFilledPolys, SHAPE_POLY_SET::PM_FAST );
|
||||||
|
DUMP_POLYS_TO_COPPER_LAYER( holes, In11_Cu, "fill-clipped-hatch-holes" );
|
||||||
|
|
||||||
|
SHAPE_POLY_SET deflatedOutline = *aZone->Outline();
|
||||||
|
deflatedOutline.Deflate( outline_margin, 16 );
|
||||||
|
holes.BooleanIntersection( deflatedOutline, SHAPE_POLY_SET::PM_FAST );
|
||||||
|
DUMP_POLYS_TO_COPPER_LAYER( holes, In12_Cu, "outline-clipped-hatch-holes" );
|
||||||
|
|
||||||
if( aZone->GetNetCode() != 0 )
|
if( aZone->GetNetCode() != 0 )
|
||||||
{
|
{
|
||||||
// Vias and pads connected to the zone must not be allowed to become isolated inside
|
// Vias and pads connected to the zone must not be allowed to become isolated inside
|
||||||
|
@ -1584,6 +1601,7 @@ void ZONE_FILLER::addHatchFillTypeOnZone( const ZONE_CONTAINER* aZone, PCB_LAYER
|
||||||
|
|
||||||
holes.BooleanSubtract( aprons, SHAPE_POLY_SET::PM_FAST );
|
holes.BooleanSubtract( aprons, SHAPE_POLY_SET::PM_FAST );
|
||||||
}
|
}
|
||||||
|
DUMP_POLYS_TO_COPPER_LAYER( holes, In13_Cu, "pad-via-clipped-hatch-holes" );
|
||||||
|
|
||||||
// Now filter truncated holes to avoid small holes in pattern
|
// Now filter truncated holes to avoid small holes in pattern
|
||||||
// It happens for holes near the zone outline
|
// It happens for holes near the zone outline
|
||||||
|
@ -1600,4 +1618,7 @@ void ZONE_FILLER::addHatchFillTypeOnZone( const ZONE_CONTAINER* aZone, PCB_LAYER
|
||||||
// create grid. Use SHAPE_POLY_SET::PM_STRICTLY_SIMPLE to
|
// create grid. Use SHAPE_POLY_SET::PM_STRICTLY_SIMPLE to
|
||||||
// generate strictly simple polygons needed by Gerber files and Fracture()
|
// generate strictly simple polygons needed by Gerber files and Fracture()
|
||||||
aRawPolys.BooleanSubtract( aRawPolys, holes, SHAPE_POLY_SET::PM_STRICTLY_SIMPLE );
|
aRawPolys.BooleanSubtract( aRawPolys, holes, SHAPE_POLY_SET::PM_STRICTLY_SIMPLE );
|
||||||
|
DUMP_POLYS_TO_COPPER_LAYER( aRawPolys, In14_Cu, "after-hatching" );
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
|
@ -47,6 +47,8 @@ public:
|
||||||
bool Fill( std::vector<ZONE_CONTAINER*>& aZones, bool aCheck = false,
|
bool Fill( std::vector<ZONE_CONTAINER*>& aZones, bool aCheck = false,
|
||||||
wxWindow* aParent = nullptr );
|
wxWindow* aParent = nullptr );
|
||||||
|
|
||||||
|
bool IsDebug() const { return m_debugZoneFiller; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
void addKnockout( D_PAD* aPad, PCB_LAYER_ID aLayer, int aGap, SHAPE_POLY_SET& aHoles );
|
void addKnockout( D_PAD* aPad, PCB_LAYER_ID aLayer, int aGap, SHAPE_POLY_SET& aHoles );
|
||||||
|
@ -74,9 +76,11 @@ private:
|
||||||
* filled copper area polygon (without clearance areas
|
* filled copper area polygon (without clearance areas
|
||||||
* @param aPcb: the current board
|
* @param aPcb: the current board
|
||||||
*/
|
*/
|
||||||
void computeRawFilledArea( const ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
bool computeRawFilledArea( const ZONE_CONTAINER* aZone,
|
||||||
|
PCB_LAYER_ID aLayer, PCB_LAYER_ID aDebugLayer,
|
||||||
const SHAPE_POLY_SET& aSmoothedOutline,
|
const SHAPE_POLY_SET& aSmoothedOutline,
|
||||||
SHAPE_POLY_SET& aRawPolys, SHAPE_POLY_SET& aFinalPolys );
|
const SHAPE_POLY_SET& aMaxExtents,
|
||||||
|
SHAPE_POLY_SET& aRawPolys );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Function buildThermalSpokes
|
* Function buildThermalSpokes
|
||||||
|
@ -109,8 +113,8 @@ private:
|
||||||
* @param aRawPolys: A reference to a SHAPE_POLY_SET buffer containing the initial
|
* @param aRawPolys: A reference to a SHAPE_POLY_SET buffer containing the initial
|
||||||
* filled areas, and after adding the grid pattern, the modified filled areas with holes
|
* filled areas, and after adding the grid pattern, the modified filled areas with holes
|
||||||
*/
|
*/
|
||||||
void addHatchFillTypeOnZone( const ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
bool addHatchFillTypeOnZone( const ZONE_CONTAINER* aZone, PCB_LAYER_ID aLayer,
|
||||||
SHAPE_POLY_SET& aRawPolys );
|
PCB_LAYER_ID aDebugLayer, SHAPE_POLY_SET& aRawPolys );
|
||||||
|
|
||||||
BOARD* m_board;
|
BOARD* m_board;
|
||||||
SHAPE_POLY_SET m_boardOutline; // the board outlines, if exists
|
SHAPE_POLY_SET m_boardOutline; // the board outlines, if exists
|
||||||
|
|
Loading…
Reference in New Issue