Add some distances to a few DRC errors.

This commit is contained in:
Jeff Young 2020-04-24 19:56:44 +01:00
parent e14e3faed6
commit 32db9eb0f1
15 changed files with 273 additions and 116 deletions

View File

@ -48,8 +48,7 @@ PANEL_SETUP_SEVERITIES::PANEL_SETUP_SEVERITIES( PAGED_DIALOG* aParent, RC_ITEM&
for( int errorCode = m_firstErrorCode; errorCode <= m_lastErrorCode; ++errorCode ) for( int errorCode = m_firstErrorCode; errorCode <= m_lastErrorCode; ++errorCode )
{ {
aDummyItem.SetErrorCode( errorCode ); wxString msg = aDummyItem.GetErrorText( errorCode );
wxString msg = aDummyItem.GetErrorText();
// When msg is empty, for some reason, the current errorCode is not supported // When msg is empty, for some reason, the current errorCode is not supported
// by the RC_ITEM aDummyItem. // by the RC_ITEM aDummyItem.

View File

@ -33,6 +33,15 @@
#define WX_DATAVIEW_WINDOW_PADDING 6 #define WX_DATAVIEW_WINDOW_PADDING 6
wxString RC_ITEM::GetErrorMessage() const
{
if( m_errorMessage.IsEmpty() )
return GetErrorText( m_errorCode );
else
return m_errorMessage;
}
wxString RC_ITEM::ShowCoord( EDA_UNITS aUnits, const wxPoint& aPos ) wxString RC_ITEM::ShowCoord( EDA_UNITS aUnits, const wxPoint& aPos )
{ {
return wxString::Format( "@(%s, %s)", return wxString::Format( "@(%s, %s)",
@ -52,13 +61,11 @@ wxString RC_ITEM::ShowReport( EDA_UNITS aUnits, const std::map<KIID, EDA_ITEM*>&
if( m_auxItemUuid != niluuid ) if( m_auxItemUuid != niluuid )
auxItem = aItemMap.at( m_auxItemUuid ); auxItem = aItemMap.at( m_auxItemUuid );
wxString msg = m_errorMessage.IsEmpty() ? GetErrorText() : m_errorMessage;
if( mainItem && auxItem ) if( mainItem && auxItem )
{ {
return wxString::Format( wxT( "ErrType(%d): %s\n %s: %s\n %s: %s\n" ), return wxString::Format( wxT( "ErrType(%d): %s\n %s: %s\n %s: %s\n" ),
m_errorCode, GetErrorCode(),
msg, GetErrorMessage(),
ShowCoord( aUnits, mainItem->GetPosition() ), ShowCoord( aUnits, mainItem->GetPosition() ),
mainItem->GetSelectMenuText( aUnits ), mainItem->GetSelectMenuText( aUnits ),
ShowCoord( aUnits, auxItem->GetPosition() ), ShowCoord( aUnits, auxItem->GetPosition() ),
@ -67,16 +74,16 @@ wxString RC_ITEM::ShowReport( EDA_UNITS aUnits, const std::map<KIID, EDA_ITEM*>&
else if( mainItem ) else if( mainItem )
{ {
return wxString::Format( wxT( "ErrType(%d): %s\n %s: %s\n" ), return wxString::Format( wxT( "ErrType(%d): %s\n %s: %s\n" ),
m_errorCode, GetErrorCode(),
msg, GetErrorMessage(),
ShowCoord( aUnits, mainItem->GetPosition() ), ShowCoord( aUnits, mainItem->GetPosition() ),
mainItem->GetSelectMenuText( aUnits ) ); mainItem->GetSelectMenuText( aUnits ) );
} }
else else
{ {
return wxString::Format( wxT( "ErrType(%d): %s\n" ), return wxString::Format( wxT( "ErrType(%d): %s\n" ),
m_errorCode, GetErrorCode(),
msg ); GetErrorMessage() );
} }
} }
@ -254,7 +261,7 @@ void RC_TREE_MODEL::GetValue( wxVariant& aVariant,
excluded ? _( "Excluded " ) : wxString( "" ), excluded ? _( "Excluded " ) : wxString( "" ),
error ? _( "Error: " ) : _( "Warning: " ) ); error ? _( "Error: " ) : _( "Warning: " ) );
aVariant = prefix + rcItem->GetErrorText(); aVariant = prefix + rcItem->GetErrorMessage();
} }
break; break;

View File

@ -140,9 +140,15 @@ public:
/** /**
* Function GetErrorText * Function GetErrorText
* returns the string form of a drc error code. * returns the string form of a RC error code
*/ */
virtual wxString GetErrorText() const = 0; virtual wxString GetErrorText( int aCode = -1 ) const = 0;
/**
* Function GetErrorMessage
* returns the error message of a RC_ITEM
*/
virtual wxString GetErrorMessage() const;
/** /**
* Function ShowCoord * Function ShowCoord

View File

@ -394,6 +394,7 @@ void DIALOG_ERC::OnERCItemRClick( wxDataViewEvent& aEvent )
RC_ITEM* rcItem = node->m_RcItem; RC_ITEM* rcItem = node->m_RcItem;
wxString listName; wxString listName;
wxMenu menu; wxMenu menu;
wxString msg;
switch( GetSeverity( rcItem->GetErrorCode() ) ) switch( GetSeverity( rcItem->GetErrorCode() ) )
{ {
@ -417,20 +418,23 @@ void DIALOG_ERC::OnERCItemRClick( wxDataViewEvent& aEvent )
if( GetSeverity( rcItem->GetErrorCode() ) == RPT_SEVERITY_WARNING ) if( GetSeverity( rcItem->GetErrorCode() ) == RPT_SEVERITY_WARNING )
{ {
menu.Append( 3, wxString::Format( _( "Change severity to Error for all '%s' violations" ), msg.Printf( _( "Change severity to Error for all '%s' violations" ),
rcItem->GetErrorText(), rcItem->GetErrorText( rcItem->GetErrorCode() ),
_( "Violation severities can also be edited in the Board Setup... dialog" ) ) ); _( "Violation severities can also be edited in the Board Setup... dialog" ) );
menu.Append( 3, msg );
} }
else else
{ {
menu.Append( 4, wxString::Format( _( "Change severity to Warning for all '%s' violations" ), msg.Printf( _( "Change severity to Warning for all '%s' violations" ),
rcItem->GetErrorText(), rcItem->GetErrorText( rcItem->GetErrorCode() ),
_( "Violation severities can also be edited in the Board Setup... dialog" ) ) ); _( "Violation severities can also be edited in the Board Setup... dialog" ) );
menu.Append( 4, msg );
} }
menu.Append( 5, wxString::Format( _( "Ignore all '%s' violations" ), msg.Printf( _( "Ignore all '%s' violations" ),
rcItem->GetErrorText() ), rcItem->GetErrorText( rcItem->GetErrorCode() ),
_( "Violations will not be checked or reported" ) ); _( "Violations will not be checked or reported" ) );
menu.Append( 5, msg );
menu.AppendSeparator(); menu.AppendSeparator();

View File

@ -30,8 +30,11 @@
#include <erc_item.h> #include <erc_item.h>
wxString ERC_ITEM::GetErrorText() const wxString ERC_ITEM::GetErrorText( int aErrorCode ) const
{ {
if( aErrorCode < 0 )
aErrorCode = m_errorCode;
switch( m_errorCode ) switch( m_errorCode )
{ {
case ERCE_UNSPECIFIED: case ERCE_UNSPECIFIED:

View File

@ -37,9 +37,9 @@ public:
/** /**
* Function GetErrorText * Function GetErrorText
* returns the string form of a drc error code. * returns the string form of an erc error code.
*/ */
wxString GetErrorText() const override; wxString GetErrorText( int aErrorCode = -1 ) const override;
}; };

View File

@ -117,7 +117,7 @@ void SCH_MARKER::Print( RENDER_SETTINGS* aSettings, const wxPoint& aOffset )
bool SCH_MARKER::Matches( wxFindReplaceData& aSearchData, void* aAuxData ) bool SCH_MARKER::Matches( wxFindReplaceData& aSearchData, void* aAuxData )
{ {
return SCH_ITEM::Matches( m_rcItem->GetErrorText(), aSearchData ); return SCH_ITEM::Matches( m_rcItem->GetErrorMessage(), aSearchData );
} }
@ -130,7 +130,7 @@ const EDA_RECT SCH_MARKER::GetBoundingBox() const
void SCH_MARKER::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, MSG_PANEL_ITEMS& aList ) void SCH_MARKER::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, MSG_PANEL_ITEMS& aList )
{ {
aList.push_back( MSG_PANEL_ITEM( _( "Electronics Rule Check Error" ), aList.push_back( MSG_PANEL_ITEM( _( "Electronics Rule Check Error" ),
m_rcItem->GetErrorText(), DARKRED ) ); m_rcItem->GetErrorMessage(), DARKRED ) );
} }

View File

@ -100,7 +100,7 @@ bool MARKER_PCB::IsOnLayer( PCB_LAYER_ID aLayer ) const
void MARKER_PCB::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList ) void MARKER_PCB::GetMsgPanelInfo( EDA_DRAW_FRAME* aFrame, std::vector<MSG_PANEL_ITEM>& aList )
{ {
aList.emplace_back( _( "Type" ), _( "Marker" ), DARKCYAN ); aList.emplace_back( _( "Type" ), _( "Marker" ), DARKCYAN );
aList.emplace_back( _( "Violation" ), m_rcItem->GetErrorText(), RED ); aList.emplace_back( _( "Violation" ), m_rcItem->GetErrorMessage(), RED );
wxString mainText; wxString mainText;
wxString auxText; wxString auxText;
@ -140,7 +140,10 @@ void MARKER_PCB::Flip(const wxPoint& aCentre, bool aFlipLeftRight )
wxString MARKER_PCB::GetSelectMenuText( EDA_UNITS aUnits ) const wxString MARKER_PCB::GetSelectMenuText( EDA_UNITS aUnits ) const
{ {
return wxString::Format( _( "Marker (%s)" ), m_rcItem->GetErrorText() ); // m_rcItem->GetErrorMessage() could be used instead, but is probably too long
// for menu duty.
return wxString::Format( _( "Marker (%s)" ),
m_rcItem->GetErrorText( m_rcItem->GetErrorCode() ) );
} }

View File

@ -84,7 +84,7 @@ public:
bool Matches( wxFindReplaceData& aSearchData, void* aAuxData ) override bool Matches( wxFindReplaceData& aSearchData, void* aAuxData ) override
{ {
return BOARD_ITEM::Matches( m_rcItem->GetErrorText(), aSearchData ); return BOARD_ITEM::Matches( m_rcItem->GetErrorMessage(), aSearchData );
} }
wxString GetSelectMenuText( EDA_UNITS aUnits ) const override; wxString GetSelectMenuText( EDA_UNITS aUnits ) const override;

View File

@ -285,6 +285,7 @@ void DIALOG_DRC::OnDRCItemRClick( wxDataViewEvent& aEvent )
RC_ITEM* rcItem = node->m_RcItem; RC_ITEM* rcItem = node->m_RcItem;
wxString listName; wxString listName;
wxMenu menu; wxMenu menu;
wxString msg;
switch( m_BrdSettings.m_DRCSeverities[ rcItem->GetErrorCode() ] ) switch( m_BrdSettings.m_DRCSeverities[ rcItem->GetErrorCode() ] )
{ {
@ -308,20 +309,23 @@ void DIALOG_DRC::OnDRCItemRClick( wxDataViewEvent& aEvent )
if( m_BrdSettings.m_DRCSeverities[ rcItem->GetErrorCode() ] == RPT_SEVERITY_WARNING ) if( m_BrdSettings.m_DRCSeverities[ rcItem->GetErrorCode() ] == RPT_SEVERITY_WARNING )
{ {
menu.Append( 3, wxString::Format( _( "Change severity to Error for all '%s' violations" ), msg.Printf( _( "Change severity to Error for all '%s' violations" ),
rcItem->GetErrorText(), rcItem->GetErrorText( rcItem->GetErrorCode() ),
_( "Violation severities can also be edited in the Board Setup... dialog" ) ) ); _( "Violation severities can also be edited in the Board Setup... dialog" ) );
menu.Append( 3, msg );
} }
else else
{ {
menu.Append( 4, wxString::Format( _( "Change severity to Warning for all '%s' violations" ), msg.Printf( _( "Change severity to Warning for all '%s' violations" ),
rcItem->GetErrorText(), rcItem->GetErrorText( rcItem->GetErrorCode() ),
_( "Violation severities can also be edited in the Board Setup... dialog" ) ) ); _( "Violation severities can also be edited in the Board Setup... dialog" ) );
menu.Append( 4, msg );
} }
menu.Append( 5, wxString::Format( _( "Ignore all '%s' violations" ), msg.Printf( _( "Ignore all '%s' violations" ),
rcItem->GetErrorText() ), rcItem->GetErrorText( rcItem->GetErrorCode() ),
_( "Violations will not be checked or reported" ) ); _( "Violations will not be checked or reported" ) );
menu.Append( 5, msg );
menu.AppendSeparator(); menu.AppendSeparator();

View File

@ -696,6 +696,7 @@ void DRC::testDrilledHoles()
std::vector<DRILLED_HOLE> holes; std::vector<DRILLED_HOLE> holes;
DRILLED_HOLE hole; DRILLED_HOLE hole;
wxString msg;
for( MODULE* mod : m_pcb->Modules() ) for( MODULE* mod : m_pcb->Modules() )
{ {
@ -735,10 +736,18 @@ void DRC::testDrilledHoles()
if( checkHole.m_location == refHole.m_location ) if( checkHole.m_location == refHole.m_location )
continue; continue;
if( KiROUND( GetLineLength( checkHole.m_location, refHole.m_location ) ) int actual = KiROUND( GetLineLength( checkHole.m_location, refHole.m_location ) );
< checkHole.m_drillRadius + refHole.m_drillRadius + holeToHoleMin ) actual = std::max( 0, actual - checkHole.m_drillRadius - refHole.m_drillRadius );
if( actual < holeToHoleMin )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_DRILLED_HOLES_TOO_CLOSE ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_DRILLED_HOLES_TOO_CLOSE );
msg.Printf( drcItem->GetErrorText() + _( " (minimum %s; actual %s)" ),
MessageTextFromValue( userUnits(), holeToHoleMin, true ),
MessageTextFromValue( userUnits(), actual, true ) );
drcItem->SetErrorMessage( msg );
drcItem->SetItems( refHole.m_owner, checkHole.m_owner ); drcItem->SetItems( refHole.m_owner, checkHole.m_owner );
MARKER_PCB* marker = new MARKER_PCB( drcItem, refHole.m_location ); MARKER_PCB* marker = new MARKER_PCB( drcItem, refHole.m_location );
@ -1319,7 +1328,10 @@ bool DRC::doPadToPadsDrc( D_PAD* aRefPad, D_PAD** aStart, D_PAD** aEnd, int x_li
for( D_PAD** pad_list = aStart; pad_list<aEnd; ++pad_list ) for( D_PAD** pad_list = aStart; pad_list<aEnd; ++pad_list )
{ {
D_PAD* pad = *pad_list; D_PAD* pad = *pad_list;
int allowed;
int actual;
wxString msg;
if( pad == aRefPad ) if( pad == aRefPad )
continue; continue;
@ -1363,9 +1375,15 @@ bool DRC::doPadToPadsDrc( D_PAD* aRefPad, D_PAD** aStart, D_PAD** aEnd, int x_li
PAD_SHAPE_OVAL : PAD_SHAPE_CIRCLE ); PAD_SHAPE_OVAL : PAD_SHAPE_CIRCLE );
dummypad.SetOrientation( pad->GetOrientation() ); dummypad.SetOrientation( pad->GetOrientation() );
if( !checkClearancePadToPad( aRefPad, &dummypad ) ) if( !checkClearancePadToPad( aRefPad, &dummypad, &allowed, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_HOLE_NEAR_PAD ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_HOLE_NEAR_PAD );
msg.Printf( drcItem->GetErrorText() + _( "(minimum %s; actual %s)" ),
MessageTextFromValue( userUnits(), allowed, true ),
MessageTextFromValue( userUnits(), actual, true ) );
drcItem->SetErrorMessage( msg );
drcItem->SetItems( pad, aRefPad ); drcItem->SetItems( pad, aRefPad );
MARKER_PCB* marker = new MARKER_PCB( drcItem, pad->GetPosition() ); MARKER_PCB* marker = new MARKER_PCB( drcItem, pad->GetPosition() );
@ -1382,9 +1400,15 @@ bool DRC::doPadToPadsDrc( D_PAD* aRefPad, D_PAD** aStart, D_PAD** aEnd, int x_li
PAD_SHAPE_OVAL : PAD_SHAPE_CIRCLE ); PAD_SHAPE_OVAL : PAD_SHAPE_CIRCLE );
dummypad.SetOrientation( aRefPad->GetOrientation() ); dummypad.SetOrientation( aRefPad->GetOrientation() );
if( !checkClearancePadToPad( pad, &dummypad ) ) if( !checkClearancePadToPad( pad, &dummypad, &allowed, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_HOLE_NEAR_PAD ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_HOLE_NEAR_PAD );
msg.Printf( drcItem->GetErrorText() + _( "(minimum %s; actual %s)" ),
MessageTextFromValue( userUnits(), allowed, true ),
MessageTextFromValue( userUnits(), actual, true ) );
drcItem->SetErrorMessage( msg );
drcItem->SetItems( aRefPad, pad ); drcItem->SetItems( aRefPad, pad );
MARKER_PCB* marker = new MARKER_PCB( drcItem, aRefPad->GetPosition() ); MARKER_PCB* marker = new MARKER_PCB( drcItem, aRefPad->GetPosition() );
@ -1420,9 +1444,15 @@ bool DRC::doPadToPadsDrc( D_PAD* aRefPad, D_PAD** aStart, D_PAD** aEnd, int x_li
continue; continue;
} }
if( !checkClearancePadToPad( aRefPad, pad ) ) if( !checkClearancePadToPad( aRefPad, pad, &allowed, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_PAD_NEAR_PAD1 ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_PAD_NEAR_PAD1 );
msg.Printf( drcItem->GetErrorText() + _( "(minimum %s; actual %s)" ),
MessageTextFromValue( userUnits(), allowed, true ),
MessageTextFromValue( userUnits(), actual, true ) );
drcItem->SetErrorMessage( msg );
drcItem->SetItems( aRefPad, pad ); drcItem->SetItems( aRefPad, pad );
MARKER_PCB* marker = new MARKER_PCB( drcItem, aRefPad->GetPosition() ); MARKER_PCB* marker = new MARKER_PCB( drcItem, aRefPad->GetPosition() );

View File

@ -301,9 +301,11 @@ private:
/** /**
* @param aRefPad The reference pad to check * @param aRefPad The reference pad to check
* @param aPad Another pad to check against * @param aPad Another pad to check against
* @param aAllowed [out] is the allowed distance (only guaranteed to be set for violations)
* @param aActual [out] it the actual difference (only guaranteed to be set for violations)
* @return bool - true if clearance between aRefPad and aPad is >= dist_min, else false * @return bool - true if clearance between aRefPad and aPad is >= dist_min, else false
*/ */
bool checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad ); bool checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad, int* aAllowed, int* aActual );
/** /**
@ -316,11 +318,13 @@ private:
* @param aPad Is the pad involved in the check * @param aPad Is the pad involved in the check
* @param aSegmentWidth width of the segment to test * @param aSegmentWidth width of the segment to test
* @param aMinDist Is the minimum clearance needed * @param aMinDist Is the minimum clearance needed
* @param aActualDist [out] Is the actual clearance (only guarantted to be set on violations)
* *
* @return true distance >= dist_min, * @return true distance >= dist_min,
* false if distance < dist_min * false if distance < dist_min
*/ */
bool checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMinDist ); bool checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMinDist,
int* aActualDist );
/** /**
@ -329,13 +333,14 @@ private:
* The segment is expected starting at 0,0, and on the X axis * The segment is expected starting at 0,0, and on the X axis
* (used to test DRC between a segment and a round pad, via or round end of a track * (used to test DRC between a segment and a round pad, via or round end of a track
* @param aCentre The coordinate of the circle's center * @param aCentre The coordinate of the circle's center
* @param aRadius A "keep out" radius centered over the circle * @param aAllowed A "keep out" radius centered over the circle
* @param aLength The length of the segment (i.e. coordinate of end, because it is on * @param aLength The length of the segment (i.e. coordinate of end, because it is on
* the X axis) * the X axis)
* @param aActual [out] is the actual distance (only guaranteed to be set on violations)
* @return bool - true if distance >= radius, else * @return bool - true if distance >= radius, else
* false when distance < aRadius * false when distance < aAllowed
*/ */
static bool checkMarginToCircle( wxPoint aCentre, int aRadius, int aLength ); static bool checkMarginToCircle( wxPoint aCentre, int aAllowed, int aLength, int* aActual );
/** /**

View File

@ -46,17 +46,24 @@
* i.e if for each edge of the first polygon distance from each edge of the other polygon * i.e if for each edge of the first polygon distance from each edge of the other polygon
* is >= aDist * is >= aDist
*/ */
bool poly2polyDRC( wxPoint* aTref, int aTrefCount, wxPoint* aTtest, int aTtestCount, int aDist ) bool poly2polyDRC( wxPoint* aTref, int aTrefCount, wxPoint* aTtest, int aTtestCount,
int aAllowedDist, int* actualDist )
{ {
/* Test if one polygon is contained in the other and thus the polygon overlap. /* Test if one polygon is contained in the other and thus the polygon overlap.
* This case is not covered by the following check if one polygone is * This case is not covered by the following check if one polygone is
* completely contained in the other (because edges don't intersect)! * completely contained in the other (because edges don't intersect)!
*/ */
if( TestPointInsidePolygon( aTref, aTrefCount, aTtest[0] ) ) if( TestPointInsidePolygon( aTref, aTrefCount, aTtest[0] ) )
{
*actualDist = 0;
return false; return false;
}
if( TestPointInsidePolygon( aTtest, aTtestCount, aTref[0] ) ) if( TestPointInsidePolygon( aTtest, aTtestCount, aTref[0] ) )
{
*actualDist = 0;
return false; return false;
}
for( int ii = 0, jj = aTrefCount - 1; ii < aTrefCount; jj = ii, ii++ ) for( int ii = 0, jj = aTrefCount - 1; ii < aTrefCount; jj = ii, ii++ )
{ {
@ -70,8 +77,17 @@ bool poly2polyDRC( wxPoint* aTref, int aTrefCount, wxPoint* aTtest, int aTtestCo
aTtest[kk].x, aTtest[kk].y, aTtest[ll].x, aTtest[ll].y, aTtest[kk].x, aTtest[kk].y, aTtest[ll].x, aTtest[ll].y,
nullptr, nullptr, &d ); nullptr, nullptr, &d );
if( intersect || ( d < aDist ) ) if( intersect )
{
*actualDist = 0;
return false; return false;
}
if( d < aAllowedDist )
{
*actualDist = KiROUND( d );
return false;
}
} }
} }
@ -82,25 +98,36 @@ bool poly2polyDRC( wxPoint* aTref, int aTrefCount, wxPoint* aTtest, int aTtestCo
/* /*
* compare a trapezoid (can be rectangle) and a segment and return true if distance > aDist * compare a trapezoid (can be rectangle) and a segment and return true if distance > aDist
*/ */
bool poly2segmentDRC( wxPoint* aTref, int aTrefCount, wxPoint aSegStart, wxPoint aSegEnd, int aDist ) bool poly2segmentDRC( wxPoint* aTref, int aTrefCount, wxPoint aSegStart, wxPoint aSegEnd,
int aDist, int* aActual )
{ {
/* Test if the segment is contained in the polygon. /* Test if the segment is contained in the polygon.
* This case is not covered by the following check if the segment is * This case is not covered by the following check if the segment is
* completely contained in the polygon (because edges don't intersect)! * completely contained in the polygon (because edges don't intersect)!
*/ */
if( TestPointInsidePolygon( aTref, aTrefCount, aSegStart ) ) if( TestPointInsidePolygon( aTref, aTrefCount, aSegStart ) )
{
*aActual = 0;
return false; return false;
}
for( int ii = 0, jj = aTrefCount-1; ii < aTrefCount; jj = ii, ii++ ) for( int ii = 0, jj = aTrefCount-1; ii < aTrefCount; jj = ii, ii++ )
{ // for all edges in polygon { // for all edges in polygon
double d; double d;
int intersect = TestForIntersectionOfStraightLineSegments(
aTref[ii].x, aTref[ii].y, aTref[jj].x, aTref[jj].y,
aSegStart.x, aSegStart.y, aSegEnd.x, aSegEnd.y,
NULL, NULL, &d );
if( intersect || ( d < aDist) ) if( TestForIntersectionOfStraightLineSegments( aTref[ii].x, aTref[ii].y, aTref[jj].x,
aTref[jj].y, aSegStart.x, aSegStart.y,
aSegEnd.x, aSegEnd.y, NULL, NULL, &d ) )
{
*aActual = 0;
return false; return false;
}
if( d < aDist )
{
*aActual = KiROUND( d );
return false;
}
} }
return true; return true;
@ -128,6 +155,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
int net_code_ref = aRefSeg->GetNetCode(); int net_code_ref = aRefSeg->GetNetCode();
int ref_seg_clearance = netclass->GetClearance(); int ref_seg_clearance = netclass->GetClearance();
int ref_seg_width = aRefSeg->GetWidth(); int ref_seg_width = aRefSeg->GetWidth();
int actual;
/******************************************/ /******************************************/
@ -311,7 +339,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
m_padToTestPos = dummypad.GetPosition() - origin; m_padToTestPos = dummypad.GetPosition() - origin;
if( !checkClearanceSegmToPad( &dummypad, ref_seg_width, ref_seg_clearance ) ) if( !checkClearanceSegmToPad( &dummypad, ref_seg_width, ref_seg_clearance, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_NEAR_THROUGH_HOLE ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_NEAR_THROUGH_HOLE );
drcItem->SetItems( aRefSeg, pad ); drcItem->SetItems( aRefSeg, pad );
@ -337,7 +365,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
m_padToTestPos = shape_pos - origin; m_padToTestPos = shape_pos - origin;
int segToPadClearance = std::max( ref_seg_clearance, pad->GetClearance() ); int segToPadClearance = std::max( ref_seg_clearance, pad->GetClearance() );
if( !checkClearanceSegmToPad( pad, ref_seg_width, segToPadClearance ) ) if( !checkClearanceSegmToPad( pad, ref_seg_width, segToPadClearance, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_NEAR_PAD ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_NEAR_PAD );
drcItem->SetItems( aRefSeg, pad ); drcItem->SetItems( aRefSeg, pad );
@ -414,7 +442,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
RotatePoint( &delta, angle ); RotatePoint( &delta, angle );
RotatePoint( &segStartPoint, angle ); RotatePoint( &segStartPoint, angle );
if( !checkMarginToCircle( segStartPoint, w_dist, delta.x ) ) if( !checkMarginToCircle( segStartPoint, w_dist, delta.x, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_VIA_NEAR_TRACK ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_VIA_NEAR_TRACK );
drcItem->SetItems( aRefSeg, track ); drcItem->SetItems( aRefSeg, track );
@ -443,7 +471,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
if( track->Type() == PCB_VIA_T ) if( track->Type() == PCB_VIA_T )
{ {
if( checkMarginToCircle( segStartPoint, w_dist, m_segmLength ) ) if( checkMarginToCircle( segStartPoint, w_dist, m_segmLength, &actual ) )
continue; continue;
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_NEAR_VIA ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_NEAR_VIA );
@ -488,7 +516,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
return; return;
} }
if( !checkMarginToCircle( segStartPoint, w_dist, m_segmLength ) ) if( !checkMarginToCircle( segStartPoint, w_dist, m_segmLength, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS );
drcItem->SetItems( aRefSeg, track ); drcItem->SetItems( aRefSeg, track );
@ -519,7 +547,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
return; return;
} }
if( !checkMarginToCircle( segEndPoint, w_dist, m_segmLength ) ) if( !checkMarginToCircle( segEndPoint, w_dist, m_segmLength, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS );
drcItem->SetItems( aRefSeg, track ); drcItem->SetItems( aRefSeg, track );
@ -572,7 +600,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
} }
// At this point the drc error is due to an end near a reference segm end // At this point the drc error is due to an end near a reference segm end
if( !checkMarginToCircle( segStartPoint, w_dist, m_segmLength ) ) if( !checkMarginToCircle( segStartPoint, w_dist, m_segmLength, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS );
drcItem->SetItems( aRefSeg, track ); drcItem->SetItems( aRefSeg, track );
@ -583,7 +611,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
if( !m_reportAllTrackErrors ) if( !m_reportAllTrackErrors )
return; return;
} }
if( !checkMarginToCircle( segEndPoint, w_dist, m_segmLength ) ) if( !checkMarginToCircle( segEndPoint, w_dist, m_segmLength, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS );
drcItem->SetItems( aRefSeg, track ); drcItem->SetItems( aRefSeg, track );
@ -664,7 +692,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
RotatePoint( &relStartPos, angle ); RotatePoint( &relStartPos, angle );
RotatePoint( &relEndPos, angle ); RotatePoint( &relEndPos, angle );
if( !checkMarginToCircle( relStartPos, w_dist, delta.x ) ) if( !checkMarginToCircle( relStartPos, w_dist, delta.x, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS );
drcItem->SetItems( aRefSeg, track ); drcItem->SetItems( aRefSeg, track );
@ -676,7 +704,7 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
return; return;
} }
if( !checkMarginToCircle( relEndPos, w_dist, delta.x ) ) if( !checkMarginToCircle( relEndPos, w_dist, delta.x, &actual ) )
{ {
DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS ); DRC_ITEM* drcItem = new DRC_ITEM( DRCE_TRACK_ENDS );
drcItem->SetItems( aRefSeg, track ); drcItem->SetItems( aRefSeg, track );
@ -781,13 +809,16 @@ void DRC::doTrackDrc( TRACK* aRefSeg, TRACKS::iterator aStartIt, TRACKS::iterato
} }
bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad ) bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad, int* aAllowed, int* aActual )
{ {
int dist; int dist;
double pad_angle; double pad_angle;
// Get the clearance between the 2 pads. this is the min distance between aRefPad and aPad // Get the clearance between the 2 pads. this is the min distance between aRefPad and aPad
int dist_min = aRefPad->GetClearance( aPad ); int dist_min = aRefPad->GetClearance( aPad );
int dist_extra = 0;
*aAllowed = dist_min;
// relativePadPos is the aPad shape position relative to the aRefPad shape position // relativePadPos is the aPad shape position relative to the aRefPad shape position
wxPoint relativePadPos = aPad->ShapePos() - aRefPad->ShapePos(); wxPoint relativePadPos = aPad->ShapePos() - aRefPad->ShapePos();
@ -876,7 +907,7 @@ bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad )
m_segmEnd.x = m_segmEnd.y = 0; m_segmEnd.x = m_segmEnd.y = 0;
m_padToTestPos = relativePadPos; m_padToTestPos = relativePadPos;
diag = checkClearanceSegmToPad( aPad, aRefPad->GetSize().x, dist_min ); diag = checkClearanceSegmToPad( aPad, aRefPad->GetSize().x, dist_min, aActual );
break; break;
case PAD_SHAPE_TRAPEZOID: case PAD_SHAPE_TRAPEZOID:
@ -891,7 +922,7 @@ bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad )
if( aRefPad->GetShape() == PAD_SHAPE_ROUNDRECT ) if( aRefPad->GetShape() == PAD_SHAPE_ROUNDRECT )
{ {
int padRadius = aRefPad->GetRoundRectCornerRadius(); int padRadius = aRefPad->GetRoundRectCornerRadius();
dist_min += padRadius; dist_extra = padRadius;
GetRoundRectCornerCenters( polyref, padRadius, wxPoint( 0, 0 ), aRefPad->GetSize(), GetRoundRectCornerCenters( polyref, padRadius, wxPoint( 0, 0 ), aRefPad->GetSize(),
aRefPad->GetOrientation() ); aRefPad->GetOrientation() );
} }
@ -939,7 +970,7 @@ bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad )
if( aPad->GetShape() == PAD_SHAPE_ROUNDRECT ) if( aPad->GetShape() == PAD_SHAPE_ROUNDRECT )
{ {
int padRadius = aPad->GetRoundRectCornerRadius(); int padRadius = aPad->GetRoundRectCornerRadius();
dist_min += padRadius; dist_extra = padRadius;
GetRoundRectCornerCenters( polycompare, padRadius, relativePadPos, aPad->GetSize(), GetRoundRectCornerCenters( polycompare, padRadius, relativePadPos, aPad->GetSize(),
aPad->GetOrientation() ); aPad->GetOrientation() );
} }
@ -986,17 +1017,23 @@ bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad )
{ {
const SHAPE_LINE_CHAIN& refpoly = polysetref.COutline( 0 ); const SHAPE_LINE_CHAIN& refpoly = polysetref.COutline( 0 );
// And now test polygons: // And now test polygons:
if( !poly2polyDRC( (wxPoint*) &refpoly.CPoint( 0 ), refpoly.PointCount(), if( !poly2polyDRC((wxPoint*) &refpoly.CPoint( 0 ), refpoly.PointCount(),
polycompare, 4, dist_min ) ) // Therefore error polycompare, 4, dist_min + dist_extra, aActual ) )
{
*aActual = std::max( 0, *aActual - dist_extra );
diag = false; diag = false;
}
} }
else if( polysetref.OutlineCount() == 0 && polysetcompare.OutlineCount()) else if( polysetref.OutlineCount() == 0 && polysetcompare.OutlineCount())
{ {
const SHAPE_LINE_CHAIN& cmppoly = polysetcompare.COutline( 0 ); const SHAPE_LINE_CHAIN& cmppoly = polysetcompare.COutline( 0 );
// And now test polygons: // And now test polygons:
if( !poly2polyDRC( (wxPoint*) &cmppoly.CPoint( 0 ), cmppoly.PointCount(), if( !poly2polyDRC((wxPoint*) &cmppoly.CPoint( 0 ), cmppoly.PointCount(),
polyref, 4, dist_min ) ) // Therefore error polyref, 4, dist_min + dist_extra, aActual ) )
{
*aActual = std::max( 0, *aActual - dist_extra );
diag = false; diag = false;
}
} }
else if( polysetref.OutlineCount() && polysetcompare.OutlineCount() ) else if( polysetref.OutlineCount() && polysetcompare.OutlineCount() )
{ {
@ -1004,15 +1041,21 @@ bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad )
const SHAPE_LINE_CHAIN& cmppoly = polysetcompare.COutline( 0 ); const SHAPE_LINE_CHAIN& cmppoly = polysetcompare.COutline( 0 );
// And now test polygons: // And now test polygons:
if( !poly2polyDRC( (wxPoint*) &refpoly.CPoint( 0 ), refpoly.PointCount(), if( !poly2polyDRC((wxPoint*) &refpoly.CPoint( 0 ), refpoly.PointCount(),
(wxPoint*) &cmppoly.CPoint( 0 ), cmppoly.PointCount(), (wxPoint*) &cmppoly.CPoint( 0 ), cmppoly.PointCount(),
dist_min ) ) // Therefore error dist_min + dist_extra, aActual ) )
{
*aActual = std::max( 0, *aActual - dist_extra );
diag = false; diag = false;
}
} }
else else
{ {
if( !poly2polyDRC( polyref, 4, polycompare, 4, dist_min ) ) // Therefore error if( !poly2polyDRC( polyref, 4, polycompare, 4, dist_min + dist_extra, aActual ) )
{
*aActual = std::max( 0, *aActual - dist_extra );
diag = false; diag = false;
}
} }
break; break;
@ -1060,7 +1103,7 @@ bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad )
m_padToTestPos = relativePadPos - segstart; m_padToTestPos = relativePadPos - segstart;
// Use segment to pad check to test the second pad: // Use segment to pad check to test the second pad:
diag = checkClearanceSegmToPad( aPad, segm_width, dist_min ); diag = checkClearanceSegmToPad( aPad, segm_width, dist_min, aActual );
break; break;
} }
@ -1078,7 +1121,8 @@ bool DRC::checkClearancePadToPad( D_PAD* aRefPad, D_PAD* aPad )
* and its orientation is m_segmAngle (m_segmAngle must be already initialized) * and its orientation is m_segmAngle (m_segmAngle must be already initialized)
* and have aSegmentWidth. * and have aSegmentWidth.
*/ */
bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMinDist ) bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMinDist,
int* aActualDist )
{ {
// Note: // Note:
// we are using a horizontal segment for test, because we know here // we are using a horizontal segment for test, because we know here
@ -1119,7 +1163,15 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
* calculate pad coordinates in the X,Y axis with X axis = segment to test * calculate pad coordinates in the X,Y axis with X axis = segment to test
*/ */
RotatePoint( &m_padToTestPos, m_segmAngle ); RotatePoint( &m_padToTestPos, m_segmAngle );
return checkMarginToCircle( m_padToTestPos, distToLine + padHalfsize.x, m_segmLength );
if( !checkMarginToCircle( m_padToTestPos, aMinDist + segmHalfWidth + padHalfsize.x,
m_segmLength, aActualDist ) )
{
*aActualDist = std::max( 0, *aActualDist - segmHalfWidth - padHalfsize.x );
return false;
}
return true;
} }
/* calculate the bounding box of the pad, including the clearance and the segment width /* calculate the bounding box of the pad, including the clearance and the segment width
@ -1196,6 +1248,7 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
// If the segment legth is zero, only check the endpoints, skip the rectangle // If the segment legth is zero, only check the endpoints, skip the rectangle
if( m_segmLength && !checkLine( startPoint, endPoint ) ) if( m_segmLength && !checkLine( startPoint, endPoint ) )
{ {
// JEY TODO: set aActual
return false; return false;
} }
@ -1207,8 +1260,10 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
// to the segment: // to the segment:
RotatePoint( &cstart, m_segmAngle ); RotatePoint( &cstart, m_segmAngle );
if( !checkMarginToCircle( cstart, radius + distToLine, m_segmLength ) ) if( !checkMarginToCircle( cstart, aMinDist + radius + segmHalfWidth, m_segmLength,
aActualDist) )
{ {
*aActualDist = std::max( 0, *aActualDist - radius - segmHalfWidth );
return false; return false;
} }
@ -1216,8 +1271,10 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
RotatePoint( &cend, m_padToTestPos, orient ); RotatePoint( &cend, m_padToTestPos, orient );
RotatePoint( &cend, m_segmAngle ); RotatePoint( &cend, m_segmAngle );
if( !checkMarginToCircle( cend, radius + distToLine, m_segmLength ) ) if( !checkMarginToCircle( cend, aMinDist + radius + segmHalfWidth, m_segmLength,
aActualDist ) )
{ {
*aActualDist = std::max( 0, *aActualDist - radius - segmHalfWidth );
return false; return false;
} }
} }
@ -1243,7 +1300,10 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
m_ycliphi = m_padToTestPos.y + padHalfsize.y; m_ycliphi = m_padToTestPos.y + padHalfsize.y;
if( !checkLine( startPoint, endPoint ) ) if( !checkLine( startPoint, endPoint ) )
{
// JEY TODO: set aActual
return false; return false;
}
// Testing the second rectangle dimx , dimy + distToLine // Testing the second rectangle dimx , dimy + distToLine
m_xcliplo = m_padToTestPos.x - padHalfsize.x; m_xcliplo = m_padToTestPos.x - padHalfsize.x;
@ -1252,7 +1312,10 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
m_ycliphi = m_padToTestPos.y + padHalfsize.y + distToLine; m_ycliphi = m_padToTestPos.y + padHalfsize.y + distToLine;
if( !checkLine( startPoint, endPoint ) ) if( !checkLine( startPoint, endPoint ) )
{
// JEY TODO: set aActual
return false; return false;
}
// testing the 4 circles which are the clearance area of each corner: // testing the 4 circles which are the clearance area of each corner:
@ -1262,8 +1325,12 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
RotatePoint( &startPoint, m_padToTestPos, orient ); RotatePoint( &startPoint, m_padToTestPos, orient );
RotatePoint( &startPoint, m_segmAngle ); RotatePoint( &startPoint, m_segmAngle );
if( !checkMarginToCircle( startPoint, distToLine, m_segmLength ) ) if( !checkMarginToCircle( startPoint, aMinDist + segmHalfWidth, m_segmLength,
aActualDist ) )
{
*aActualDist = std::max( 0, *aActualDist - segmHalfWidth );
return false; return false;
}
// testing the right top corner of the rectangle // testing the right top corner of the rectangle
startPoint.x = m_padToTestPos.x + padHalfsize.x; startPoint.x = m_padToTestPos.x + padHalfsize.x;
@ -1271,8 +1338,12 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
RotatePoint( &startPoint, m_padToTestPos, orient ); RotatePoint( &startPoint, m_padToTestPos, orient );
RotatePoint( &startPoint, m_segmAngle ); RotatePoint( &startPoint, m_segmAngle );
if( !checkMarginToCircle( startPoint, distToLine, m_segmLength ) ) if( !checkMarginToCircle( startPoint, aMinDist + segmHalfWidth, m_segmLength,
aActualDist ) )
{
*aActualDist = std::max( 0, *aActualDist - segmHalfWidth );
return false; return false;
}
// testing the left bottom corner of the rectangle // testing the left bottom corner of the rectangle
startPoint.x = m_padToTestPos.x - padHalfsize.x; startPoint.x = m_padToTestPos.x - padHalfsize.x;
@ -1280,8 +1351,12 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
RotatePoint( &startPoint, m_padToTestPos, orient ); RotatePoint( &startPoint, m_padToTestPos, orient );
RotatePoint( &startPoint, m_segmAngle ); RotatePoint( &startPoint, m_segmAngle );
if( !checkMarginToCircle( startPoint, distToLine, m_segmLength ) ) if( !checkMarginToCircle( startPoint, aMinDist + segmHalfWidth, m_segmLength,
aActualDist ) )
{
*aActualDist = std::max( 0, *aActualDist - segmHalfWidth );
return false; return false;
}
// testing the right bottom corner of the rectangle // testing the right bottom corner of the rectangle
startPoint.x = m_padToTestPos.x + padHalfsize.x; startPoint.x = m_padToTestPos.x + padHalfsize.x;
@ -1289,8 +1364,12 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
RotatePoint( &startPoint, m_padToTestPos, orient ); RotatePoint( &startPoint, m_padToTestPos, orient );
RotatePoint( &startPoint, m_segmAngle ); RotatePoint( &startPoint, m_segmAngle );
if( !checkMarginToCircle( startPoint, distToLine, m_segmLength ) ) if( !checkMarginToCircle( startPoint, aMinDist + segmHalfWidth, m_segmLength,
aActualDist ) )
{
*aActualDist = std::max( 0, *aActualDist - segmHalfWidth );
return false; return false;
}
break; break;
@ -1306,10 +1385,13 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
RotatePoint( &poly[ii], m_segmAngle ); RotatePoint( &poly[ii], m_segmAngle );
} }
if( !poly2segmentDRC( poly, 4, wxPoint( 0, 0 ), if( !poly2segmentDRC( poly, 4, wxPoint( 0, 0 ), wxPoint( m_segmLength, 0 ),
wxPoint(m_segmLength,0), distToLine ) ) aMinDist + segmHalfWidth, aActualDist ) )
{
*aActualDist = std::max( 0, *aActualDist - segmHalfWidth );
return false; return false;
} }
}
break; break;
case PAD_SHAPE_CUSTOM: case PAD_SHAPE_CUSTOM:
@ -1320,8 +1402,7 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
// relatives to the segment being tested // relatives to the segment being tested
// Note, the pad position relative to the segment origin // Note, the pad position relative to the segment origin
// is m_padToTestPos // is m_padToTestPos
aPad->CustomShapeAsPolygonToBoardPosition( &polyset, aPad->CustomShapeAsPolygonToBoardPosition( &polyset, m_padToTestPos, orient );
m_padToTestPos, orient );
// Rotate all coordinates by m_segmAngle, because the segment orient // Rotate all coordinates by m_segmAngle, because the segment orient
// is m_segmAngle // is m_segmAngle
@ -1329,17 +1410,18 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
// only the lenght and orientation+ of the segment // only the lenght and orientation+ of the segment
// therefore all coordinates of the pad to test must be rotated by // therefore all coordinates of the pad to test must be rotated by
// m_segmAngle (they are already relative to the segment origin) // m_segmAngle (they are already relative to the segment origin)
aPad->CustomShapeAsPolygonToBoardPosition( &polyset, aPad->CustomShapeAsPolygonToBoardPosition( &polyset, wxPoint( 0, 0 ), m_segmAngle );
wxPoint( 0, 0 ), m_segmAngle );
const SHAPE_LINE_CHAIN& refpoly = polyset.COutline( 0 ); const SHAPE_LINE_CHAIN& refpoly = polyset.COutline( 0 );
if( !poly2segmentDRC( (wxPoint*) &refpoly.CPoint( 0 ), if( !poly2segmentDRC( (wxPoint*) &refpoly.CPoint( 0 ), refpoly.PointCount(),
refpoly.PointCount(),
wxPoint( 0, 0 ), wxPoint(m_segmLength,0), wxPoint( 0, 0 ), wxPoint(m_segmLength,0),
distToLine ) ) aMinDist + segmHalfWidth, aActualDist ) )
{
*aActualDist = std::max( 0, *aActualDist - segmHalfWidth );
return false; return false;
} }
}
break; break;
case PAD_SHAPE_CHAMFERED_RECT: case PAD_SHAPE_CHAMFERED_RECT:
@ -1370,12 +1452,14 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
const SHAPE_LINE_CHAIN& refpoly = polyset.COutline( 0 ); const SHAPE_LINE_CHAIN& refpoly = polyset.COutline( 0 );
if( !poly2segmentDRC( (wxPoint*) &refpoly.CPoint( 0 ), if( !poly2segmentDRC( (wxPoint*) &refpoly.CPoint( 0 ), refpoly.PointCount(),
refpoly.PointCount(),
wxPoint( 0, 0 ), wxPoint(m_segmLength,0), wxPoint( 0, 0 ), wxPoint(m_segmLength,0),
distToLine ) ) aMinDist + segmHalfWidth, aActualDist ) )
{
*aActualDist = std::max( 0, *aActualDist - segmHalfWidth );
return false; return false;
} }
}
break; break;
} }
@ -1387,25 +1471,34 @@ bool DRC::checkClearanceSegmToPad( const D_PAD* aPad, int aSegmentWidth, int aMi
* Helper function checkMarginToCircle * Helper function checkMarginToCircle
* Check the distance between a circle (round pad, via or round end of track) * Check the distance between a circle (round pad, via or round end of track)
* and a segment. the segment is expected starting at 0,0, and on the X axis * and a segment. the segment is expected starting at 0,0, and on the X axis
* return true if distance >= aRadius * return true if distance >= aAllowed
*/ */
bool DRC::checkMarginToCircle( wxPoint aCentre, int aRadius, int aLength ) bool DRC::checkMarginToCircle( wxPoint aCentre, int aAllowed, int aLength, int* aActual )
{ {
if( abs( aCentre.y ) >= aRadius ) // trivial case if( abs( aCentre.y ) >= aAllowed ) // trivial case
return true; return true;
// Here, distance between aCentre and X axis is < aRadius // Here, distance between aCentre and X axis is < aAllowed
if( (aCentre.x > -aRadius ) && ( aCentre.x < (aLength + aRadius) ) ) if( ( aCentre.x > -aAllowed ) && ( aCentre.x < ( aLength + aAllowed ) ) )
{ {
if( (aCentre.x >= 0) && (aCentre.x <= aLength) ) if( ( aCentre.x >= 0 ) && ( aCentre.x <= aLength ) )
return false; // aCentre is between the starting point and the ending point of the segm {
// aCentre is between the starting point and the ending point of the segm
*aActual = abs( aCentre.y );
return false;
}
if( aCentre.x > aLength ) // aCentre is after the ending point if( aCentre.x > aLength ) // aCentre is after the ending point
aCentre.x -= aLength; // move aCentre to the starting point of the segment aCentre.x -= aLength; // move aCentre to the starting point of the segment
if( EuclideanNorm( aCentre ) < aRadius ) int distToOrigin = KiROUND( EuclideanNorm( aCentre ) );
// distance between aCentre and the starting point or the ending point is < aRadius
if( distToOrigin < aAllowed )
{
// distance between aCentre and the starting point or the ending point is < aAllowed
*aActual = distToOrigin;
return false; return false;
}
} }
return true; return true;

View File

@ -32,9 +32,12 @@
#include <class_board.h> #include <class_board.h>
wxString DRC_ITEM::GetErrorText() const wxString DRC_ITEM::GetErrorText( int aCode ) const
{ {
switch( m_errorCode ) if( aCode < 0 )
aCode = m_errorCode;
switch( aCode )
{ {
case DRCE_UNCONNECTED_ITEMS: case DRCE_UNCONNECTED_ITEMS:
return wxString( _( "Unconnected items" ) ); return wxString( _( "Unconnected items" ) );
@ -179,7 +182,7 @@ wxString DRC_ITEM::ShowHtml( PCB_BASE_FRAME* aFrame ) const
{ {
BOARD_ITEM* mainItem = nullptr; BOARD_ITEM* mainItem = nullptr;
BOARD_ITEM* auxItem = nullptr; BOARD_ITEM* auxItem = nullptr;
wxString msg = m_errorMessage.IsEmpty() ? GetErrorText() : m_errorMessage; wxString msg = m_errorMessage.IsEmpty() ? GetErrorText( m_errorCode ) : m_errorMessage;
wxString mainText; wxString mainText;
wxString auxText; wxString auxText;

View File

@ -45,7 +45,7 @@ public:
* Function GetErrorText * Function GetErrorText
* returns the string form of a drc error code. * returns the string form of a drc error code.
*/ */
wxString GetErrorText() const override; wxString GetErrorText( int aErrorCode = -1 ) const override;
/** /**
* Function ShowHtml * Function ShowHtml