2011-09-23 20:00:30 +00:00
|
|
|
/**
|
|
|
|
* @file zones_convert_to_polygons_aux_functions.cpp
|
|
|
|
*/
|
|
|
|
|
2012-06-08 09:56:42 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2013-05-01 19:01:14 +00:00
|
|
|
* Copyright (C) 2013 Jean-Pierre Charras, jean-pierre.charras@ujf-grenoble.fr
|
|
|
|
* Copyright (C) 1992-2013 KiCad Developers, see AUTHORS.txt for contributors.
|
2012-06-08 09:56:42 +00:00
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <fctsys.h>
|
|
|
|
#include <polygons_defs.h>
|
|
|
|
#include <PolyLine.h>
|
|
|
|
#include <wxPcbStruct.h>
|
|
|
|
#include <trigo.h>
|
2011-09-23 20:00:30 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <class_board.h>
|
|
|
|
#include <class_module.h>
|
|
|
|
#include <class_zone.h>
|
2011-09-23 20:00:30 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <pcbnew.h>
|
|
|
|
#include <zones.h>
|
2011-09-23 20:00:30 +00:00
|
|
|
|
2013-05-01 19:01:14 +00:00
|
|
|
/* Function TransformOutlinesShapeWithClearanceToPolygon
|
|
|
|
* Convert the zone filled areas polygons to polygons
|
|
|
|
* inflated (optional) by max( aClearanceValue, the zone clearance)
|
|
|
|
* and copy them in aCornerBuffer
|
|
|
|
* param aClearanceValue = the clearance around polygons
|
|
|
|
* param aAddClearance = true to add a clearance area to the polygon
|
|
|
|
* false to create the outline polygon.
|
|
|
|
*/
|
|
|
|
void ZONE_CONTAINER::TransformOutlinesShapeWithClearanceToPolygon(
|
2013-05-03 17:51:10 +00:00
|
|
|
CPOLYGONS_LIST& aCornerBuffer,
|
2013-05-01 19:01:14 +00:00
|
|
|
int aClearanceValue, bool aAddClearance )
|
|
|
|
{
|
|
|
|
// Creates the zone outlines polygon (with linked holes if any)
|
2013-05-03 17:51:10 +00:00
|
|
|
CPOLYGONS_LIST zoneOutines;
|
2013-05-01 19:01:14 +00:00
|
|
|
BuildFilledSolidAreasPolygons( NULL, &zoneOutines );
|
|
|
|
|
|
|
|
// add clearance to outline
|
|
|
|
int clearance = 0;
|
|
|
|
if( aAddClearance )
|
|
|
|
{
|
|
|
|
clearance = GetClearance();
|
|
|
|
if( aClearanceValue > clearance )
|
|
|
|
clearance = aClearanceValue;
|
|
|
|
}
|
|
|
|
// Calculate the polygon with clearance
|
|
|
|
// holes are linked to the main outline, so only one polygon should be created.
|
|
|
|
KI_POLYGON_SET polyset_zone_solid_areas;
|
|
|
|
std::vector<KI_POLY_POINT> cornerslist;
|
|
|
|
unsigned ic = 0;
|
2013-05-09 19:08:12 +00:00
|
|
|
unsigned corners_count = zoneOutines.GetCornersCount();
|
2013-05-01 19:01:14 +00:00
|
|
|
while( ic < corners_count )
|
|
|
|
{
|
|
|
|
cornerslist.clear();
|
|
|
|
KI_POLYGON poly;
|
|
|
|
{
|
|
|
|
for( ; ic < corners_count; ic++ )
|
|
|
|
{
|
|
|
|
CPolyPt* corner = &zoneOutines[ic];
|
|
|
|
cornerslist.push_back( KI_POLY_POINT( corner->x, corner->y ) );
|
|
|
|
if( corner->end_contour )
|
|
|
|
{
|
|
|
|
ic++;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bpl::set_points( poly, cornerslist.begin(), cornerslist.end() );
|
|
|
|
polyset_zone_solid_areas.push_back( poly );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
polyset_zone_solid_areas += clearance;
|
|
|
|
|
|
|
|
// Put the resulting polygon in aCornerBuffer corners list
|
|
|
|
for( unsigned ii = 0; ii < polyset_zone_solid_areas.size(); ii++ )
|
|
|
|
{
|
|
|
|
KI_POLYGON& poly = polyset_zone_solid_areas[ii];
|
|
|
|
CPolyPt corner( 0, 0, false );
|
|
|
|
|
|
|
|
for( unsigned jj = 0; jj < poly.size(); jj++ )
|
|
|
|
{
|
|
|
|
KI_POLY_POINT point = *(poly.begin() + jj);
|
|
|
|
corner.x = point.x();
|
|
|
|
corner.y = point.y();
|
|
|
|
corner.end_contour = false;
|
2013-05-09 19:08:12 +00:00
|
|
|
aCornerBuffer.Append( corner );
|
2013-05-01 19:01:14 +00:00
|
|
|
}
|
|
|
|
|
2013-05-09 19:08:12 +00:00
|
|
|
aCornerBuffer.CloseLastContour();
|
2013-05-01 19:01:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-09-23 20:00:30 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Function BuildUnconnectedThermalStubsPolygonList
|
|
|
|
* Creates a set of polygons corresponding to stubs created by thermal shapes on pads
|
|
|
|
* which are not connected to a zone (dangling bridges)
|
2013-05-03 17:51:10 +00:00
|
|
|
* @param aCornerBuffer = a CPOLYGONS_LIST where to store polygons
|
2011-09-23 20:00:30 +00:00
|
|
|
* @param aPcb = the board.
|
|
|
|
* @param aZone = a pointer to the ZONE_CONTAINER to examine.
|
|
|
|
* @param aArcCorrection = a pointer to the ZONE_CONTAINER to examine.
|
|
|
|
* @param aRoundPadThermalRotation = the rotation in 1.0 degree for thermal stubs in round pads
|
|
|
|
*/
|
|
|
|
|
2013-05-03 17:51:10 +00:00
|
|
|
void BuildUnconnectedThermalStubsPolygonList( CPOLYGONS_LIST& aCornerBuffer,
|
2011-09-23 20:00:30 +00:00
|
|
|
BOARD* aPcb,
|
|
|
|
ZONE_CONTAINER* aZone,
|
|
|
|
double aArcCorrection,
|
2013-05-05 07:17:48 +00:00
|
|
|
double aRoundPadThermalRotation )
|
2011-09-23 20:00:30 +00:00
|
|
|
{
|
|
|
|
std::vector<wxPoint> corners_buffer; // a local polygon buffer to store one stub
|
|
|
|
corners_buffer.reserve( 4 );
|
|
|
|
wxPoint ptTest[4];
|
|
|
|
|
2013-03-18 19:36:07 +00:00
|
|
|
int zone_clearance = aZone->GetZoneClearance();
|
2011-09-23 20:00:30 +00:00
|
|
|
|
|
|
|
EDA_RECT item_boundingbox;
|
|
|
|
EDA_RECT zone_boundingbox = aZone->GetBoundingBox();
|
|
|
|
int biggest_clearance = aPcb->GetBiggestClearanceValue();
|
2012-08-03 15:43:15 +00:00
|
|
|
biggest_clearance = std::max( biggest_clearance, zone_clearance );
|
2011-09-23 20:00:30 +00:00
|
|
|
zone_boundingbox.Inflate( biggest_clearance );
|
|
|
|
|
|
|
|
// half size of the pen used to draw/plot zones outlines
|
2013-03-18 19:36:07 +00:00
|
|
|
int pen_radius = aZone->GetMinThickness() / 2;
|
2011-09-23 20:00:30 +00:00
|
|
|
|
|
|
|
for( MODULE* module = aPcb->m_Modules; module; module = module->Next() )
|
|
|
|
{
|
2013-03-18 19:36:07 +00:00
|
|
|
for( D_PAD* pad = module->Pads(); pad != NULL; pad = pad->Next() )
|
2011-09-23 20:00:30 +00:00
|
|
|
{
|
2012-07-09 07:10:07 +00:00
|
|
|
// Rejects non-standard pads with tht-only thermal reliefs
|
|
|
|
if( aZone->GetPadConnection( pad ) == THT_THERMAL
|
|
|
|
&& pad->GetAttribute() != PAD_STANDARD )
|
|
|
|
continue;
|
|
|
|
|
2012-08-03 15:43:15 +00:00
|
|
|
if( aZone->GetPadConnection( pad ) != THERMAL_PAD
|
2012-07-09 07:10:07 +00:00
|
|
|
&& aZone->GetPadConnection( pad ) != THT_THERMAL )
|
2012-02-24 23:23:46 +00:00
|
|
|
continue;
|
|
|
|
|
2011-09-23 20:00:30 +00:00
|
|
|
// check
|
|
|
|
if( !pad->IsOnLayer( aZone->GetLayer() ) )
|
|
|
|
continue;
|
2012-02-24 23:23:46 +00:00
|
|
|
|
2014-02-25 10:47:27 +00:00
|
|
|
if( pad->GetNetCode() != aZone->GetNetCode() )
|
2011-09-23 20:00:30 +00:00
|
|
|
continue;
|
|
|
|
|
2013-02-09 20:03:20 +00:00
|
|
|
// Calculate thermal bridge half width
|
|
|
|
int thermalBridgeWidth = aZone->GetThermalReliefCopperBridge( pad )
|
2013-03-18 19:36:07 +00:00
|
|
|
- aZone->GetMinThickness();
|
2013-02-09 20:03:20 +00:00
|
|
|
if( thermalBridgeWidth <= 0 )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// we need the thermal bridge half width
|
|
|
|
// with a small extra size to be sure we create a stub
|
|
|
|
// slightly larger than the actual stub
|
|
|
|
thermalBridgeWidth = ( thermalBridgeWidth + 4 ) / 2;
|
|
|
|
|
|
|
|
int thermalReliefGap = aZone->GetThermalReliefGap( pad );
|
|
|
|
|
2011-09-23 20:00:30 +00:00
|
|
|
item_boundingbox = pad->GetBoundingBox();
|
2012-08-22 20:16:21 +00:00
|
|
|
item_boundingbox.Inflate( thermalReliefGap );
|
2011-09-23 20:00:30 +00:00
|
|
|
if( !( item_boundingbox.Intersects( zone_boundingbox ) ) )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// Thermal bridges are like a segment from a starting point inside the pad
|
|
|
|
// to an ending point outside the pad
|
2013-02-09 20:03:20 +00:00
|
|
|
|
|
|
|
// calculate the ending point of the thermal pad, outside the pad
|
|
|
|
wxPoint endpoint;
|
2012-08-22 20:16:21 +00:00
|
|
|
endpoint.x = ( pad->GetSize().x / 2 ) + thermalReliefGap;
|
|
|
|
endpoint.y = ( pad->GetSize().y / 2 ) + thermalReliefGap;
|
2011-09-23 20:00:30 +00:00
|
|
|
|
2013-02-09 20:03:20 +00:00
|
|
|
// Calculate the starting point of the thermal stub
|
|
|
|
// inside the pad
|
|
|
|
wxPoint startpoint;
|
|
|
|
int copperThickness = aZone->GetThermalReliefCopperBridge( pad )
|
2013-03-18 19:36:07 +00:00
|
|
|
- aZone->GetMinThickness();
|
|
|
|
|
2011-09-23 20:00:30 +00:00
|
|
|
if( copperThickness < 0 )
|
|
|
|
copperThickness = 0;
|
|
|
|
|
2013-02-09 20:03:20 +00:00
|
|
|
// Leave a small extra size to the copper area inside to pad
|
2013-05-04 11:57:09 +00:00
|
|
|
copperThickness += KiROUND( IU_PER_MM * 0.04 );
|
2013-02-09 20:03:20 +00:00
|
|
|
|
2012-08-03 15:43:15 +00:00
|
|
|
startpoint.x = std::min( pad->GetSize().x, copperThickness );
|
|
|
|
startpoint.y = std::min( pad->GetSize().y, copperThickness );
|
2013-02-09 20:03:20 +00:00
|
|
|
|
2011-09-23 20:00:30 +00:00
|
|
|
startpoint.x /= 2;
|
|
|
|
startpoint.y /= 2;
|
|
|
|
|
2013-02-02 17:39:59 +00:00
|
|
|
// This is a CIRCLE pad tweak
|
|
|
|
// for circle pads, the thermal stubs orientation is 45 deg
|
2013-05-05 07:17:48 +00:00
|
|
|
double fAngle = pad->GetOrientation();
|
2012-02-19 04:02:19 +00:00
|
|
|
if( pad->GetShape() == PAD_CIRCLE )
|
2011-09-23 20:00:30 +00:00
|
|
|
{
|
2013-05-04 11:57:09 +00:00
|
|
|
endpoint.x = KiROUND( endpoint.x * aArcCorrection );
|
2011-09-23 20:00:30 +00:00
|
|
|
endpoint.y = endpoint.x;
|
|
|
|
fAngle = aRoundPadThermalRotation;
|
|
|
|
}
|
|
|
|
|
|
|
|
// contour line width has to be taken into calculation to avoid "thermal stub bleed"
|
|
|
|
endpoint.x += pen_radius;
|
|
|
|
endpoint.y += pen_radius;
|
|
|
|
// compute north, south, west and east points for zone connection.
|
|
|
|
ptTest[0] = wxPoint( 0, endpoint.y ); // lower point
|
|
|
|
ptTest[1] = wxPoint( 0, -endpoint.y ); // upper point
|
|
|
|
ptTest[2] = wxPoint( endpoint.x, 0 ); // right point
|
|
|
|
ptTest[3] = wxPoint( -endpoint.x, 0 ); // left point
|
|
|
|
|
|
|
|
// Test all sides
|
|
|
|
for( int i = 0; i < 4; i++ )
|
|
|
|
{
|
|
|
|
// rotate point
|
|
|
|
RotatePoint( &ptTest[i], fAngle );
|
|
|
|
|
|
|
|
// translate point
|
|
|
|
ptTest[i] += pad->ReturnShapePos();
|
2013-02-09 20:03:20 +00:00
|
|
|
|
2011-09-23 20:00:30 +00:00
|
|
|
if( aZone->HitTestFilledArea( ptTest[i] ) )
|
|
|
|
continue;
|
|
|
|
|
|
|
|
corners_buffer.clear();
|
|
|
|
|
|
|
|
// polygons are rectangles with width of copper bridge value
|
|
|
|
switch( i )
|
|
|
|
{
|
|
|
|
case 0: // lower stub
|
2012-08-22 20:16:21 +00:00
|
|
|
corners_buffer.push_back( wxPoint( -thermalBridgeWidth, endpoint.y ) );
|
|
|
|
corners_buffer.push_back( wxPoint( +thermalBridgeWidth, endpoint.y ) );
|
|
|
|
corners_buffer.push_back( wxPoint( +thermalBridgeWidth, startpoint.y ) );
|
|
|
|
corners_buffer.push_back( wxPoint( -thermalBridgeWidth, startpoint.y ) );
|
2011-09-23 20:00:30 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 1: // upper stub
|
2012-08-22 20:16:21 +00:00
|
|
|
corners_buffer.push_back( wxPoint( -thermalBridgeWidth, -endpoint.y ) );
|
|
|
|
corners_buffer.push_back( wxPoint( +thermalBridgeWidth, -endpoint.y ) );
|
|
|
|
corners_buffer.push_back( wxPoint( +thermalBridgeWidth, -startpoint.y ) );
|
|
|
|
corners_buffer.push_back( wxPoint( -thermalBridgeWidth, -startpoint.y ) );
|
2011-09-23 20:00:30 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 2: // right stub
|
2012-08-22 20:16:21 +00:00
|
|
|
corners_buffer.push_back( wxPoint( endpoint.x, -thermalBridgeWidth ) );
|
|
|
|
corners_buffer.push_back( wxPoint( endpoint.x, thermalBridgeWidth ) );
|
|
|
|
corners_buffer.push_back( wxPoint( +startpoint.x, thermalBridgeWidth ) );
|
|
|
|
corners_buffer.push_back( wxPoint( +startpoint.x, -thermalBridgeWidth ) );
|
2011-09-23 20:00:30 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case 3: // left stub
|
2012-08-22 20:16:21 +00:00
|
|
|
corners_buffer.push_back( wxPoint( -endpoint.x, -thermalBridgeWidth ) );
|
|
|
|
corners_buffer.push_back( wxPoint( -endpoint.x, thermalBridgeWidth ) );
|
|
|
|
corners_buffer.push_back( wxPoint( -startpoint.x, thermalBridgeWidth ) );
|
|
|
|
corners_buffer.push_back( wxPoint( -startpoint.x, -thermalBridgeWidth ) );
|
2011-09-23 20:00:30 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// add computed polygon to list
|
|
|
|
for( unsigned ic = 0; ic < corners_buffer.size(); ic++ )
|
|
|
|
{
|
|
|
|
wxPoint cpos = corners_buffer[ic];
|
|
|
|
RotatePoint( &cpos, fAngle ); // Rotate according to module orientation
|
|
|
|
cpos += pad->ReturnShapePos(); // Shift origin to position
|
|
|
|
CPolyPt corner;
|
|
|
|
corner.x = cpos.x;
|
|
|
|
corner.y = cpos.y;
|
2013-05-09 19:08:12 +00:00
|
|
|
corner.end_contour = ( ic < (corners_buffer.size() - 1) ) ? false : true;
|
|
|
|
aCornerBuffer.Append( corner );
|
2011-09-23 20:00:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|