Don't apply "normal" locking strategy to alignment options.

It's much more useful to use the locked items as targets.

Also fixes some issues with commits getting the wrong items when
pads are promoted to footprints.

Fixes https://gitlab.com/kicad/code/kicad/issues/6859
This commit is contained in:
Jeff Young 2020-12-27 15:00:47 +00:00
parent e3d95cc2c8
commit 10e51bfea7
2 changed files with 102 additions and 114 deletions

View File

@ -84,11 +84,11 @@ bool ALIGN_DISTRIBUTE_TOOL::Init()
template <class T> template <class T>
ALIGNMENT_RECTS GetBoundingBoxes( const T &sel ) ALIGNMENT_RECTS GetBoundingBoxes( const T& aItems )
{ {
ALIGNMENT_RECTS rects; ALIGNMENT_RECTS rects;
for( EDA_ITEM* item : sel ) for( EDA_ITEM* item : aItems )
{ {
BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( item ); BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( item );
@ -113,36 +113,33 @@ int ALIGN_DISTRIBUTE_TOOL::selectTarget( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECT
{ {
wxPoint curPos = (wxPoint) getViewControls()->GetCursorPosition(); wxPoint curPos = (wxPoint) getViewControls()->GetCursorPosition();
// after sorting, the fist item acts as the target for all others unless we have a locked // Prefer locked items to unlocked items.
// item, in which case we use that for the target // Secondly, prefer items under the cursor to other items.
int target = !aLocked.size() ? aGetValue( aItems.front() ) : aGetValue( aLocked.front() );
// We take the first target that overlaps our cursor. if( aLocked.size() >= 1 )
// This is deterministic because we assume sorted arrays {
for( const ALIGNMENT_RECT& item : aLocked ) for( const ALIGNMENT_RECT& item : aLocked )
{ {
if( item.second.Contains( curPos ) ) if( item.second.Contains( curPos ) )
return aGetValue( item ); return aGetValue( item );
} }
// If there are locked items, prefer aligning to them over aligning to the cursor as they do return aGetValue( aLocked.front() );
// not move }
if( aLocked.empty() )
{
for( const ALIGNMENT_RECT& item : aItems ) for( const ALIGNMENT_RECT& item : aItems )
{ {
if( item.second.Contains( curPos ) ) if( item.second.Contains( curPos ) )
return aGetValue( item ); return aGetValue( item );
} }
}
return target; return aGetValue( aItems.front() );
} }
template< typename T > template< typename T >
size_t ALIGN_DISTRIBUTE_TOOL::GetSelections( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECTS& aLocked, size_t ALIGN_DISTRIBUTE_TOOL::GetSelections( ALIGNMENT_RECTS& aItemsToAlign,
T aCompare ) ALIGNMENT_RECTS& aLockedItems, T aCompare )
{ {
PCB_SELECTION& selection = m_selectionTool->RequestSelection( PCB_SELECTION& selection = m_selectionTool->RequestSelection(
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector, PCB_SELECTION_TOOL* sTool ) []( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector, PCB_SELECTION_TOOL* sTool )
@ -158,6 +155,7 @@ size_t ALIGN_DISTRIBUTE_TOOL::GetSelections( ALIGNMENT_RECTS& aItems, ALIGNMENT_
} ); } );
std::vector<BOARD_ITEM*> lockedItems; std::vector<BOARD_ITEM*> lockedItems;
std::vector<BOARD_ITEM*> itemsToAlign;
for( EDA_ITEM* item : selection ) for( EDA_ITEM* item : selection )
{ {
@ -165,28 +163,16 @@ size_t ALIGN_DISTRIBUTE_TOOL::GetSelections( ALIGNMENT_RECTS& aItems, ALIGNMENT_
if( boardItem->IsLocked() ) if( boardItem->IsLocked() )
lockedItems.push_back( boardItem ); lockedItems.push_back( boardItem );
else
itemsToAlign.push_back( boardItem );
} }
selection = m_selectionTool->RequestSelection( aItemsToAlign = GetBoundingBoxes( itemsToAlign );
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector, PCB_SELECTION_TOOL* sTool ) aLockedItems = GetBoundingBoxes( lockedItems );
{ std::sort( aItemsToAlign.begin(), aItemsToAlign.end(), aCompare );
// Iterate from the back so we don't have to worry about removals. std::sort( aLockedItems.begin(), aLockedItems.end(), aCompare );
for( int i = aCollector.GetCount() - 1; i >= 0; --i )
{
BOARD_ITEM* item = aCollector[i];
if( item->Type() == PCB_MARKER_T ) return aItemsToAlign.size();
aCollector.Remove( item );
}
},
true /* prompt user regarding locked items */ );
aItems = GetBoundingBoxes( selection );
aLocked = GetBoundingBoxes( lockedItems );
std::sort( aItems.begin(), aItems.end(), aCompare );
std::sort( aLocked.begin(), aLocked.end(), aCompare );
return aItems.size();
} }
@ -205,7 +191,7 @@ int ALIGN_DISTRIBUTE_TOOL::AlignTop( const TOOL_EVENT& aEvent )
} }
BOARD_COMMIT commit( m_frame ); BOARD_COMMIT commit( m_frame );
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
int targetTop = selectTarget( itemsToAlign, locked_items, int targetTop = selectTarget( itemsToAlign, locked_items,
[]( const ALIGNMENT_RECT& aVal ) []( const ALIGNMENT_RECT& aVal )
{ {
@ -222,6 +208,7 @@ int ALIGN_DISTRIBUTE_TOOL::AlignTop( const TOOL_EVENT& aEvent )
if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) ) if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) )
item = item->GetParent(); item = item->GetParent();
commit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( 0, difference ) ); item->Move( wxPoint( 0, difference ) );
} }
@ -246,8 +233,8 @@ int ALIGN_DISTRIBUTE_TOOL::AlignBottom( const TOOL_EVENT& aEvent )
} }
BOARD_COMMIT commit( m_frame ); BOARD_COMMIT commit( m_frame );
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
auto targetBottom = selectTarget( itemsToAlign, locked_items, int targetBottom = selectTarget( itemsToAlign, locked_items,
[]( const ALIGNMENT_RECT& aVal ) []( const ALIGNMENT_RECT& aVal )
{ {
return aVal.second.GetBottom(); return aVal.second.GetBottom();
@ -263,6 +250,7 @@ int ALIGN_DISTRIBUTE_TOOL::AlignBottom( const TOOL_EVENT& aEvent )
if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) ) if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) )
item = item->GetParent(); item = item->GetParent();
commit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( 0, difference ) ); item->Move( wxPoint( 0, difference ) );
} }
@ -302,8 +290,8 @@ int ALIGN_DISTRIBUTE_TOOL::doAlignLeft()
} }
BOARD_COMMIT commit( m_frame ); BOARD_COMMIT commit( m_frame );
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
auto targetLeft = selectTarget( itemsToAlign, locked_items, int targetLeft = selectTarget( itemsToAlign, locked_items,
[]( const ALIGNMENT_RECT& aVal ) []( const ALIGNMENT_RECT& aVal )
{ {
return aVal.second.GetLeft(); return aVal.second.GetLeft();
@ -319,6 +307,7 @@ int ALIGN_DISTRIBUTE_TOOL::doAlignLeft()
if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) ) if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) )
item = item->GetParent(); item = item->GetParent();
commit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( difference, 0 ) ); item->Move( wxPoint( difference, 0 ) );
} }
@ -358,8 +347,8 @@ int ALIGN_DISTRIBUTE_TOOL::doAlignRight()
} }
BOARD_COMMIT commit( m_frame ); BOARD_COMMIT commit( m_frame );
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
auto targetRight = selectTarget( itemsToAlign, locked_items, int targetRight = selectTarget( itemsToAlign, locked_items,
[]( const ALIGNMENT_RECT& aVal ) []( const ALIGNMENT_RECT& aVal )
{ {
return aVal.second.GetRight(); return aVal.second.GetRight();
@ -375,6 +364,7 @@ int ALIGN_DISTRIBUTE_TOOL::doAlignRight()
if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) ) if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) )
item = item->GetParent(); item = item->GetParent();
commit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( difference, 0 ) ); item->Move( wxPoint( difference, 0 ) );
} }
@ -399,8 +389,8 @@ int ALIGN_DISTRIBUTE_TOOL::AlignCenterX( const TOOL_EVENT& aEvent )
} }
BOARD_COMMIT commit( m_frame ); BOARD_COMMIT commit( m_frame );
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
auto targetX = selectTarget( itemsToAlign, locked_items, int targetX = selectTarget( itemsToAlign, locked_items,
[]( const ALIGNMENT_RECT& aVal ) []( const ALIGNMENT_RECT& aVal )
{ {
return aVal.second.GetCenter().x; return aVal.second.GetCenter().x;
@ -416,6 +406,7 @@ int ALIGN_DISTRIBUTE_TOOL::AlignCenterX( const TOOL_EVENT& aEvent )
if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) ) if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) )
item = item->GetParent(); item = item->GetParent();
commit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( difference, 0 ) ); item->Move( wxPoint( difference, 0 ) );
} }
@ -440,8 +431,8 @@ int ALIGN_DISTRIBUTE_TOOL::AlignCenterY( const TOOL_EVENT& aEvent )
} }
BOARD_COMMIT commit( m_frame ); BOARD_COMMIT commit( m_frame );
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
auto targetY = selectTarget( itemsToAlign, locked_items, int targetY = selectTarget( itemsToAlign, locked_items,
[]( const ALIGNMENT_RECT& aVal ) []( const ALIGNMENT_RECT& aVal )
{ {
return aVal.second.GetCenter().y; return aVal.second.GetCenter().y;
@ -457,6 +448,7 @@ int ALIGN_DISTRIBUTE_TOOL::AlignCenterY( const TOOL_EVENT& aEvent )
if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) ) if( item->Type() == PCB_PAD_T && m_frame->IsType( FRAME_PCB_EDITOR ) )
item = item->GetParent(); item = item->GetParent();
commit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( 0, difference ) ); item->Move( wxPoint( 0, difference ) );
} }
@ -486,9 +478,7 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeHorizontally( const TOOL_EVENT& aEvent )
return 0; return 0;
BOARD_COMMIT commit( m_frame ); BOARD_COMMIT commit( m_frame );
commit.StageItems( selection, CHT_MODIFY ); ALIGNMENT_RECTS itemsToDistribute = GetBoundingBoxes( selection );
auto itemsToDistribute = GetBoundingBoxes( selection );
// find the last item by reverse sorting // find the last item by reverse sorting
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
@ -496,9 +486,9 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeHorizontally( const TOOL_EVENT& aEvent )
{ {
return ( left.second.GetRight() > right.second.GetRight() ); return ( left.second.GetRight() > right.second.GetRight() );
} ); } );
const auto lastItem = itemsToDistribute.begin()->first;
const auto maxRight = itemsToDistribute.begin()->second.GetRight(); BOARD_ITEM* lastItem = itemsToDistribute.begin()->first;
const int maxRight = itemsToDistribute.begin()->second.GetRight();
// sort to get starting order // sort to get starting order
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
@ -506,24 +496,23 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeHorizontally( const TOOL_EVENT& aEvent )
{ {
return ( left.second.GetX() < right.second.GetX() ); return ( left.second.GetX() < right.second.GetX() );
} ); } );
const auto minX = itemsToDistribute.begin()->second.GetX();
auto totalGap = maxRight - minX; const int minX = itemsToDistribute.begin()->second.GetX();
int totalGap = maxRight - minX;
int totalWidth = 0; int totalWidth = 0;
for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute ) for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute )
{
totalWidth += i.second.GetWidth(); totalWidth += i.second.GetWidth();
}
if( totalGap < totalWidth ) if( totalGap < totalWidth )
{ {
// the width of the items exceeds the gap (overlapping items) -> use center point spacing // the width of the items exceeds the gap (overlapping items) -> use center point spacing
doDistributeCentersHorizontally( itemsToDistribute ); doDistributeCentersHorizontally( itemsToDistribute, commit );
} }
else else
{ {
totalGap -= totalWidth; totalGap -= totalWidth;
doDistributeGapsHorizontally( itemsToDistribute, lastItem, totalGap ); doDistributeGapsHorizontally( itemsToDistribute, commit, lastItem, totalGap );
} }
commit.Push( _( "Distribute horizontally" ) ); commit.Push( _( "Distribute horizontally" ) );
@ -533,11 +522,12 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeHorizontally( const TOOL_EVENT& aEvent )
void ALIGN_DISTRIBUTE_TOOL::doDistributeGapsHorizontally( ALIGNMENT_RECTS& itemsToDistribute, void ALIGN_DISTRIBUTE_TOOL::doDistributeGapsHorizontally( ALIGNMENT_RECTS& itemsToDistribute,
BOARD_COMMIT& aCommit,
const BOARD_ITEM* lastItem, const BOARD_ITEM* lastItem,
int totalGap ) const int totalGap ) const
{ {
const auto itemGap = totalGap / ( itemsToDistribute.size() - 1 ); const int itemGap = totalGap / ( itemsToDistribute.size() - 1 );
auto targetX = itemsToDistribute.begin()->second.GetX(); int targetX = itemsToDistribute.begin()->second.GetX();
for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute ) for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute )
{ {
@ -552,23 +542,26 @@ void ALIGN_DISTRIBUTE_TOOL::doDistributeGapsHorizontally( ALIGNMENT_RECTS& items
item = item->GetParent(); item = item->GetParent();
int difference = targetX - i.second.GetX(); int difference = targetX - i.second.GetX();
aCommit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( difference, 0 ) ); item->Move( wxPoint( difference, 0 ) );
targetX += ( i.second.GetWidth() + itemGap ); targetX += ( i.second.GetWidth() + itemGap );
} }
} }
void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersHorizontally( ALIGNMENT_RECTS &itemsToDistribute ) const void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersHorizontally( ALIGNMENT_RECTS &itemsToDistribute,
BOARD_COMMIT& aCommit ) const
{ {
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
[] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right) [] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
{ {
return ( left.second.GetCenter().x < right.second.GetCenter().x ); return ( left.second.GetCenter().x < right.second.GetCenter().x );
} ); } );
const auto totalGap = ( itemsToDistribute.end()-1 )->second.GetCenter().x
const int totalGap = ( itemsToDistribute.end()-1 )->second.GetCenter().x
- itemsToDistribute.begin()->second.GetCenter().x; - itemsToDistribute.begin()->second.GetCenter().x;
const auto itemGap = totalGap / ( itemsToDistribute.size() - 1 ); const int itemGap = totalGap / ( itemsToDistribute.size() - 1 );
auto targetX = itemsToDistribute.begin()->second.GetCenter().x; int targetX = itemsToDistribute.begin()->second.GetCenter().x;
for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute ) for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute )
{ {
@ -579,6 +572,7 @@ void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersHorizontally( ALIGNMENT_RECTS &it
item = item->GetParent(); item = item->GetParent();
int difference = targetX - i.second.GetCenter().x; int difference = targetX - i.second.GetCenter().x;
aCommit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( difference, 0 ) ); item->Move( wxPoint( difference, 0 ) );
targetX += ( itemGap ); targetX += ( itemGap );
} }
@ -605,9 +599,7 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeVertically( const TOOL_EVENT& aEvent )
return 0; return 0;
BOARD_COMMIT commit( m_frame ); BOARD_COMMIT commit( m_frame );
commit.StageItems( selection, CHT_MODIFY ); ALIGNMENT_RECTS itemsToDistribute = GetBoundingBoxes( selection );
auto itemsToDistribute = GetBoundingBoxes( selection );
// find the last item by reverse sorting // find the last item by reverse sorting
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
@ -615,8 +607,9 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeVertically( const TOOL_EVENT& aEvent )
{ {
return ( left.second.GetBottom() > right.second.GetBottom() ); return ( left.second.GetBottom() > right.second.GetBottom() );
} ); } );
const auto maxBottom = itemsToDistribute.begin()->second.GetBottom();
const auto lastItem = itemsToDistribute.begin()->first; BOARD_ITEM* lastItem = itemsToDistribute.begin()->first;
const int maxBottom = itemsToDistribute.begin()->second.GetBottom();
// sort to get starting order // sort to get starting order
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
@ -624,25 +617,23 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeVertically( const TOOL_EVENT& aEvent )
{ {
return ( left.second.GetCenter().y < right.second.GetCenter().y ); return ( left.second.GetCenter().y < right.second.GetCenter().y );
} ); } );
auto minY = itemsToDistribute.begin()->second.GetY();
auto totalGap = maxBottom - minY; int minY = itemsToDistribute.begin()->second.GetY();
int totalGap = maxBottom - minY;
int totalHeight = 0; int totalHeight = 0;
for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute ) for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute )
{
totalHeight += i.second.GetHeight(); totalHeight += i.second.GetHeight();
}
if( totalGap < totalHeight ) if( totalGap < totalHeight )
{ {
// the width of the items exceeds the gap (overlapping items) -> use center point spacing // the width of the items exceeds the gap (overlapping items) -> use center point spacing
doDistributeCentersVertically( itemsToDistribute ); doDistributeCentersVertically( itemsToDistribute, commit );
} }
else else
{ {
totalGap -= totalHeight; totalGap -= totalHeight;
doDistributeGapsVertically( itemsToDistribute, lastItem, totalGap ); doDistributeGapsVertically( itemsToDistribute, commit, lastItem, totalGap );
} }
commit.Push( _( "Distribute vertically" ) ); commit.Push( _( "Distribute vertically" ) );
@ -652,10 +643,12 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeVertically( const TOOL_EVENT& aEvent )
void ALIGN_DISTRIBUTE_TOOL::doDistributeGapsVertically( ALIGNMENT_RECTS& itemsToDistribute, void ALIGN_DISTRIBUTE_TOOL::doDistributeGapsVertically( ALIGNMENT_RECTS& itemsToDistribute,
const BOARD_ITEM* lastItem, int totalGap ) const BOARD_COMMIT& aCommit,
const BOARD_ITEM* lastItem,
int totalGap ) const
{ {
const auto itemGap = totalGap / ( itemsToDistribute.size() - 1 ); const int itemGap = totalGap / ( itemsToDistribute.size() - 1 );
auto targetY = itemsToDistribute.begin()->second.GetY(); int targetY = itemsToDistribute.begin()->second.GetY();
for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute ) for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute )
{ {
@ -670,23 +663,26 @@ void ALIGN_DISTRIBUTE_TOOL::doDistributeGapsVertically( ALIGNMENT_RECTS& itemsTo
item = item->GetParent(); item = item->GetParent();
int difference = targetY - i.second.GetY(); int difference = targetY - i.second.GetY();
i.first->Move( wxPoint( 0, difference ) ); aCommit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( 0, difference ) );
targetY += ( i.second.GetHeight() + itemGap ); targetY += ( i.second.GetHeight() + itemGap );
} }
} }
void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersVertically( ALIGNMENT_RECTS& itemsToDistribute ) const void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersVertically( ALIGNMENT_RECTS& itemsToDistribute,
BOARD_COMMIT& aCommit ) const
{ {
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
[] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right) [] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
{ {
return ( left.second.GetCenter().y < right.second.GetCenter().y ); return ( left.second.GetCenter().y < right.second.GetCenter().y );
} ); } );
const auto totalGap = ( itemsToDistribute.end()-1 )->second.GetCenter().y
const int totalGap = ( itemsToDistribute.end()-1 )->second.GetCenter().y
- itemsToDistribute.begin()->second.GetCenter().y; - itemsToDistribute.begin()->second.GetCenter().y;
const auto itemGap = totalGap / ( itemsToDistribute.size() - 1 ); const int itemGap = totalGap / ( itemsToDistribute.size() - 1 );
auto targetY = itemsToDistribute.begin()->second.GetCenter().y; int targetY = itemsToDistribute.begin()->second.GetCenter().y;
for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute ) for( std::pair<BOARD_ITEM*, EDA_RECT>& i : itemsToDistribute )
{ {
@ -697,6 +693,7 @@ void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersVertically( ALIGNMENT_RECTS& item
item = item->GetParent(); item = item->GetParent();
int difference = targetY - i.second.GetCenter().y; int difference = targetY - i.second.GetCenter().y;
aCommit.Stage( item, CHT_MODIFY );
item->Move( wxPoint( 0, difference ) ); item->Move( wxPoint( 0, difference ) );
targetY += ( itemGap ); targetY += ( itemGap );
} }

View File

@ -104,10 +104,11 @@ private:
/** /**
* Function GetSelections() * Function GetSelections()
* Populates two vectors with the sorted selection and sorted locked items * Populates two vectors with the sorted selection and sorted locked items
* Returns the size of aItems() * Returns the size of aItemsToAlign()
*/ */
template< typename T > template< typename T >
size_t GetSelections( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECTS& aLocked, T aCompare ); size_t GetSelections( ALIGNMENT_RECTS& aItemsToAlign, ALIGNMENT_RECTS& aLockedItems,
T aCompare );
template< typename T > template< typename T >
int selectTarget( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECTS& aLocked, T aGetValue ); int selectTarget( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECTS& aLocked, T aGetValue );
@ -129,14 +130,13 @@ private:
int doAlignRight(); int doAlignRight();
/** /**
* Check a selection to ensure locks are valid for alignment. * Distributes selected items using an even spacing between the centers of their bounding boxes
* *
* This is slightly different from the standard lock checking in that we ignore the lock * NOTE: Using the centers of bounding box of items can give unsatisfactory visual results since
* of the first element in the selection as this is meant to be our anchor. * items of differing widths will be placed with different gaps. Is only used if items overlap
* We also check the lock of a pad's parent as we will not move pads independently of
* the parent footprint
*/ */
int checkLockedStatus( const PCB_SELECTION &selection ) const; void doDistributeCentersHorizontally( ALIGNMENT_RECTS &itemsToDistribute,
BOARD_COMMIT& aCommit ) const;
/** /**
* Distributes selected items using an even spacing between the centers of their bounding boxes * Distributes selected items using an even spacing between the centers of their bounding boxes
@ -144,15 +144,8 @@ private:
* NOTE: Using the centers of bounding box of items can give unsatisfactory visual results since * NOTE: Using the centers of bounding box of items can give unsatisfactory visual results since
* items of differing widths will be placed with different gaps. Is only used if items overlap * items of differing widths will be placed with different gaps. Is only used if items overlap
*/ */
void doDistributeCentersHorizontally( ALIGNMENT_RECTS &itemsToDistribute ) const; void doDistributeCentersVertically( ALIGNMENT_RECTS &itemsToDistribute,
BOARD_COMMIT& aCommit ) const;
/**
* Distributes selected items using an even spacing between the centers of their bounding boxes
*
* NOTE: Using the centers of bounding box of items can give unsatisfactory visual results since
* items of differing widths will be placed with different gaps. Is only used if items overlap
*/
void doDistributeCentersVertically( ALIGNMENT_RECTS &itemsToDistribute ) const;
/** /**
* Distributes selected items using an even spacing between their bounding boxes * Distributes selected items using an even spacing between their bounding boxes
@ -160,9 +153,8 @@ private:
* NOTE: Using the edges of bounding box of items is only possible if there is enough space * NOTE: Using the edges of bounding box of items is only possible if there is enough space
* between them. If this is not the case, use the center spacing method * between them. If this is not the case, use the center spacing method
*/ */
void doDistributeGapsHorizontally( ALIGNMENT_RECTS &itemsToDistribute, void doDistributeGapsHorizontally( ALIGNMENT_RECTS &itemsToDistribute, BOARD_COMMIT& aCommit,
const BOARD_ITEM *lastItem, const BOARD_ITEM *lastItem, int totalGap ) const;
int totalGap ) const;
/** /**
* Distributes selected items using an even spacing between their bounding boxes * Distributes selected items using an even spacing between their bounding boxes
@ -170,9 +162,8 @@ private:
* NOTE: Using the edges of bounding box of items is only possible if there is enough space * NOTE: Using the edges of bounding box of items is only possible if there is enough space
* between them. If this is not the case, use the center spacing method * between them. If this is not the case, use the center spacing method
*/ */
void doDistributeGapsVertically( ALIGNMENT_RECTS &itemsToDistribute, void doDistributeGapsVertically( ALIGNMENT_RECTS &itemsToDistribute, BOARD_COMMIT& aCommit,
const BOARD_ITEM *lastItem, const BOARD_ITEM *lastItem, int totalGap ) const;
int totalGap ) const;
private: private:
PCB_SELECTION_TOOL* m_selectionTool; PCB_SELECTION_TOOL* m_selectionTool;