2012-01-14 19:50:32 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2012-06-08 09:56:42 +00:00
|
|
|
* Copyright (C) 2012 Jean-Pierre Charras, jean-pierre.charras@ujf-grenoble.fr
|
|
|
|
* Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
|
|
|
|
* Copyright (C) 1992-2012 KiCad Developers, see AUTHORS.txt for contributors.
|
2012-01-14 19:50:32 +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
|
|
|
|
*/
|
|
|
|
|
2011-09-23 13:57:12 +00:00
|
|
|
/**
|
|
|
|
* @file class_pad.cpp
|
|
|
|
* D_PAD class implementation.
|
|
|
|
*/
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <fctsys.h>
|
|
|
|
#include <PolyLine.h>
|
|
|
|
#include <common.h>
|
|
|
|
#include <confirm.h>
|
|
|
|
#include <kicad_string.h>
|
|
|
|
#include <trigo.h>
|
2012-02-19 04:02:19 +00:00
|
|
|
#include <protos.h>
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <richio.h>
|
|
|
|
#include <wxstruct.h>
|
|
|
|
#include <macros.h>
|
2013-01-12 17:32:24 +00:00
|
|
|
#include <msgpanel.h>
|
|
|
|
#include <base_units.h>
|
2009-02-04 15:25:03 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <pcbnew.h>
|
|
|
|
#include <pcbnew_id.h> // ID_TRACK_BUTT
|
2011-09-23 13:57:12 +00:00
|
|
|
|
2012-01-23 04:33:36 +00:00
|
|
|
#include <class_board.h>
|
|
|
|
#include <class_module.h>
|
2012-07-31 17:51:58 +00:00
|
|
|
#include <polygon_test_point_inside.h>
|
2012-10-17 10:57:21 +00:00
|
|
|
#include <convert_from_iu.h>
|
2011-09-23 13:57:12 +00:00
|
|
|
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2011-07-07 10:22:46 +00:00
|
|
|
int D_PAD::m_PadSketchModePenSize = 0; // Pen size used to draw pads in sketch mode
|
2010-02-22 19:56:32 +00:00
|
|
|
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
D_PAD::D_PAD( MODULE* parent ) :
|
|
|
|
BOARD_CONNECTED_ITEM( parent, PCB_PAD_T )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2009-11-04 19:08:08 +00:00
|
|
|
m_NumPadName = 0;
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2011-11-24 17:32:51 +00:00
|
|
|
m_Size.x = m_Size.y = 500; // give it a reasonable size
|
2009-11-04 19:08:08 +00:00
|
|
|
m_Orient = 0; // Pad rotation in 1/10 degrees
|
2012-12-12 11:57:17 +00:00
|
|
|
m_LengthPadToDie = 0;
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
if( m_Parent && m_Parent->Type() == PCB_MODULE_T )
|
2007-08-07 06:21:19 +00:00
|
|
|
{
|
2012-02-19 04:02:19 +00:00
|
|
|
m_Pos = GetParent()->GetPosition();
|
2007-08-07 06:21:19 +00:00
|
|
|
}
|
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
m_PadShape = PAD_CIRCLE; // Shape: PAD_CIRCLE, PAD_RECT PAD_OVAL
|
|
|
|
// PAD_TRAPEZOID
|
|
|
|
m_Attribute = PAD_STANDARD; // Type: NORMAL, PAD_SMD, PAD_CONN
|
|
|
|
m_DrillShape = PAD_CIRCLE; // Drill shape = circle
|
2011-11-24 17:32:51 +00:00
|
|
|
m_LocalClearance = 0;
|
|
|
|
m_LocalSolderMaskMargin = 0;
|
|
|
|
m_LocalSolderPasteMargin = 0;
|
2009-11-04 19:08:08 +00:00
|
|
|
m_LocalSolderPasteMarginRatio = 0.0;
|
2012-02-24 23:23:46 +00:00
|
|
|
m_ZoneConnection = UNDEFINED_CONNECTION; // Use parent setting by default
|
2012-03-08 20:44:03 +00:00
|
|
|
m_ThermalWidth = 0; // Use parent setting by default
|
|
|
|
m_ThermalGap = 0; // Use parent setting by default
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
// set layers mask to default for a standard pad
|
|
|
|
m_layerMask = PAD_STANDARD_DEFAULT_LAYERS;
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
SetSubRatsnest( 0 ); // used in ratsnest calculations
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
m_boundingRadius = -1;
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
int D_PAD::boundingRadius() const
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2010-12-12 02:29:33 +00:00
|
|
|
int x, y;
|
|
|
|
int radius;
|
2010-12-10 19:47:44 +00:00
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
switch( GetShape() )
|
2007-08-07 06:21:19 +00:00
|
|
|
{
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_CIRCLE:
|
2011-11-24 17:32:51 +00:00
|
|
|
radius = m_Size.x / 2;
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_OVAL:
|
2012-09-22 11:19:37 +00:00
|
|
|
radius = std::max( m_Size.x, m_Size.y ) / 2;
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_RECT:
|
2011-11-24 17:32:51 +00:00
|
|
|
radius = 1 + (int) ( sqrt( (double) m_Size.y * m_Size.y
|
2012-03-15 14:31:16 +00:00
|
|
|
+ (double) m_Size.x * m_Size.x ) / 2 );
|
2010-09-18 17:55:08 +00:00
|
|
|
break;
|
|
|
|
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_TRAPEZOID:
|
2012-09-22 11:19:37 +00:00
|
|
|
x = m_Size.x + std::abs( m_DeltaSize.y ); // Remember: m_DeltaSize.y is the m_Size.x change
|
|
|
|
y = m_Size.y + std::abs( m_DeltaSize.x ); // Remember: m_DeltaSize.x is the m_Size.y change
|
2010-12-10 19:47:44 +00:00
|
|
|
radius = 1 + (int) ( sqrt( (double) y * y + (double) x * x ) / 2 );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
2010-12-12 02:29:33 +00:00
|
|
|
|
|
|
|
default:
|
2012-02-19 04:02:19 +00:00
|
|
|
radius = 0;
|
2007-08-07 06:21:19 +00:00
|
|
|
}
|
2010-12-10 19:47:44 +00:00
|
|
|
|
|
|
|
return radius;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-03-29 19:33:07 +00:00
|
|
|
EDA_RECT D_PAD::GetBoundingBox() const
|
2008-04-18 13:28:56 +00:00
|
|
|
{
|
2011-03-29 19:33:07 +00:00
|
|
|
EDA_RECT area;
|
2012-02-19 04:02:19 +00:00
|
|
|
|
|
|
|
// radius of pad area, enclosed in minimum sized circle
|
|
|
|
int radius = boundingRadius();
|
2010-12-10 19:47:44 +00:00
|
|
|
|
2011-11-24 17:32:51 +00:00
|
|
|
area.SetOrigin( m_Pos );
|
2010-12-10 19:47:44 +00:00
|
|
|
area.Inflate( radius );
|
2008-07-31 15:30:57 +00:00
|
|
|
|
2008-11-24 06:53:43 +00:00
|
|
|
return area;
|
2008-04-18 13:28:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
void D_PAD::SetOrientation( double aAngle )
|
|
|
|
{
|
|
|
|
NORMALIZE_ANGLE_POS( aAngle );
|
|
|
|
m_Orient = aAngle;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D_PAD::Flip( int aTranslationY )
|
|
|
|
{
|
|
|
|
int y = GetPosition().y - aTranslationY;
|
|
|
|
|
|
|
|
y = -y; // invert about x axis.
|
|
|
|
|
|
|
|
y += aTranslationY;
|
|
|
|
|
|
|
|
SetY( y );
|
|
|
|
|
|
|
|
NEGATE( m_Pos0.y );
|
|
|
|
NEGATE( m_Offset.y );
|
|
|
|
NEGATE( m_DeltaSize.y );
|
|
|
|
|
|
|
|
SetOrientation( -GetOrientation() );
|
|
|
|
|
|
|
|
// flip pads layers
|
|
|
|
SetLayerMask( ChangeSideMaskLayer( m_layerMask ) );
|
|
|
|
|
|
|
|
// m_boundingRadius = -1; the shape has not been changed
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void D_PAD::AppendConfigs( PARAM_CFG_ARRAY* aResult )
|
|
|
|
{
|
2012-10-17 10:57:21 +00:00
|
|
|
aResult->push_back( new PARAM_CFG_INT_WITH_SCALE( wxT( "PadDrill" ),
|
|
|
|
&m_Drill.x,
|
|
|
|
Millimeter2iu( 0.6 ),
|
|
|
|
Millimeter2iu( 0.1 ), Millimeter2iu( 10.0 ),
|
|
|
|
NULL, MM_PER_IU ) );
|
|
|
|
|
|
|
|
aResult->push_back( new PARAM_CFG_INT_WITH_SCALE( wxT( "PadSizeH" ),
|
|
|
|
&m_Size.x,
|
|
|
|
Millimeter2iu( 1.4 ),
|
|
|
|
Millimeter2iu( 0.1 ), Millimeter2iu( 20.0 ),
|
|
|
|
NULL, MM_PER_IU ) );
|
|
|
|
|
|
|
|
aResult->push_back( new PARAM_CFG_INT_WITH_SCALE( wxT( "PadSizeV" ),
|
|
|
|
&m_Size.y,
|
|
|
|
Millimeter2iu( 1.4 ),
|
|
|
|
Millimeter2iu( 0.1 ), Millimeter2iu( 20.0 ),
|
|
|
|
NULL, MM_PER_IU ) );
|
2012-02-19 04:02:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
// Returns the position of the pad.
|
2012-03-15 14:31:16 +00:00
|
|
|
const wxPoint D_PAD::ReturnShapePos() const
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2011-11-24 17:32:51 +00:00
|
|
|
if( m_Offset.x == 0 && m_Offset.y == 0 )
|
|
|
|
return m_Pos;
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
wxPoint shape_pos;
|
|
|
|
int dX, dY;
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2011-11-24 17:32:51 +00:00
|
|
|
dX = m_Offset.x;
|
|
|
|
dY = m_Offset.y;
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
RotatePoint( &dX, &dY, m_Orient );
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2011-11-24 17:32:51 +00:00
|
|
|
shape_pos.x = m_Pos.x + dX;
|
|
|
|
shape_pos.y = m_Pos.y + dY;
|
2007-06-05 12:10:51 +00:00
|
|
|
|
|
|
|
return shape_pos;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-12 08:37:05 +00:00
|
|
|
const wxString D_PAD::GetPadName() const
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2011-12-16 17:03:25 +00:00
|
|
|
#if 0 // m_Padname is not ASCII and not UTF8, it is LATIN1 basically, whatever
|
|
|
|
// 8 bit font is supported in KiCad plotting and drawing.
|
|
|
|
|
2011-12-12 08:37:05 +00:00
|
|
|
// Return pad name as wxString, assume it starts as a non-terminated
|
|
|
|
// utf8 character sequence
|
|
|
|
|
|
|
|
char temp[sizeof(m_Padname)+1]; // a place to terminate with '\0'
|
|
|
|
|
|
|
|
strncpy( temp, m_Padname, sizeof(m_Padname) );
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2011-12-12 08:37:05 +00:00
|
|
|
temp[sizeof(m_Padname)] = 0;
|
|
|
|
|
|
|
|
return FROM_UTF8( temp );
|
2011-12-16 17:03:25 +00:00
|
|
|
#else
|
|
|
|
|
|
|
|
wxString name;
|
|
|
|
|
|
|
|
ReturnStringPadName( name );
|
|
|
|
return name;
|
|
|
|
#endif
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2011-07-14 15:42:44 +00:00
|
|
|
void D_PAD::ReturnStringPadName( wxString& text ) const
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2011-12-16 17:03:25 +00:00
|
|
|
#if 0 // m_Padname is not ASCII and not UTF8, it is LATIN1 basically, whatever
|
|
|
|
// 8 bit font is supported in KiCad plotting and drawing.
|
|
|
|
|
2011-12-12 08:37:05 +00:00
|
|
|
// Return pad name as wxString, assume it starts as a non-terminated
|
|
|
|
// utf8 character sequence
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2011-12-12 08:37:05 +00:00
|
|
|
char temp[sizeof(m_Padname)+1]; // a place to terminate with '\0'
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-12 08:37:05 +00:00
|
|
|
strncpy( temp, m_Padname, sizeof(m_Padname) );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-12 08:37:05 +00:00
|
|
|
temp[sizeof(m_Padname)] = 0;
|
|
|
|
|
|
|
|
text = FROM_UTF8( temp );
|
2011-12-16 17:03:25 +00:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
text.Empty();
|
|
|
|
|
2011-12-22 21:57:50 +00:00
|
|
|
for( int ii = 0; ii < PADNAMEZ && m_Padname[ii]; ii++ )
|
2011-12-16 17:03:25 +00:00
|
|
|
{
|
2011-12-22 21:57:50 +00:00
|
|
|
// m_Padname is 8 bit KiCad font junk, do not sign extend
|
2011-12-16 17:03:25 +00:00
|
|
|
text.Append( (unsigned char) m_Padname[ii] );
|
|
|
|
}
|
|
|
|
#endif
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Change pad name
|
2009-11-12 15:43:38 +00:00
|
|
|
void D_PAD::SetPadName( const wxString& name )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2007-08-07 06:21:19 +00:00
|
|
|
int ii, len;
|
|
|
|
|
|
|
|
len = name.Length();
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-16 17:03:25 +00:00
|
|
|
if( len > PADNAMEZ )
|
|
|
|
len = PADNAMEZ;
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2011-12-16 17:03:25 +00:00
|
|
|
// m_Padname[] is not UTF8, it is an 8 bit character that matches the KiCad font,
|
|
|
|
// so only copy the lower 8 bits of each character.
|
2011-12-02 21:56:47 +00:00
|
|
|
|
2011-12-16 17:03:25 +00:00
|
|
|
for( ii = 0; ii < len; ii++ )
|
|
|
|
m_Padname[ii] = (char) name.GetChar( ii );
|
2011-12-02 21:56:47 +00:00
|
|
|
|
2011-12-16 17:03:25 +00:00
|
|
|
for( ii = len; ii < PADNAMEZ; ii++ )
|
|
|
|
m_Padname[ii] = '\0';
|
2011-12-02 21:56:47 +00:00
|
|
|
}
|
|
|
|
|
2009-11-04 19:08:08 +00:00
|
|
|
|
2008-12-14 19:45:05 +00:00
|
|
|
/**
|
|
|
|
* Function SetNetname
|
2010-12-29 17:47:32 +00:00
|
|
|
* @param aNetname: the new netname
|
2008-12-14 19:45:05 +00:00
|
|
|
*/
|
2009-11-12 15:43:38 +00:00
|
|
|
void D_PAD::SetNetname( const wxString& aNetname )
|
2008-12-14 19:45:05 +00:00
|
|
|
{
|
|
|
|
m_Netname = aNetname;
|
|
|
|
m_ShortNetname = m_Netname.AfterLast( '/' );
|
|
|
|
}
|
|
|
|
|
2007-06-05 12:10:51 +00:00
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
void D_PAD::Copy( D_PAD* source )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2007-08-07 06:21:19 +00:00
|
|
|
if( source == NULL )
|
|
|
|
return;
|
|
|
|
|
|
|
|
m_Pos = source->m_Pos;
|
2011-09-07 19:41:04 +00:00
|
|
|
m_layerMask = source->m_layerMask;
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2010-09-11 16:33:43 +00:00
|
|
|
m_NumPadName = source->m_NumPadName;
|
2009-11-12 15:43:38 +00:00
|
|
|
SetNet( source->GetNet() );
|
|
|
|
m_Drill = source->m_Drill;
|
2007-08-07 06:21:19 +00:00
|
|
|
m_DrillShape = source->m_DrillShape;
|
2009-11-12 15:43:38 +00:00
|
|
|
m_Offset = source->m_Offset;
|
|
|
|
m_Size = source->m_Size;
|
|
|
|
m_DeltaSize = source->m_DeltaSize;
|
|
|
|
m_Pos0 = source->m_Pos0;
|
2012-02-19 04:02:19 +00:00
|
|
|
m_boundingRadius = source->m_boundingRadius;
|
2009-11-12 15:43:38 +00:00
|
|
|
m_PadShape = source->m_PadShape;
|
2012-02-19 04:02:19 +00:00
|
|
|
m_Attribute = source->m_Attribute;
|
2009-11-12 15:43:38 +00:00
|
|
|
m_Orient = source->m_Orient;
|
2012-12-12 11:57:17 +00:00
|
|
|
m_LengthPadToDie = source->m_LengthPadToDie;
|
2009-11-12 15:43:38 +00:00
|
|
|
m_LocalClearance = source->m_LocalClearance;
|
2009-11-04 19:08:08 +00:00
|
|
|
m_LocalSolderMaskMargin = source->m_LocalSolderMaskMargin;
|
|
|
|
m_LocalSolderPasteMargin = source->m_LocalSolderPasteMargin;
|
|
|
|
m_LocalSolderPasteMarginRatio = source->m_LocalSolderPasteMarginRatio;
|
2012-02-24 23:23:46 +00:00
|
|
|
m_ZoneConnection = source->m_ZoneConnection;
|
2012-03-08 20:44:03 +00:00
|
|
|
m_ThermalWidth = source->m_ThermalWidth;
|
|
|
|
m_ThermalGap = source->m_ThermalGap;
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2008-11-18 18:13:55 +00:00
|
|
|
SetSubRatsnest( 0 );
|
|
|
|
SetSubNet( 0 );
|
2007-08-07 06:21:19 +00:00
|
|
|
m_Netname = source->m_Netname;
|
2008-12-14 19:45:05 +00:00
|
|
|
m_ShortNetname = source->m_ShortNetname;
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
|
2010-11-12 16:59:16 +00:00
|
|
|
/**
|
|
|
|
* Function GetClearance (virtual)
|
2010-04-08 11:33:43 +00:00
|
|
|
* returns the clearance in internal units. If \a aItem is not NULL then the
|
|
|
|
* returned clearance is the greater of this object's clearance and
|
2010-10-28 19:50:12 +00:00
|
|
|
* aItem's clearance. If \a aItem is NULL, then this object clearance is returned.
|
2009-11-05 20:59:42 +00:00
|
|
|
* @param aItem is another BOARD_CONNECTED_ITEM or NULL
|
2010-04-08 11:33:43 +00:00
|
|
|
* @return int - the clearance in internal units.
|
2009-11-12 15:43:38 +00:00
|
|
|
*/
|
2009-11-05 20:59:42 +00:00
|
|
|
int D_PAD::GetClearance( BOARD_CONNECTED_ITEM* aItem ) const
|
|
|
|
{
|
2010-04-08 11:33:43 +00:00
|
|
|
// A pad can have specific clearance parameters that
|
|
|
|
// overrides its NETCLASS clearance value
|
2011-11-24 17:32:51 +00:00
|
|
|
int clearance = m_LocalClearance;
|
2009-11-12 15:43:38 +00:00
|
|
|
|
|
|
|
if( clearance == 0 )
|
2012-04-17 01:35:43 +00:00
|
|
|
{
|
|
|
|
// If local clearance is 0, use the parent footprint clearance value
|
|
|
|
if( GetParent() && GetParent()->GetLocalClearance() )
|
|
|
|
clearance = GetParent()->GetLocalClearance();
|
2009-11-05 20:59:42 +00:00
|
|
|
}
|
|
|
|
|
2010-04-08 11:33:43 +00:00
|
|
|
if( clearance == 0 ) // If the parent footprint clearance value = 0, use NETCLASS value
|
2009-11-05 20:59:42 +00:00
|
|
|
return BOARD_CONNECTED_ITEM::GetClearance( aItem );
|
|
|
|
|
2010-04-08 11:33:43 +00:00
|
|
|
// We have a specific clearance.
|
|
|
|
// if aItem, return the biggest clearance
|
2009-11-05 20:59:42 +00:00
|
|
|
if( aItem )
|
|
|
|
{
|
2010-04-08 11:33:43 +00:00
|
|
|
int hisClearance = aItem->GetClearance();
|
2012-08-03 15:43:15 +00:00
|
|
|
return std::max( hisClearance, clearance );
|
2009-11-05 20:59:42 +00:00
|
|
|
}
|
|
|
|
|
2010-04-08 11:33:43 +00:00
|
|
|
// Return the specific clearance.
|
2009-11-05 20:59:42 +00:00
|
|
|
return clearance;
|
|
|
|
}
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
|
2009-11-04 19:08:08 +00:00
|
|
|
// Mask margins handling:
|
|
|
|
|
2010-11-12 16:36:43 +00:00
|
|
|
/**
|
|
|
|
* Function GetSolderMaskMargin
|
2009-11-04 19:08:08 +00:00
|
|
|
* @return the margin for the solder mask layer
|
|
|
|
* usually > 0 (mask shape bigger than pad
|
|
|
|
* value is
|
|
|
|
* 1 - the local value
|
|
|
|
* 2 - if null, the parent footprint value
|
|
|
|
* 1 - if null, the global value
|
|
|
|
*/
|
|
|
|
int D_PAD::GetSolderMaskMargin()
|
|
|
|
{
|
2012-02-02 17:45:37 +00:00
|
|
|
int margin = m_LocalSolderMaskMargin;
|
2012-04-17 01:35:43 +00:00
|
|
|
MODULE* module = GetParent();
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-01-31 20:01:46 +00:00
|
|
|
if( module )
|
2009-11-04 19:08:08 +00:00
|
|
|
{
|
2010-01-31 20:01:46 +00:00
|
|
|
if( margin == 0 )
|
|
|
|
{
|
2012-04-17 01:35:43 +00:00
|
|
|
if( module->GetLocalSolderMaskMargin() )
|
|
|
|
margin = module->GetLocalSolderMaskMargin();
|
2010-01-31 20:01:46 +00:00
|
|
|
}
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-01-31 20:01:46 +00:00
|
|
|
if( margin == 0 )
|
|
|
|
{
|
2012-02-02 17:45:37 +00:00
|
|
|
BOARD* brd = GetBoard();
|
++PCBNew
* Removed Pcb_Frame argument from BOARD() constructor, since it precludes
having a BOARD being edited by more than one editor, it was a bad design.
And this meant removing m_PcbFrame from BOARD.
* removed BOARD::SetWindowFrame(), and BOARD::m_PcbFrame
* Removed the global BOARD_DESIGN_SETTINGS which was in class_board.cpp
* added BOARD_DESIGN_SETTINGS to the BOARD class, a full instance
* a couple dialogs now only change BOARD_DESIGN_SETTINGS when OK is pressed,
such as dialog_mask_clearance, dialog_drc, etc.
* Removed common/pcbcommon.cpp's int g_CurrentVersionPCB = 1 and replaced it
with build_version.h's #define BOARD_FILE_VERSION, although there may be a
better place for this constant.
* Made the public functions in PARAM_CFG_ARRAY be type const.
void SaveParam(..) const and void ReadParam(..) const
* PARAM_CFG_BASE now has virtual destructor since we have various way of
destroying the derived class and boost::ptr_vector must be told about this.
* Pass const PARAM_CFG_ARRAY& instead of PARAM_CFG_ARRAY so that we can use
an automatic PARAM_CFG_ARRAY which is on the stack.\
* PCB_EDIT_FRAME::GetProjectFileParameters() may no longer cache the array,
since it has to access the current BOARD and the BOARD can change.
Remember BOARD_DESIGN_SETTINGS are now in the BOARD.
* Made the m_BoundingBox member private, this was a brutally hard task,
and indicative of the lack of commitment to accessors and object oriented
design on the part of KiCad developers. We must do better.
Added BOARD::GetBoundingBox, SetBoundingBox(), ComputeBoundingBox().
* Added PCB_BASE_FRAME::GetBoardBoundingBox() which calls BOARD::ComputeBoundingBox()
2011-12-05 06:15:33 +00:00
|
|
|
margin = brd->GetDesignSettings().m_SolderMaskMargin;
|
2010-01-31 20:01:46 +00:00
|
|
|
}
|
2009-11-04 19:08:08 +00:00
|
|
|
}
|
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
// ensure mask have a size always >= 0
|
2009-11-04 19:08:08 +00:00
|
|
|
if( margin < 0 )
|
|
|
|
{
|
2012-09-22 11:19:37 +00:00
|
|
|
int minsize = -std::min( m_Size.x, m_Size.y ) / 2;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
if( margin < minsize )
|
2009-11-04 19:08:08 +00:00
|
|
|
minsize = minsize;
|
|
|
|
}
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2009-11-04 19:08:08 +00:00
|
|
|
return margin;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-11-12 16:36:43 +00:00
|
|
|
/**
|
|
|
|
* Function GetSolderPasteMargin
|
2009-11-04 19:08:08 +00:00
|
|
|
* @return the margin for the solder mask layer
|
|
|
|
* usually < 0 (mask shape smaller than pad
|
|
|
|
* value is
|
|
|
|
* 1 - the local value
|
|
|
|
* 2 - if null, the parent footprint value
|
|
|
|
* 1 - if null, the global value
|
|
|
|
*/
|
|
|
|
wxSize D_PAD::GetSolderPasteMargin()
|
|
|
|
{
|
2012-04-17 01:35:43 +00:00
|
|
|
int margin = m_LocalSolderPasteMargin;
|
|
|
|
double mratio = m_LocalSolderPasteMarginRatio;
|
|
|
|
MODULE* module = GetParent();
|
2009-11-12 15:43:38 +00:00
|
|
|
|
2010-01-31 20:01:46 +00:00
|
|
|
if( module )
|
|
|
|
{
|
2012-04-17 01:35:43 +00:00
|
|
|
if( margin == 0 )
|
|
|
|
margin = module->GetLocalSolderPasteMargin();
|
2009-11-04 19:08:08 +00:00
|
|
|
|
2010-01-31 20:01:46 +00:00
|
|
|
BOARD * brd = GetBoard();
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2012-04-17 01:35:43 +00:00
|
|
|
if( margin == 0 )
|
++PCBNew
* Removed Pcb_Frame argument from BOARD() constructor, since it precludes
having a BOARD being edited by more than one editor, it was a bad design.
And this meant removing m_PcbFrame from BOARD.
* removed BOARD::SetWindowFrame(), and BOARD::m_PcbFrame
* Removed the global BOARD_DESIGN_SETTINGS which was in class_board.cpp
* added BOARD_DESIGN_SETTINGS to the BOARD class, a full instance
* a couple dialogs now only change BOARD_DESIGN_SETTINGS when OK is pressed,
such as dialog_mask_clearance, dialog_drc, etc.
* Removed common/pcbcommon.cpp's int g_CurrentVersionPCB = 1 and replaced it
with build_version.h's #define BOARD_FILE_VERSION, although there may be a
better place for this constant.
* Made the public functions in PARAM_CFG_ARRAY be type const.
void SaveParam(..) const and void ReadParam(..) const
* PARAM_CFG_BASE now has virtual destructor since we have various way of
destroying the derived class and boost::ptr_vector must be told about this.
* Pass const PARAM_CFG_ARRAY& instead of PARAM_CFG_ARRAY so that we can use
an automatic PARAM_CFG_ARRAY which is on the stack.\
* PCB_EDIT_FRAME::GetProjectFileParameters() may no longer cache the array,
since it has to access the current BOARD and the BOARD can change.
Remember BOARD_DESIGN_SETTINGS are now in the BOARD.
* Made the m_BoundingBox member private, this was a brutally hard task,
and indicative of the lack of commitment to accessors and object oriented
design on the part of KiCad developers. We must do better.
Added BOARD::GetBoundingBox, SetBoundingBox(), ComputeBoundingBox().
* Added PCB_BASE_FRAME::GetBoardBoundingBox() which calls BOARD::ComputeBoundingBox()
2011-12-05 06:15:33 +00:00
|
|
|
margin = brd->GetDesignSettings().m_SolderPasteMargin;
|
2010-01-31 20:01:46 +00:00
|
|
|
|
|
|
|
if( mratio == 0.0 )
|
2012-04-17 01:35:43 +00:00
|
|
|
mratio = module->GetLocalSolderPasteMarginRatio();
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-01-31 20:01:46 +00:00
|
|
|
if( mratio == 0.0 )
|
|
|
|
{
|
2012-04-17 01:35:43 +00:00
|
|
|
mratio = brd->GetDesignSettings().m_SolderPasteMarginRatio;
|
2010-01-31 20:01:46 +00:00
|
|
|
}
|
|
|
|
}
|
2009-11-04 19:08:08 +00:00
|
|
|
|
|
|
|
wxSize pad_margin;
|
// Dick Hollenbeck's KiROUND R&D
// This provides better project control over rounding to int from double
// than wxRound() did. This scheme provides better logging in Debug builds
// and it provides for compile time calculation of constants.
#include <stdio.h>
#include <assert.h>
#include <limits.h>
//-----<KiROUND KIT>------------------------------------------------------------
/**
* KiROUND
* rounds a floating point number to an int using
* "round halfway cases away from zero".
* In Debug build an assert fires if will not fit into an int.
*/
#if defined( DEBUG )
// DEBUG: a macro to capture line and file, then calls this inline
static inline int KiRound( double v, int line, const char* filename )
{
v = v < 0 ? v - 0.5 : v + 0.5;
if( v > INT_MAX + 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' > 0 ' for int\n", __FUNCTION__, filename, line, v );
}
else if( v < INT_MIN - 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' < 0 ' for int\n", __FUNCTION__, filename, line, v );
}
return int( v );
}
#define KiROUND( v ) KiRound( v, __LINE__, __FILE__ )
#else
// RELEASE: a macro so compile can pre-compute constants.
#define KiROUND( v ) int( (v) < 0 ? (v) - 0.5 : (v) + 0.5 )
#endif
//-----</KiROUND KIT>-----------------------------------------------------------
// Only a macro is compile time calculated, an inline function causes a static constructor
// in a situation like this.
// Therefore the Release build is best done with a MACRO not an inline function.
int Computed = KiROUND( 14.3 * 8 );
int main( int argc, char** argv )
{
for( double d = double(INT_MAX)-1; d < double(INT_MAX)+8; d += 2.0 )
{
int i = KiROUND( d );
printf( "t: %d %.16g\n", i, d );
}
return 0;
}
2012-04-19 06:55:45 +00:00
|
|
|
pad_margin.x = margin + KiROUND( m_Size.x * mratio );
|
|
|
|
pad_margin.y = margin + KiROUND( m_Size.y * mratio );
|
2009-11-04 19:08:08 +00:00
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
// ensure mask have a size always >= 0
|
2011-11-24 17:32:51 +00:00
|
|
|
if( pad_margin.x < -m_Size.x / 2 )
|
|
|
|
pad_margin.x = -m_Size.x / 2;
|
2009-11-04 19:08:08 +00:00
|
|
|
|
2011-11-24 17:32:51 +00:00
|
|
|
if( pad_margin.y < -m_Size.y / 2 )
|
|
|
|
pad_margin.y = -m_Size.y / 2;
|
2009-11-04 19:08:08 +00:00
|
|
|
|
|
|
|
return pad_margin;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-02-24 23:23:46 +00:00
|
|
|
ZoneConnection D_PAD::GetZoneConnection() const
|
|
|
|
{
|
|
|
|
MODULE* module = (MODULE*) GetParent();
|
|
|
|
|
|
|
|
if( m_ZoneConnection == UNDEFINED_CONNECTION && module )
|
|
|
|
return module->GetZoneConnection();
|
|
|
|
else
|
|
|
|
return m_ZoneConnection;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-08 20:44:03 +00:00
|
|
|
int D_PAD::GetThermalWidth() const
|
|
|
|
{
|
|
|
|
MODULE* module = (MODULE*) GetParent();
|
|
|
|
|
|
|
|
if( m_ThermalWidth == 0 && module )
|
|
|
|
return module->GetThermalWidth();
|
|
|
|
else
|
|
|
|
return m_ThermalWidth;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int D_PAD::GetThermalGap() const
|
|
|
|
{
|
|
|
|
MODULE* module = (MODULE*) GetParent();
|
|
|
|
|
|
|
|
if( m_ThermalGap == 0 && module )
|
|
|
|
return module->GetThermalGap();
|
|
|
|
else
|
|
|
|
return m_ThermalGap;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
void D_PAD::GetMsgPanelInfo( std::vector< MSG_PANEL_ITEM>& aList )
|
2007-06-05 12:10:51 +00:00
|
|
|
{
|
2009-12-07 03:46:13 +00:00
|
|
|
MODULE* module;
|
|
|
|
wxString Line;
|
|
|
|
BOARD* board;
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2008-02-19 00:30:10 +00:00
|
|
|
module = (MODULE*) m_Parent;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2008-02-19 00:30:10 +00:00
|
|
|
if( module )
|
2007-08-07 06:21:19 +00:00
|
|
|
{
|
2008-02-19 00:30:10 +00:00
|
|
|
wxString msg = module->GetReference();
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "Module" ), msg, DARKCYAN ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
ReturnStringPadName( Line );
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "RefP" ), Line, BROWN ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
}
|
2009-12-07 03:46:13 +00:00
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "Net" ), m_Netname, DARKCYAN ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
/* For test and debug only: display m_physical_connexion and
|
|
|
|
* m_logical_connexion */
|
2008-11-18 18:13:55 +00:00
|
|
|
#if 1 // Used only to debug connectivity calculations
|
2011-12-07 15:49:32 +00:00
|
|
|
Line.Printf( wxT( "%d-%d-%d " ), GetSubRatsnest(), GetSubNet(), GetZoneSubNet() );
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( wxT( "L-P-Z" ), Line, DARKGREEN ) );
|
2007-06-05 12:10:51 +00:00
|
|
|
#endif
|
|
|
|
|
2009-12-07 03:46:13 +00:00
|
|
|
board = GetBoard();
|
2008-02-19 00:30:10 +00:00
|
|
|
|
2009-12-07 03:46:13 +00:00
|
|
|
wxString layerInfo;
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
if( (m_layerMask & ALL_CU_LAYERS) == 0 ) // pad is not on any copper layers
|
2007-08-07 06:21:19 +00:00
|
|
|
{
|
2011-09-07 19:41:04 +00:00
|
|
|
switch( m_layerMask & ~ALL_CU_LAYERS )
|
2007-08-07 06:21:19 +00:00
|
|
|
{
|
2009-12-21 18:51:37 +00:00
|
|
|
case ADHESIVE_LAYER_BACK:
|
2009-12-21 17:56:25 +00:00
|
|
|
layerInfo = board->GetLayerName( ADHESIVE_N_BACK );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2009-12-21 18:51:37 +00:00
|
|
|
case ADHESIVE_LAYER_FRONT:
|
2009-12-21 17:56:25 +00:00
|
|
|
layerInfo = board->GetLayerName( ADHESIVE_N_FRONT );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2009-12-21 18:51:37 +00:00
|
|
|
case SOLDERPASTE_LAYER_BACK:
|
2009-12-21 17:56:25 +00:00
|
|
|
layerInfo = board->GetLayerName( SOLDERPASTE_N_BACK );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2009-12-21 18:51:37 +00:00
|
|
|
case SOLDERPASTE_LAYER_FRONT:
|
2009-12-21 17:56:25 +00:00
|
|
|
layerInfo = board->GetLayerName( SOLDERPASTE_N_FRONT );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2009-12-21 18:51:37 +00:00
|
|
|
case SILKSCREEN_LAYER_BACK:
|
2009-12-21 17:56:25 +00:00
|
|
|
layerInfo = board->GetLayerName( SILKSCREEN_N_BACK );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2009-12-21 18:51:37 +00:00
|
|
|
case SILKSCREEN_LAYER_FRONT:
|
2009-12-21 17:56:25 +00:00
|
|
|
layerInfo = board->GetLayerName( SILKSCREEN_N_FRONT );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2009-12-21 18:51:37 +00:00
|
|
|
case SOLDERMASK_LAYER_BACK:
|
2009-12-21 17:56:25 +00:00
|
|
|
layerInfo = board->GetLayerName( SOLDERMASK_N_BACK );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
2009-12-21 18:51:37 +00:00
|
|
|
case SOLDERMASK_LAYER_FRONT:
|
2009-12-21 17:56:25 +00:00
|
|
|
layerInfo = board->GetLayerName( SOLDERMASK_N_FRONT );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DRAW_LAYER:
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo = board->GetLayerName( DRAW_N );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case COMMENT_LAYER:
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo = board->GetLayerName( COMMENT_N );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ECO1_LAYER:
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo = board->GetLayerName( ECO1_N );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ECO2_LAYER:
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo = board->GetLayerName( ECO2_N );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case EDGE_LAYER:
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo = board->GetLayerName( EDGE_N );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo = _( "Non-copper" );
|
2007-08-07 06:21:19 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2009-12-07 03:46:13 +00:00
|
|
|
else
|
|
|
|
{
|
2009-12-21 13:05:11 +00:00
|
|
|
#define INTERIOR_COPPER (ALL_CU_LAYERS & ~(LAYER_BACK | LAYER_FRONT))
|
2009-12-07 03:46:13 +00:00
|
|
|
|
|
|
|
static const wxChar* andInternal = _( " & int" );
|
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
if( (m_layerMask & (LAYER_BACK | LAYER_FRONT)) == LAYER_BACK )
|
2009-12-07 03:46:13 +00:00
|
|
|
{
|
|
|
|
layerInfo = board->GetLayerName( LAYER_N_BACK );
|
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
if( m_layerMask & INTERIOR_COPPER )
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo += andInternal;
|
|
|
|
}
|
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
else if( (m_layerMask & (LAYER_BACK | LAYER_FRONT)) == (LAYER_BACK | LAYER_FRONT) )
|
2009-12-07 03:46:13 +00:00
|
|
|
{
|
|
|
|
layerInfo = board->GetLayerName( LAYER_N_BACK ) + wxT(", ") +
|
|
|
|
board->GetLayerName( LAYER_N_FRONT );
|
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
if( m_layerMask & INTERIOR_COPPER )
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo += andInternal;
|
|
|
|
}
|
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
else if( (m_layerMask & (LAYER_BACK | LAYER_FRONT)) == LAYER_FRONT )
|
2009-12-07 03:46:13 +00:00
|
|
|
{
|
|
|
|
layerInfo = board->GetLayerName( LAYER_N_FRONT );
|
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
if( m_layerMask & INTERIOR_COPPER )
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo += andInternal;
|
|
|
|
}
|
2011-09-07 19:41:04 +00:00
|
|
|
else // necessarily true: if( m_layerMask & INTERIOR_COPPER )
|
|
|
|
{
|
2009-12-07 03:46:13 +00:00
|
|
|
layerInfo = _( "internal" );
|
2011-09-07 19:41:04 +00:00
|
|
|
}
|
2009-12-07 03:46:13 +00:00
|
|
|
}
|
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "Layer" ), layerInfo, DARKGREEN ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( ShowPadShape(), ShowPadAttr(), DARKGREEN ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
Line = ::CoordinateToString( m_Size.x );
|
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "H Size" ), Line, RED ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
Line = ::CoordinateToString( m_Size.y );
|
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "V Size" ), Line, RED ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
Line = ::CoordinateToString( (unsigned) m_Drill.x );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2008-01-05 17:30:56 +00:00
|
|
|
if( m_DrillShape == PAD_CIRCLE )
|
2007-08-07 06:21:19 +00:00
|
|
|
{
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "Drill" ), Line, RED ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2013-01-12 17:32:24 +00:00
|
|
|
Line = ::CoordinateToString( (unsigned) m_Drill.x );
|
2007-08-07 06:21:19 +00:00
|
|
|
wxString msg;
|
2013-01-12 17:32:24 +00:00
|
|
|
msg = ::CoordinateToString( (unsigned) m_Drill.y );
|
2012-11-06 17:58:53 +00:00
|
|
|
Line += wxT( "/" ) + msg;
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "Drill X / Y" ), Line, RED ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
}
|
|
|
|
|
2011-11-29 17:25:30 +00:00
|
|
|
int module_orient = module ? module->GetOrientation() : 0;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
if( module_orient )
|
|
|
|
Line.Printf( wxT( "%3.1f(+%3.1f)" ),
|
2012-11-06 17:58:53 +00:00
|
|
|
(double) ( m_Orient - module_orient ) / 10,
|
|
|
|
(double) module_orient / 10 );
|
2007-08-07 06:21:19 +00:00
|
|
|
else
|
2012-11-06 17:58:53 +00:00
|
|
|
Line.Printf( wxT( "%3.1f" ), (double) m_Orient / 10 );
|
2009-12-07 03:46:13 +00:00
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "Orient" ), Line, LIGHTBLUE ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
Line = ::CoordinateToString( m_Pos.x );
|
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "X Pos" ), Line, LIGHTBLUE ) );
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2013-01-12 17:32:24 +00:00
|
|
|
Line = ::CoordinateToString( m_Pos.y );
|
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "Y pos" ), Line, LIGHTBLUE ) );
|
2011-08-19 13:08:24 +00:00
|
|
|
|
2012-12-12 11:57:17 +00:00
|
|
|
if( GetPadToDieLength() )
|
2011-08-19 13:08:24 +00:00
|
|
|
{
|
2013-01-12 17:32:24 +00:00
|
|
|
Line = ::CoordinateToString( GetPadToDieLength() );
|
|
|
|
aList.push_back( MSG_PANEL_ITEM( _( "Length in package" ), Line, CYAN ) );
|
2011-08-19 13:08:24 +00:00
|
|
|
}
|
2007-06-05 12:10:51 +00:00
|
|
|
}
|
2007-08-07 06:21:19 +00:00
|
|
|
|
2007-08-08 03:50:44 +00:00
|
|
|
|
2007-08-30 22:20:52 +00:00
|
|
|
// see class_pad.h
|
|
|
|
bool D_PAD::IsOnLayer( int aLayer ) const
|
|
|
|
{
|
2011-09-07 19:41:04 +00:00
|
|
|
return (1 << aLayer) & m_layerMask;
|
2007-08-30 22:20:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-03-15 14:31:16 +00:00
|
|
|
bool D_PAD::HitTest( const wxPoint& aPosition )
|
2007-08-08 03:50:44 +00:00
|
|
|
{
|
|
|
|
int dx, dy;
|
|
|
|
double dist;
|
|
|
|
|
|
|
|
wxPoint shape_pos = ReturnShapePos();
|
|
|
|
|
2012-03-15 14:31:16 +00:00
|
|
|
wxPoint delta = aPosition - shape_pos;
|
2007-08-08 03:50:44 +00:00
|
|
|
|
2012-02-19 04:02:19 +00:00
|
|
|
// first test: a test point must be inside a minimum sized bounding circle.
|
|
|
|
int radius = GetBoundingRadius();
|
|
|
|
|
|
|
|
if( ( abs( delta.x ) > radius ) || ( abs( delta.y ) > radius ) )
|
2007-08-08 03:50:44 +00:00
|
|
|
return false;
|
|
|
|
|
2011-11-24 17:32:51 +00:00
|
|
|
dx = m_Size.x >> 1; // dx also is the radius for rounded pads
|
|
|
|
dy = m_Size.y >> 1;
|
2007-08-08 03:50:44 +00:00
|
|
|
|
|
|
|
switch( m_PadShape & 0x7F )
|
|
|
|
{
|
2008-01-05 17:30:56 +00:00
|
|
|
case PAD_CIRCLE:
|
2010-09-20 16:21:47 +00:00
|
|
|
dist = hypot( delta.x, delta.y );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
// Dick Hollenbeck's KiROUND R&D
// This provides better project control over rounding to int from double
// than wxRound() did. This scheme provides better logging in Debug builds
// and it provides for compile time calculation of constants.
#include <stdio.h>
#include <assert.h>
#include <limits.h>
//-----<KiROUND KIT>------------------------------------------------------------
/**
* KiROUND
* rounds a floating point number to an int using
* "round halfway cases away from zero".
* In Debug build an assert fires if will not fit into an int.
*/
#if defined( DEBUG )
// DEBUG: a macro to capture line and file, then calls this inline
static inline int KiRound( double v, int line, const char* filename )
{
v = v < 0 ? v - 0.5 : v + 0.5;
if( v > INT_MAX + 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' > 0 ' for int\n", __FUNCTION__, filename, line, v );
}
else if( v < INT_MIN - 0.5 )
{
printf( "%s: in file %s on line %d, val: %.16g too ' < 0 ' for int\n", __FUNCTION__, filename, line, v );
}
return int( v );
}
#define KiROUND( v ) KiRound( v, __LINE__, __FILE__ )
#else
// RELEASE: a macro so compile can pre-compute constants.
#define KiROUND( v ) int( (v) < 0 ? (v) - 0.5 : (v) + 0.5 )
#endif
//-----</KiROUND KIT>-----------------------------------------------------------
// Only a macro is compile time calculated, an inline function causes a static constructor
// in a situation like this.
// Therefore the Release build is best done with a MACRO not an inline function.
int Computed = KiROUND( 14.3 * 8 );
int main( int argc, char** argv )
{
for( double d = double(INT_MAX)-1; d < double(INT_MAX)+8; d += 2.0 )
{
int i = KiROUND( d );
printf( "t: %d %.16g\n", i, d );
}
return 0;
}
2012-04-19 06:55:45 +00:00
|
|
|
if( KiROUND( dist ) <= dx )
|
2007-08-08 03:50:44 +00:00
|
|
|
return true;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2007-08-08 03:50:44 +00:00
|
|
|
break;
|
|
|
|
|
2010-09-11 16:33:43 +00:00
|
|
|
case PAD_TRAPEZOID:
|
|
|
|
{
|
|
|
|
wxPoint poly[4];
|
|
|
|
BuildPadPolygon( poly, wxSize(0,0), 0 );
|
2010-09-20 16:21:47 +00:00
|
|
|
RotatePoint( &delta, -m_Orient );
|
|
|
|
return TestPointInsidePolygon( poly, 4, delta );
|
2010-09-11 16:33:43 +00:00
|
|
|
}
|
|
|
|
|
2007-08-08 03:50:44 +00:00
|
|
|
default:
|
2010-09-20 16:21:47 +00:00
|
|
|
RotatePoint( &delta, -m_Orient );
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2010-09-20 16:21:47 +00:00
|
|
|
if( (abs( delta.x ) <= dx ) && (abs( delta.y ) <= dy) )
|
2007-08-08 03:50:44 +00:00
|
|
|
return true;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2007-08-08 03:50:44 +00:00
|
|
|
break;
|
|
|
|
}
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2007-08-08 03:50:44 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-02-19 00:30:10 +00:00
|
|
|
int D_PAD::Compare( const D_PAD* padref, const D_PAD* padcmp )
|
2008-01-24 21:50:12 +00:00
|
|
|
{
|
2008-12-14 19:45:05 +00:00
|
|
|
int diff;
|
2008-01-24 21:50:12 +00:00
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_PadShape - padcmp->m_PadShape ) != 0 )
|
2008-01-24 21:50:12 +00:00
|
|
|
return diff;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_DrillShape - padcmp->m_DrillShape ) != 0)
|
2008-01-24 21:50:12 +00:00
|
|
|
return diff;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_Drill.x - padcmp->m_Drill.x ) != 0 )
|
2008-01-24 21:50:12 +00:00
|
|
|
return diff;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_Drill.y - padcmp->m_Drill.y ) != 0 )
|
2008-01-24 21:50:12 +00:00
|
|
|
return diff;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_Size.x - padcmp->m_Size.x ) != 0 )
|
2008-01-24 21:50:12 +00:00
|
|
|
return diff;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_Size.y - padcmp->m_Size.y ) != 0 )
|
2008-01-24 21:50:12 +00:00
|
|
|
return diff;
|
2011-09-07 19:41:04 +00:00
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_Offset.x - padcmp->m_Offset.x ) != 0 )
|
2011-12-02 15:09:57 +00:00
|
|
|
return diff;
|
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_Offset.y - padcmp->m_Offset.y ) != 0 )
|
2011-12-02 15:09:57 +00:00
|
|
|
return diff;
|
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_DeltaSize.x - padcmp->m_DeltaSize.x ) != 0 )
|
2011-12-02 15:09:57 +00:00
|
|
|
return diff;
|
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
if( ( diff = padref->m_DeltaSize.y - padcmp->m_DeltaSize.y ) != 0 )
|
2008-01-24 21:50:12 +00:00
|
|
|
return diff;
|
|
|
|
|
2011-12-02 16:55:31 +00:00
|
|
|
// Dick: specctra_export needs this
|
|
|
|
// Lorenzo: gencad also needs it to implement padstacks!
|
|
|
|
if( ( diff = padref->m_layerMask - padcmp->m_layerMask ) != 0 )
|
2008-01-24 21:50:12 +00:00
|
|
|
return diff;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-14 19:41:20 +00:00
|
|
|
wxString D_PAD::ShowPadShape() const
|
2008-01-24 21:50:12 +00:00
|
|
|
{
|
2011-07-14 19:41:20 +00:00
|
|
|
switch( m_PadShape )
|
2008-01-24 21:50:12 +00:00
|
|
|
{
|
2008-12-14 19:45:05 +00:00
|
|
|
case PAD_CIRCLE:
|
2011-09-07 19:41:04 +00:00
|
|
|
return _( "Circle" );
|
2008-12-14 19:45:05 +00:00
|
|
|
|
|
|
|
case PAD_OVAL:
|
2011-09-07 19:41:04 +00:00
|
|
|
return _( "Oval" );
|
2008-12-14 19:45:05 +00:00
|
|
|
|
|
|
|
case PAD_RECT:
|
2011-09-07 19:41:04 +00:00
|
|
|
return _( "Rect" );
|
2008-12-14 19:45:05 +00:00
|
|
|
|
|
|
|
case PAD_TRAPEZOID:
|
2011-09-07 19:41:04 +00:00
|
|
|
return _( "Trap" );
|
2008-12-14 19:45:05 +00:00
|
|
|
|
|
|
|
default:
|
2011-09-07 19:41:04 +00:00
|
|
|
return wxT( "??Unknown??" );
|
2008-01-24 21:50:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-14 19:41:20 +00:00
|
|
|
wxString D_PAD::ShowPadAttr() const
|
2008-01-24 21:50:12 +00:00
|
|
|
{
|
2012-02-19 04:02:19 +00:00
|
|
|
switch( GetAttribute() )
|
2008-01-24 21:50:12 +00:00
|
|
|
{
|
2008-12-14 19:45:05 +00:00
|
|
|
case PAD_STANDARD:
|
2011-09-07 19:41:04 +00:00
|
|
|
return _( "Std" );
|
2008-12-14 19:45:05 +00:00
|
|
|
|
|
|
|
case PAD_SMD:
|
2011-09-07 19:41:04 +00:00
|
|
|
return _( "Smd" );
|
2008-12-14 19:45:05 +00:00
|
|
|
|
|
|
|
case PAD_CONN:
|
2011-09-07 19:41:04 +00:00
|
|
|
return _( "Conn" );
|
2008-12-14 19:45:05 +00:00
|
|
|
|
|
|
|
case PAD_HOLE_NOT_PLATED:
|
2011-09-07 19:41:04 +00:00
|
|
|
return _( "Not Plated" );
|
2008-12-14 19:45:05 +00:00
|
|
|
|
|
|
|
default:
|
2011-09-07 19:41:04 +00:00
|
|
|
return wxT( "??Unkown??" );
|
2008-01-24 21:50:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-14 15:42:44 +00:00
|
|
|
wxString D_PAD::GetSelectMenuText() const
|
|
|
|
{
|
|
|
|
wxString text;
|
2011-10-22 18:21:57 +00:00
|
|
|
wxString padlayers;
|
2011-08-19 13:08:24 +00:00
|
|
|
BOARD * board = GetBoard();
|
2011-07-14 15:42:44 +00:00
|
|
|
|
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
if ( (m_layerMask & ALL_CU_LAYERS) == ALL_CU_LAYERS )
|
2011-10-22 18:21:57 +00:00
|
|
|
padlayers = _("all copper layers");
|
2011-09-07 19:41:04 +00:00
|
|
|
else if( (m_layerMask & LAYER_BACK ) == LAYER_BACK )
|
2011-10-22 18:21:57 +00:00
|
|
|
padlayers = board->GetLayerName(LAYER_N_BACK);
|
2011-09-07 19:41:04 +00:00
|
|
|
else if( (m_layerMask & LAYER_FRONT) == LAYER_FRONT )
|
2011-10-22 18:21:57 +00:00
|
|
|
padlayers = board->GetLayerName(LAYER_N_FRONT);
|
2011-07-14 15:42:44 +00:00
|
|
|
else
|
2011-10-22 18:21:57 +00:00
|
|
|
padlayers = _( "???" );
|
2011-07-14 15:42:44 +00:00
|
|
|
|
2011-10-22 18:21:57 +00:00
|
|
|
text.Printf( _( "Pad [%s] (%s) of %s" ),
|
2011-12-12 08:37:05 +00:00
|
|
|
GetChars(GetPadName() ), GetChars( padlayers ),
|
2011-10-22 18:21:57 +00:00
|
|
|
GetChars(( (MODULE*) GetParent() )->GetReference() ) );
|
2011-07-14 15:42:44 +00:00
|
|
|
|
|
|
|
return text;
|
|
|
|
}
|
|
|
|
|
2012-03-17 14:39:27 +00:00
|
|
|
EDA_ITEM* D_PAD::Clone() const
|
2012-01-14 19:50:32 +00:00
|
|
|
{
|
|
|
|
return new D_PAD( *this );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-07-14 19:41:20 +00:00
|
|
|
#if defined(DEBUG)
|
2011-07-14 15:42:44 +00:00
|
|
|
|
2011-12-14 17:25:42 +00:00
|
|
|
void D_PAD::Show( int nestLevel, std::ostream& os ) const
|
2007-08-07 06:21:19 +00:00
|
|
|
{
|
2011-09-07 19:41:04 +00:00
|
|
|
char padname[5] = { m_Padname[0], m_Padname[1], m_Padname[2], m_Padname[3], 0 };
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
char layerMask[16];
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2011-09-07 19:41:04 +00:00
|
|
|
sprintf( layerMask, "0x%08X", m_layerMask );
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
// for now, make it look like XML:
|
|
|
|
NestedSpace( nestLevel, os ) << '<' << GetClass().Lower().mb_str() <<
|
2011-07-14 19:41:20 +00:00
|
|
|
" shape=\"" << ShowPadShape() << '"' <<
|
|
|
|
" attr=\"" << ShowPadAttr( ) << '"' <<
|
2008-12-14 19:45:05 +00:00
|
|
|
" num=\"" << padname << '"' <<
|
|
|
|
" net=\"" << m_Netname.mb_str() << '"' <<
|
|
|
|
" netcode=\"" << GetNet() << '"' <<
|
2011-11-24 17:32:51 +00:00
|
|
|
" layerMask=\"" << layerMask << '"' << m_Pos << "/>\n";
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
// NestedSpace( nestLevel+1, os ) << m_Text.mb_str() << '\n';
|
2007-08-30 08:15:05 +00:00
|
|
|
|
2009-11-12 15:43:38 +00:00
|
|
|
// NestedSpace( nestLevel, os ) << "</" << GetClass().Lower().mb_str()
|
|
|
|
// << ">\n";
|
2007-08-07 06:21:19 +00:00
|
|
|
}
|
|
|
|
|
2008-12-14 19:45:05 +00:00
|
|
|
|
2007-08-07 06:21:19 +00:00
|
|
|
#endif
|