Point editor hookup for non-custom-shaped pads.
Fixes https://gitlab.com/kicad/code/kicad/issues/2153
This commit is contained in:
parent
a229e51962
commit
516de9c85e
|
@ -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:
|
||||||
{
|
{
|
||||||
|
|
|
@ -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;
|
||||||
|
|
|
@ -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:
|
||||||
{
|
{
|
||||||
|
|
Loading…
Reference in New Issue