Point editor hookup for non-custom-shaped pads.

Fixes https://gitlab.com/kicad/code/kicad/issues/2153
This commit is contained in:
Jeff Young 2020-06-28 23:34:10 +01:00
parent a229e51962
commit 516de9c85e
3 changed files with 252 additions and 70 deletions

View File

@ -39,6 +39,7 @@
#include <geometry/shape_circle.h> #include <geometry/shape_circle.h>
#include <geometry/shape_segment.h> #include <geometry/shape_segment.h>
#include <geometry/shape_simple.h> #include <geometry/shape_simple.h>
#include <geometry/shape_rect.h>
#include <pcbnew.h> #include <pcbnew.h>
#include <view/view.h> #include <view/view.h>
#include <class_board.h> #include <class_board.h>
@ -252,6 +253,15 @@ void D_PAD::BuildEffectiveShapes() const
break; break;
case PAD_SHAPE_RECT: case PAD_SHAPE_RECT:
if( (int) m_Orient % 900 == 0 )
{
add( new SHAPE_RECT( shapePos - m_Size / 2, m_Size.x, m_Size.y ) );
break;
}
// Not at a cartesian angle; fall through to general case
KI_FALLTHROUGH;
case PAD_SHAPE_TRAPEZOID: case PAD_SHAPE_TRAPEZOID:
case PAD_SHAPE_ROUNDRECT: case PAD_SHAPE_ROUNDRECT:
{ {

View File

@ -581,7 +581,6 @@ PCB_LAYER_ID PAD_TOOL::explodePad( D_PAD* aPad )
ds->SetLocalCoord(); ds->SetLocalCoord();
ds->Move( aPad->GetPosition() ); ds->Move( aPad->GetPosition() );
ds->Rotate( aPad->GetPosition(), aPad->GetOrientation() ); ds->Rotate( aPad->GetPosition(), aPad->GetOrientation() );
ds->SetLayer( layer ); ds->SetLayer( layer );
commit.Add( ds ); commit.Add( ds );
@ -656,8 +655,8 @@ void PAD_TOOL::recombinePad( D_PAD* aPad )
aPad->TransformShapeWithClearanceToPolygon( existingOutline, 0, maxError ); aPad->TransformShapeWithClearanceToPolygon( existingOutline, 0, maxError );
aPad->SetAnchorPadShape( PAD_SHAPE_CIRCLE ); aPad->SetAnchorPadShape( PAD_SHAPE_CIRCLE );
int r = aPad->GetDrillSize().x + Millimeter2iu( 0.2 ); wxSize minAnnulus( Millimeter2iu( 0.2 ), Millimeter2iu( 0.2 ) );
aPad->SetSize( wxSize( r, r ) ); aPad->SetSize( aPad->GetDrillSize() + minAnnulus );
aPad->SetOffset( wxPoint( 0, 0 ) ); aPad->SetOffset( wxPoint( 0, 0 ) );
DRAWSEGMENT* shape = new DRAWSEGMENT; DRAWSEGMENT* shape = new DRAWSEGMENT;

View File

@ -137,93 +137,130 @@ public:
// Generate list of edit points basing on the item type // Generate list of edit points basing on the item type
switch( aItem->Type() ) switch( aItem->Type() )
{ {
case PCB_LINE_T: case PCB_LINE_T:
case PCB_MODULE_EDGE_T: case PCB_MODULE_EDGE_T:
{
const DRAWSEGMENT* segment = static_cast<const DRAWSEGMENT*>( aItem );
switch( segment->GetShape() )
{ {
const DRAWSEGMENT* segment = static_cast<const DRAWSEGMENT*>( aItem ); case S_SEGMENT:
points->AddPoint( segment->GetStart() );
points->AddPoint( segment->GetEnd() );
break;
switch( segment->GetShape() ) case S_RECT:
{ points->AddPoint( segment->GetStart() );
case S_SEGMENT: points->AddPoint( wxPoint( segment->GetEnd().x, segment->GetStart().y ) );
points->AddPoint( segment->GetStart() ); points->AddPoint( segment->GetEnd() );
points->AddPoint( segment->GetEnd() ); points->AddPoint( wxPoint( segment->GetStart().x, segment->GetEnd().y ) );
break; break;
case S_RECT: case S_ARC:
points->AddPoint( segment->GetStart() ); points->AddPoint( segment->GetCenter() );
points->AddPoint( wxPoint( segment->GetEnd().x, segment->GetStart().y ) ); points->AddPoint( segment->GetArcStart() );
points->AddPoint( segment->GetEnd() ); points->AddPoint( segment->GetArcMid() );
points->AddPoint( wxPoint( segment->GetStart().x, segment->GetEnd().y ) ); points->AddPoint( segment->GetArcEnd() );
break;
case S_ARC: // Set constraints
points->AddPoint( segment->GetCenter() ); // Arc end has to stay at the same radius as the start
points->AddPoint( segment->GetArcStart() ); points->Point( ARC_END ).SetConstraint( new EC_CIRCLE( points->Point( ARC_END ),
points->AddPoint( segment->GetArcMid() ); points->Point( ARC_CENTER ),
points->AddPoint( segment->GetArcEnd() ); points->Point( ARC_START ) ) );
// Set constraints points->Point( ARC_MID ).SetConstraint( new EC_LINE( points->Point( ARC_MID ),
// Arc end has to stay at the same radius as the start points->Point( ARC_CENTER ) ) );
points->Point( ARC_END ).SetConstraint( new EC_CIRCLE( points->Point( ARC_END ), break;
points->Point( ARC_CENTER ),
points->Point( ARC_START ) ) );
points->Point( ARC_MID ).SetConstraint( new EC_LINE( points->Point( ARC_MID ), case S_CIRCLE:
points->Point( ARC_CENTER ) ) ); points->AddPoint( segment->GetCenter() );
break; points->AddPoint( segment->GetEnd() );
break;
case S_CIRCLE: case S_POLYGON:
points->AddPoint( segment->GetCenter() ); buildForPolyOutline( points, &segment->GetPolyShape(), aGal );
points->AddPoint( segment->GetEnd() ); break;
break;
case S_POLYGON: case S_CURVE:
buildForPolyOutline( points, &segment->GetPolyShape(), aGal ); points->AddPoint( segment->GetStart() );
break; points->AddPoint( segment->GetBezControl1() );
points->AddPoint( segment->GetBezControl2() );
case S_CURVE: points->AddPoint( segment->GetEnd() );
points->AddPoint( segment->GetStart() ); break;
points->AddPoint( segment->GetBezControl1() );
points->AddPoint( segment->GetBezControl2() );
points->AddPoint( segment->GetEnd() );
break;
default: // suppress warnings
break;
}
default: // suppress warnings
break; break;
} }
case PCB_MODULE_ZONE_AREA_T: break;
case PCB_ZONE_AREA_T: }
case PCB_PAD_T:
{
const D_PAD* pad = static_cast<const D_PAD*>( aItem );
wxPoint shapePos = pad->ShapePos();
wxPoint halfSize( pad->GetSize().x / 2, pad->GetSize().y / 2 );
switch( pad->GetShape() )
{ {
auto zone = static_cast<const ZONE_CONTAINER*>( aItem ); case PAD_SHAPE_CIRCLE:
buildForPolyOutline( points, zone->Outline(), aGal ); points->AddPoint( shapePos );
points->AddPoint( wxPoint( shapePos.x + halfSize.x, shapePos.y ) );
break;
case PAD_SHAPE_OVAL:
case PAD_SHAPE_TRAPEZOID:
case PAD_SHAPE_RECT:
case PAD_SHAPE_ROUNDRECT:
case PAD_SHAPE_CHAMFERED_RECT:
{
if( (int) pad->GetOrientation() % 900 != 0 )
break;
if( pad->GetOrientation() == 900 || pad->GetOrientation() == 2700 )
std::swap( halfSize.x, halfSize.y );
points->AddPoint( shapePos - halfSize );
points->AddPoint( wxPoint( shapePos.x + halfSize.x, shapePos.y - halfSize.y ) );
points->AddPoint( shapePos + halfSize );
points->AddPoint( wxPoint( shapePos.x - halfSize.x, shapePos.y + halfSize.y ) );
}
break;
default: // suppress warnings
break; break;
} }
}
break;
case PCB_DIMENSION_T: case PCB_MODULE_ZONE_AREA_T:
{ case PCB_ZONE_AREA_T:
const DIMENSION* dimension = static_cast<const DIMENSION*>( aItem ); {
auto zone = static_cast<const ZONE_CONTAINER*>( aItem );
buildForPolyOutline( points, zone->Outline(), aGal );
}
break;
points->AddPoint( dimension->m_crossBarO ); case PCB_DIMENSION_T:
points->AddPoint( dimension->m_crossBarF ); {
points->AddPoint( dimension->m_featureLineGO ); const DIMENSION* dimension = static_cast<const DIMENSION*>( aItem );
points->AddPoint( dimension->m_featureLineDO );
// Dimension height setting - edit points should move only along the feature lines points->AddPoint( dimension->m_crossBarO );
points->Point( DIM_CROSSBARO ).SetConstraint( new EC_LINE( points->Point( DIM_CROSSBARO ), points->AddPoint( dimension->m_crossBarF );
points->Point( DIM_FEATUREGO ) ) ); points->AddPoint( dimension->m_featureLineGO );
points->Point( DIM_CROSSBARF ).SetConstraint( new EC_LINE( points->Point( DIM_CROSSBARF ), points->AddPoint( dimension->m_featureLineDO );
points->Point( DIM_FEATUREDO ) ) );
break; // Dimension height setting - edit points should move only along the feature lines
} points->Point( DIM_CROSSBARO ).SetConstraint( new EC_LINE( points->Point( DIM_CROSSBARO ),
points->Point( DIM_FEATUREGO ) ) );
points->Point( DIM_CROSSBARF ).SetConstraint( new EC_LINE( points->Point( DIM_CROSSBARF ),
points->Point( DIM_FEATUREDO ) ) );
}
break;
default: default:
points.reset(); points.reset();
break; break;
} }
return points; return points;
@ -446,6 +483,7 @@ void POINT_EDITOR::updateItem() const
case PCB_MODULE_EDGE_T: case PCB_MODULE_EDGE_T:
{ {
DRAWSEGMENT* segment = static_cast<DRAWSEGMENT*>( item ); DRAWSEGMENT* segment = static_cast<DRAWSEGMENT*>( item );
switch( segment->GetShape() ) switch( segment->GetShape() )
{ {
case S_SEGMENT: case S_SEGMENT:
@ -582,6 +620,74 @@ void POINT_EDITOR::updateItem() const
break; break;
} }
case PCB_PAD_T:
{
D_PAD* pad = static_cast<D_PAD*>( item );
switch( pad->GetShape() )
{
case PAD_SHAPE_CIRCLE:
{
wxPoint center = (wxPoint) m_editPoints->Point( CIRC_CENTER ).GetPosition();
wxPoint end = (wxPoint) m_editPoints->Point( CIRC_END ).GetPosition();
if( isModified( m_editPoints->Point( CIRC_CENTER ) ) )
{
wxPoint moveVector = center - pad->ShapePos();
pad->SetOffset( pad->GetOffset() + moveVector );
}
else
{
int diameter = (int) EuclideanNorm( end - center ) * 2;
pad->SetSize( wxSize( diameter, diameter ) );
}
}
break;
case PAD_SHAPE_OVAL:
case PAD_SHAPE_TRAPEZOID:
case PAD_SHAPE_RECT:
case PAD_SHAPE_ROUNDRECT:
case PAD_SHAPE_CHAMFERED_RECT:
{
wxPoint center = pad->GetPosition();
int dist[4];
if( isModified( m_editPoints->Point( RECT_TOP_LEFT ) )
|| isModified( m_editPoints->Point( RECT_BOT_RIGHT ) ) )
{
dist[0] = center.x - m_editPoints->Point( RECT_TOP_LEFT ).GetPosition().x;
dist[1] = center.y - m_editPoints->Point( RECT_TOP_LEFT ).GetPosition().y;
dist[2] = m_editPoints->Point( RECT_BOT_RIGHT ).GetPosition().x - center.x;
dist[3] = m_editPoints->Point( RECT_BOT_RIGHT ).GetPosition().y - center.y;
}
else
{
dist[0] = center.x - m_editPoints->Point( RECT_BOT_LEFT ).GetPosition().x;
dist[1] = center.y - m_editPoints->Point( RECT_TOP_RIGHT ).GetPosition().y;
dist[2] = m_editPoints->Point( RECT_TOP_RIGHT ).GetPosition().x - center.x;
dist[3] = m_editPoints->Point( RECT_BOT_LEFT ).GetPosition().y - center.y;
}
wxSize padSize( dist[0] + dist[2], dist[1] + dist[3] );
wxPoint padOffset( padSize.x / 2 - dist[2], padSize.y / 2 - dist[3] );
if( pad->GetOrientation() == 900 || pad->GetOrientation() == 2700 )
std::swap( padSize.x, padSize.y );
RotatePoint( &padOffset, -pad->GetOrientation() );
pad->SetSize( padSize );
pad->SetOffset( -padOffset );
}
break;
default: // suppress warnings
break;
}
}
break;
case PCB_MODULE_ZONE_AREA_T: case PCB_MODULE_ZONE_AREA_T:
case PCB_ZONE_AREA_T: case PCB_ZONE_AREA_T:
{ {
@ -780,6 +886,73 @@ void POINT_EDITOR::updatePoints()
break; break;
} }
case PCB_PAD_T:
{
const D_PAD* pad = static_cast<const D_PAD*>( item );
wxPoint shapePos = pad->ShapePos();
wxPoint halfSize( pad->GetSize().x / 2, pad->GetSize().y / 2 );
switch( pad->GetShape() )
{
case PAD_SHAPE_CIRCLE:
{
// Careful; pad shape is mutable...
if( m_editPoints->PointsSize() != 2 )
{
getView()->Remove( m_editPoints.get() );
m_editedPoint = nullptr;
m_editPoints = EDIT_POINTS_FACTORY::Make( item, getView()->GetGAL() );
getView()->Add( m_editPoints.get() );
}
else
{
VECTOR2I vec = m_editPoints->Point( CIRC_END ).GetPosition()
- m_editPoints->Point( CIRC_CENTER ).GetPosition();
vec.Resize( halfSize.x );
m_editPoints->Point( CIRC_CENTER ).SetPosition( shapePos );
m_editPoints->Point( CIRC_END ).SetPosition( vec + shapePos );
}
}
break;
case PAD_SHAPE_OVAL:
case PAD_SHAPE_TRAPEZOID:
case PAD_SHAPE_RECT:
case PAD_SHAPE_ROUNDRECT:
case PAD_SHAPE_CHAMFERED_RECT:
{
// Careful; pad shape and orientation are mutable...
int target = ( (int) pad->GetOrientation() % 900 == 0 ) ? 4 : 0;
if( m_editPoints->PointsSize() != target )
{
getView()->Remove( m_editPoints.get() );
m_editedPoint = nullptr;
m_editPoints = EDIT_POINTS_FACTORY::Make( item, getView()->GetGAL() );
getView()->Add( m_editPoints.get() );
}
else if( target == 4 )
{
if( pad->GetOrientation() == 900 || pad->GetOrientation() == 2700 )
std::swap( halfSize.x, halfSize.y );
m_editPoints->Point( RECT_TOP_LEFT ).SetPosition( shapePos - halfSize );
m_editPoints->Point( RECT_TOP_RIGHT ).SetPosition( wxPoint( shapePos.x + halfSize.x,
shapePos.y - halfSize.y ) );
m_editPoints->Point( RECT_BOT_RIGHT ).SetPosition( shapePos + halfSize );
m_editPoints->Point( RECT_BOT_LEFT ).SetPosition( wxPoint( shapePos.x - halfSize.x,
shapePos.y + halfSize.y ) );
}
}
break;
default: // suppress warnings
break;
}
}
break;
case PCB_MODULE_ZONE_AREA_T: case PCB_MODULE_ZONE_AREA_T:
case PCB_ZONE_AREA_T: case PCB_ZONE_AREA_T:
{ {