pcbnew: Align/Distribute handle locking
This fixes the handling of align/distribute tool when called on locked items. Locked items cannot be moved but they may be used for the target of an align/distribute operation. Fixes: lp:1808238 * https://bugs.launchpad.net/kicad/+bug/1808238
This commit is contained in:
parent
0a26388901
commit
7e9fee285f
|
@ -127,87 +127,106 @@ bool ALIGN_DISTRIBUTE_TOOL::Init()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
bool SortLeftmostX( const std::pair<BOARD_ITEM*, EDA_RECT> left, const std::pair<BOARD_ITEM*, EDA_RECT> right)
|
template <class T>
|
||||||
|
ALIGNMENT_RECTS GetBoundingBoxes( const T &sel )
|
||||||
{
|
{
|
||||||
return ( left.second.GetX() < right.second.GetX() );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool SortRightmostX( const std::pair<BOARD_ITEM*, EDA_RECT> left, const std::pair<BOARD_ITEM*, EDA_RECT> right)
|
|
||||||
{
|
|
||||||
return ( left.second.GetRight() > right.second.GetRight() );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool SortTopmostY( const std::pair<BOARD_ITEM*, EDA_RECT> left, const std::pair<BOARD_ITEM*, EDA_RECT> right)
|
|
||||||
{
|
|
||||||
return ( left.second.GetY() < right.second.GetY() );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool SortCenterX( const std::pair<BOARD_ITEM*, EDA_RECT> left, const std::pair<BOARD_ITEM*, EDA_RECT> right)
|
|
||||||
{
|
|
||||||
return ( left.second.GetCenter().x < right.second.GetCenter().x );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool SortCenterY( const std::pair<BOARD_ITEM*, EDA_RECT> left, const std::pair<BOARD_ITEM*, EDA_RECT> right)
|
|
||||||
{
|
|
||||||
return ( left.second.GetCenter().y < right.second.GetCenter().y );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
bool SortBottommostY( const std::pair<BOARD_ITEM*, EDA_RECT> left, const std::pair<BOARD_ITEM*, EDA_RECT> right)
|
|
||||||
{
|
|
||||||
return ( left.second.GetBottom() > right.second.GetBottom() );
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
ALIGNMENT_RECTS GetBoundingBoxes( const SELECTION &sel )
|
|
||||||
{
|
|
||||||
const SELECTION& selection = sel;
|
|
||||||
|
|
||||||
ALIGNMENT_RECTS rects;
|
ALIGNMENT_RECTS rects;
|
||||||
|
|
||||||
for( auto item : selection )
|
for( auto item : sel )
|
||||||
{
|
{
|
||||||
BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( item );
|
BOARD_ITEM* boardItem = static_cast<BOARD_ITEM*>( item );
|
||||||
|
|
||||||
if( item->Type() == PCB_MODULE_T )
|
if( item->Type() == PCB_MODULE_T )
|
||||||
{
|
|
||||||
rects.emplace_back( std::make_pair( boardItem, static_cast<MODULE*>( item )->GetFootprintRect() ) );
|
rects.emplace_back( std::make_pair( boardItem, static_cast<MODULE*>( item )->GetFootprintRect() ) );
|
||||||
}
|
|
||||||
else
|
else
|
||||||
{
|
|
||||||
rects.emplace_back( std::make_pair( boardItem, item->GetBoundingBox() ) );
|
rects.emplace_back( std::make_pair( boardItem, item->GetBoundingBox() ) );
|
||||||
}
|
}
|
||||||
}
|
|
||||||
return rects;
|
return rects;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template< typename T >
|
||||||
|
int ALIGN_DISTRIBUTE_TOOL::selectTarget( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECTS& aLocked, T aGetValue )
|
||||||
|
{
|
||||||
|
wxPoint curPos( getViewControls()->GetCursorPosition().x, getViewControls()->GetCursorPosition().y );
|
||||||
|
|
||||||
|
// after sorting, the fist item acts as the target for all others
|
||||||
|
// unless we have a locked item, in which case we use that for the target
|
||||||
|
int target = !aLocked.size() ? aGetValue( aItems.front() ): aGetValue( aLocked.front() );
|
||||||
|
|
||||||
|
// Iterate through both lists to find if we are mouse-over on one of the items.
|
||||||
|
for( auto sel = aLocked.begin(); sel != aItems.end(); sel++ )
|
||||||
|
{
|
||||||
|
// If there are locked items, prefer aligning to them over
|
||||||
|
// aligning to the cursor as they do not move
|
||||||
|
if( sel == aLocked.end() )
|
||||||
|
{
|
||||||
|
if( aLocked.size() == 0 )
|
||||||
|
{
|
||||||
|
sel = aItems.begin();
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We take the first target that overlaps our cursor.
|
||||||
|
// This is deterministic because we assume sorted arrays
|
||||||
|
if( sel->second.Contains( curPos ) )
|
||||||
|
{
|
||||||
|
target = aGetValue( *sel );
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return target;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
template< typename T >
|
||||||
|
size_t ALIGN_DISTRIBUTE_TOOL::GetSelections( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECTS& aLocked, T aCompare )
|
||||||
|
{
|
||||||
|
|
||||||
|
SELECTION& selection = m_selectionTool->RequestSelection(
|
||||||
|
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector )
|
||||||
|
{ EditToolSelectionFilter( aCollector, EXCLUDE_TRANSIENTS ); } );
|
||||||
|
|
||||||
|
if( selection.Size() <= 1 )
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
std::vector<BOARD_ITEM*> lockedItems;
|
||||||
|
selection = m_selectionTool->RequestSelection(
|
||||||
|
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector )
|
||||||
|
{ EditToolSelectionFilter( aCollector, EXCLUDE_LOCKED ); }, &lockedItems );
|
||||||
|
|
||||||
|
aItems = GetBoundingBoxes( selection );
|
||||||
|
aLocked = GetBoundingBoxes( lockedItems );
|
||||||
|
std::sort( aItems.begin(), aItems.end(), aCompare );
|
||||||
|
std::sort( aLocked.begin(), aLocked.end(), aCompare );
|
||||||
|
|
||||||
|
return aItems.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
int ALIGN_DISTRIBUTE_TOOL::AlignTop( const TOOL_EVENT& aEvent )
|
int ALIGN_DISTRIBUTE_TOOL::AlignTop( const TOOL_EVENT& aEvent )
|
||||||
{
|
{
|
||||||
SELECTION& selection = m_selectionTool->RequestSelection(
|
ALIGNMENT_RECTS itemsToAlign;
|
||||||
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector )
|
ALIGNMENT_RECTS locked_items;
|
||||||
{ EditToolSelectionFilter( aCollector, EXCLUDE_LOCKED | EXCLUDE_TRANSIENTS ); } );
|
|
||||||
|
|
||||||
if( selection.Size() <= 1 )
|
if( !GetSelections( itemsToAlign, locked_items, []( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetTop() < right.second.GetTop() ); } ) )
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
auto itemsToAlign = GetBoundingBoxes( selection );
|
|
||||||
std::sort( itemsToAlign.begin(), itemsToAlign.end(), SortTopmostY );
|
|
||||||
|
|
||||||
BOARD_COMMIT commit( m_frame );
|
BOARD_COMMIT commit( m_frame );
|
||||||
commit.StageItems( selection, CHT_MODIFY );
|
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
|
||||||
|
auto targetTop = selectTarget( itemsToAlign, locked_items, []( const ALIGNMENT_RECT& aVal )
|
||||||
// after sorting, the fist item acts as the target for all others
|
{ return aVal.second.GetTop(); } );
|
||||||
const int targetTop = itemsToAlign.begin()->second.GetY();
|
|
||||||
|
|
||||||
// Move the selected items
|
// Move the selected items
|
||||||
for( auto& i : itemsToAlign )
|
for( auto& i : itemsToAlign )
|
||||||
{
|
{
|
||||||
int difference = targetTop - i.second.GetY();
|
int difference = targetTop - i.second.GetTop();
|
||||||
BOARD_ITEM* item = i.first;
|
BOARD_ITEM* item = i.first;
|
||||||
|
|
||||||
// Don't move a pad by itself unless editing the footprint
|
// Don't move a pad by itself unless editing the footprint
|
||||||
|
@ -225,21 +244,17 @@ int ALIGN_DISTRIBUTE_TOOL::AlignTop( const TOOL_EVENT& aEvent )
|
||||||
|
|
||||||
int ALIGN_DISTRIBUTE_TOOL::AlignBottom( const TOOL_EVENT& aEvent )
|
int ALIGN_DISTRIBUTE_TOOL::AlignBottom( const TOOL_EVENT& aEvent )
|
||||||
{
|
{
|
||||||
SELECTION& selection = m_selectionTool->RequestSelection(
|
ALIGNMENT_RECTS itemsToAlign;
|
||||||
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector )
|
ALIGNMENT_RECTS locked_items;
|
||||||
{ EditToolSelectionFilter( aCollector, EXCLUDE_LOCKED | EXCLUDE_TRANSIENTS ); } );
|
|
||||||
|
|
||||||
if( selection.Size() <= 1 )
|
if( !GetSelections( itemsToAlign, locked_items, []( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetBottom() < right.second.GetBottom() ); } ) )
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
auto itemsToAlign = GetBoundingBoxes( selection );
|
|
||||||
std::sort( itemsToAlign.begin(), itemsToAlign.end(), SortBottommostY );
|
|
||||||
|
|
||||||
BOARD_COMMIT commit( m_frame );
|
BOARD_COMMIT commit( m_frame );
|
||||||
commit.StageItems( selection, CHT_MODIFY );
|
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
|
||||||
|
auto targetBottom = selectTarget( itemsToAlign, locked_items, []( const ALIGNMENT_RECT& aVal )
|
||||||
// after sorting, the fist item acts as the target for all others
|
{ return aVal.second.GetBottom(); } );
|
||||||
const int targetBottom = itemsToAlign.begin()->second.GetBottom();
|
|
||||||
|
|
||||||
// Move the selected items
|
// Move the selected items
|
||||||
for( auto& i : itemsToAlign )
|
for( auto& i : itemsToAlign )
|
||||||
|
@ -277,26 +292,22 @@ int ALIGN_DISTRIBUTE_TOOL::AlignLeft( const TOOL_EVENT& aEvent )
|
||||||
|
|
||||||
int ALIGN_DISTRIBUTE_TOOL::doAlignLeft()
|
int ALIGN_DISTRIBUTE_TOOL::doAlignLeft()
|
||||||
{
|
{
|
||||||
SELECTION& selection = m_selectionTool->RequestSelection(
|
ALIGNMENT_RECTS itemsToAlign;
|
||||||
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector )
|
ALIGNMENT_RECTS locked_items;
|
||||||
{ EditToolSelectionFilter( aCollector, EXCLUDE_LOCKED | EXCLUDE_TRANSIENTS ); } );
|
|
||||||
|
|
||||||
if( selection.Size() <= 1 )
|
if( !GetSelections( itemsToAlign, locked_items, []( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetLeft() < right.second.GetLeft() ); } ) )
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
auto itemsToAlign = GetBoundingBoxes( selection );
|
|
||||||
std::sort( itemsToAlign.begin(), itemsToAlign.end(), SortLeftmostX );
|
|
||||||
|
|
||||||
BOARD_COMMIT commit( m_frame );
|
BOARD_COMMIT commit( m_frame );
|
||||||
commit.StageItems( selection, CHT_MODIFY );
|
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
|
||||||
|
auto targetLeft = selectTarget( itemsToAlign, locked_items, []( const ALIGNMENT_RECT& aVal )
|
||||||
// after sorting, the fist item acts as the target for all others
|
{ return aVal.second.GetLeft(); } );
|
||||||
const int targetLeft = itemsToAlign.begin()->second.GetX();
|
|
||||||
|
|
||||||
// Move the selected items
|
// Move the selected items
|
||||||
for( auto& i : itemsToAlign )
|
for( auto& i : itemsToAlign )
|
||||||
{
|
{
|
||||||
int difference = targetLeft - i.second.GetX();
|
int difference = targetLeft - i.second.GetLeft();
|
||||||
BOARD_ITEM* item = i.first;
|
BOARD_ITEM* item = i.first;
|
||||||
|
|
||||||
// Don't move a pad by itself unless editing the footprint
|
// Don't move a pad by itself unless editing the footprint
|
||||||
|
@ -329,21 +340,17 @@ int ALIGN_DISTRIBUTE_TOOL::AlignRight( const TOOL_EVENT& aEvent )
|
||||||
|
|
||||||
int ALIGN_DISTRIBUTE_TOOL::doAlignRight()
|
int ALIGN_DISTRIBUTE_TOOL::doAlignRight()
|
||||||
{
|
{
|
||||||
SELECTION& selection = m_selectionTool->RequestSelection(
|
ALIGNMENT_RECTS itemsToAlign;
|
||||||
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector )
|
ALIGNMENT_RECTS locked_items;
|
||||||
{ EditToolSelectionFilter( aCollector, EXCLUDE_LOCKED | EXCLUDE_TRANSIENTS ); } );
|
|
||||||
|
|
||||||
if( selection.Size() <= 1 )
|
if( !GetSelections( itemsToAlign, locked_items, []( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetRight() < right.second.GetRight() ); } ) )
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
auto itemsToAlign = GetBoundingBoxes( selection );
|
|
||||||
std::sort( itemsToAlign.begin(), itemsToAlign.end(), SortRightmostX );
|
|
||||||
|
|
||||||
BOARD_COMMIT commit( m_frame );
|
BOARD_COMMIT commit( m_frame );
|
||||||
commit.StageItems( selection, CHT_MODIFY );
|
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
|
||||||
|
auto targetRight = selectTarget( itemsToAlign, locked_items, []( const ALIGNMENT_RECT& aVal )
|
||||||
// after sorting, the fist item acts as the target for all others
|
{ return aVal.second.GetRight(); } );
|
||||||
const int targetRight = itemsToAlign.begin()->second.GetRight();
|
|
||||||
|
|
||||||
// Move the selected items
|
// Move the selected items
|
||||||
for( auto& i : itemsToAlign )
|
for( auto& i : itemsToAlign )
|
||||||
|
@ -366,22 +373,17 @@ int ALIGN_DISTRIBUTE_TOOL::doAlignRight()
|
||||||
|
|
||||||
int ALIGN_DISTRIBUTE_TOOL::AlignCenterX( const TOOL_EVENT& aEvent )
|
int ALIGN_DISTRIBUTE_TOOL::AlignCenterX( const TOOL_EVENT& aEvent )
|
||||||
{
|
{
|
||||||
SELECTION& selection = m_selectionTool->RequestSelection(
|
ALIGNMENT_RECTS itemsToAlign;
|
||||||
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector )
|
ALIGNMENT_RECTS locked_items;
|
||||||
{ EditToolSelectionFilter( aCollector, EXCLUDE_LOCKED | EXCLUDE_TRANSIENTS ); } );
|
|
||||||
|
|
||||||
if( selection.Size() <= 1 )
|
if( !GetSelections( itemsToAlign, locked_items, []( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetCenter().x < right.second.GetCenter().x ); } ) )
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
auto itemsToAlign = GetBoundingBoxes( selection );
|
|
||||||
std::sort( itemsToAlign.begin(), itemsToAlign.end(), SortCenterX );
|
|
||||||
|
|
||||||
BOARD_COMMIT commit( m_frame );
|
BOARD_COMMIT commit( m_frame );
|
||||||
commit.StageItems( selection, CHT_MODIFY );
|
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
|
||||||
|
auto targetX = selectTarget( itemsToAlign, locked_items, []( const ALIGNMENT_RECT& aVal )
|
||||||
// after sorting use the center x coordinate of the leftmost item as a target
|
{ return aVal.second.GetCenter().x; } );
|
||||||
// for all other items
|
|
||||||
const int targetX = itemsToAlign.begin()->second.GetCenter().x;
|
|
||||||
|
|
||||||
// Move the selected items
|
// Move the selected items
|
||||||
for( auto& i : itemsToAlign )
|
for( auto& i : itemsToAlign )
|
||||||
|
@ -404,22 +406,17 @@ int ALIGN_DISTRIBUTE_TOOL::AlignCenterX( const TOOL_EVENT& aEvent )
|
||||||
|
|
||||||
int ALIGN_DISTRIBUTE_TOOL::AlignCenterY( const TOOL_EVENT& aEvent )
|
int ALIGN_DISTRIBUTE_TOOL::AlignCenterY( const TOOL_EVENT& aEvent )
|
||||||
{
|
{
|
||||||
SELECTION& selection = m_selectionTool->RequestSelection(
|
ALIGNMENT_RECTS itemsToAlign;
|
||||||
[]( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector )
|
ALIGNMENT_RECTS locked_items;
|
||||||
{ EditToolSelectionFilter( aCollector, EXCLUDE_LOCKED | EXCLUDE_TRANSIENTS ); } );
|
|
||||||
|
|
||||||
if( selection.Size() <= 1 )
|
if( !GetSelections( itemsToAlign, locked_items, []( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetCenter().y < right.second.GetCenter().y ); } ) )
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
auto itemsToAlign = GetBoundingBoxes( selection );
|
|
||||||
std::sort( itemsToAlign.begin(), itemsToAlign.end(), SortCenterY );
|
|
||||||
|
|
||||||
BOARD_COMMIT commit( m_frame );
|
BOARD_COMMIT commit( m_frame );
|
||||||
commit.StageItems( selection, CHT_MODIFY );
|
commit.StageItems( m_selectionTool->GetSelection(), CHT_MODIFY );
|
||||||
|
auto targetY = selectTarget( itemsToAlign, locked_items, []( const ALIGNMENT_RECT& aVal )
|
||||||
// after sorting use the center y coordinate of the top-most item as a target
|
{ return aVal.second.GetCenter().y; } );
|
||||||
// for all other items
|
|
||||||
const int targetY = itemsToAlign.begin()->second.GetCenter().y;
|
|
||||||
|
|
||||||
// Move the selected items
|
// Move the selected items
|
||||||
for( auto& i : itemsToAlign )
|
for( auto& i : itemsToAlign )
|
||||||
|
@ -455,13 +452,17 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeHorizontally( const TOOL_EVENT& aEvent )
|
||||||
auto 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(), SortRightmostX );
|
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
|
||||||
|
[] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetRight() > right.second.GetRight() ); } );
|
||||||
const auto lastItem = itemsToDistribute.begin()->first;
|
const auto lastItem = itemsToDistribute.begin()->first;
|
||||||
|
|
||||||
const auto maxRight = itemsToDistribute.begin()->second.GetRight();
|
const auto maxRight = itemsToDistribute.begin()->second.GetRight();
|
||||||
|
|
||||||
// sort to get starting order
|
// sort to get starting order
|
||||||
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), SortLeftmostX );
|
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
|
||||||
|
[] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetX() < right.second.GetX() ); } );
|
||||||
const auto minX = itemsToDistribute.begin()->second.GetX();
|
const auto minX = itemsToDistribute.begin()->second.GetX();
|
||||||
auto totalGap = maxRight - minX;
|
auto totalGap = maxRight - minX;
|
||||||
int totalWidth = 0;
|
int totalWidth = 0;
|
||||||
|
@ -515,7 +516,9 @@ void ALIGN_DISTRIBUTE_TOOL::doDistributeGapsHorizontally( ALIGNMENT_RECTS& items
|
||||||
|
|
||||||
void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersHorizontally( ALIGNMENT_RECTS &itemsToDistribute ) const
|
void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersHorizontally( ALIGNMENT_RECTS &itemsToDistribute ) const
|
||||||
{
|
{
|
||||||
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), SortCenterX );
|
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
|
||||||
|
[] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetCenter().x < right.second.GetCenter().x ); } );
|
||||||
const auto totalGap = ( itemsToDistribute.end()-1 )->second.GetCenter().x
|
const auto 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 auto itemGap = totalGap / ( itemsToDistribute.size() - 1 );
|
||||||
|
@ -551,12 +554,16 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeVertically( const TOOL_EVENT& aEvent )
|
||||||
auto 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(), SortBottommostY );
|
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
|
||||||
|
[] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetBottom() > right.second.GetBottom() ); } );
|
||||||
const auto maxBottom = itemsToDistribute.begin()->second.GetBottom();
|
const auto maxBottom = itemsToDistribute.begin()->second.GetBottom();
|
||||||
const auto lastItem = itemsToDistribute.begin()->first;
|
const auto lastItem = itemsToDistribute.begin()->first;
|
||||||
|
|
||||||
// sort to get starting order
|
// sort to get starting order
|
||||||
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), SortTopmostY );
|
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
|
||||||
|
[] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetCenter().y < right.second.GetCenter().y ); } );
|
||||||
auto minY = itemsToDistribute.begin()->second.GetY();
|
auto minY = itemsToDistribute.begin()->second.GetY();
|
||||||
|
|
||||||
auto totalGap = maxBottom - minY;
|
auto totalGap = maxBottom - minY;
|
||||||
|
@ -611,7 +618,9 @@ void ALIGN_DISTRIBUTE_TOOL::doDistributeGapsVertically( ALIGNMENT_RECTS& itemsTo
|
||||||
|
|
||||||
void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersVertically( ALIGNMENT_RECTS& itemsToDistribute ) const
|
void ALIGN_DISTRIBUTE_TOOL::doDistributeCentersVertically( ALIGNMENT_RECTS& itemsToDistribute ) const
|
||||||
{
|
{
|
||||||
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(), SortCenterY );
|
std::sort( itemsToDistribute.begin(), itemsToDistribute.end(),
|
||||||
|
[] ( const ALIGNMENT_RECT left, const ALIGNMENT_RECT right)
|
||||||
|
{ return ( left.second.GetCenter().y < right.second.GetCenter().y ); } );
|
||||||
const auto totalGap = ( itemsToDistribute.end()-1 )->second.GetCenter().y
|
const auto 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 auto itemGap = totalGap / ( itemsToDistribute.size() - 1 );
|
||||||
|
|
|
@ -29,7 +29,8 @@
|
||||||
#include <class_board_item.h>
|
#include <class_board_item.h>
|
||||||
#include <pcb_base_frame.h>
|
#include <pcb_base_frame.h>
|
||||||
|
|
||||||
typedef std::vector<std::pair<BOARD_ITEM*, EDA_RECT>> ALIGNMENT_RECTS;
|
using ALIGNMENT_RECT = std::pair<BOARD_ITEM*, EDA_RECT>;
|
||||||
|
using ALIGNMENT_RECTS = std::vector<ALIGNMENT_RECT>;
|
||||||
|
|
||||||
class SELECTION_TOOL;
|
class SELECTION_TOOL;
|
||||||
|
|
||||||
|
@ -100,6 +101,17 @@ public:
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Function GetSelections()
|
||||||
|
* Populates two vectors with the sorted selection and sorted locked items
|
||||||
|
* Returns the size of aItems()
|
||||||
|
*/
|
||||||
|
template< typename T >
|
||||||
|
size_t GetSelections( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECTS& aLocked, T aCompare );
|
||||||
|
|
||||||
|
template< typename T >
|
||||||
|
int selectTarget( ALIGNMENT_RECTS& aItems, ALIGNMENT_RECTS& aLocked, T aGetValue );
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Sets X coordinate of the selected items to the value of the left-most selected item X coordinate.
|
* Sets X coordinate of the selected items to the value of the left-most selected item X coordinate.
|
||||||
*
|
*
|
||||||
|
|
Loading…
Reference in New Issue