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:
parent
e3d95cc2c8
commit
10e51bfea7
|
@ -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 ) )
|
||||||
|
return aGetValue( item );
|
||||||
|
}
|
||||||
|
|
||||||
|
return aGetValue( aLocked.front() );
|
||||||
|
}
|
||||||
|
|
||||||
|
for( const ALIGNMENT_RECT& item : aItems )
|
||||||
{
|
{
|
||||||
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( aItems.front() );
|
||||||
// not move
|
|
||||||
if( aLocked.empty() )
|
|
||||||
{
|
|
||||||
for( const ALIGNMENT_RECT& item : aItems )
|
|
||||||
{
|
|
||||||
if( item.second.Contains( curPos ) )
|
|
||||||
return aGetValue( item );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return target;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
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 ) );
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -485,10 +477,8 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeHorizontally( const TOOL_EVENT& aEvent )
|
||||||
if( selection.Size() <= 1 )
|
if( selection.Size() <= 1 )
|
||||||
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 totalWidth = 0;
|
int totalGap = maxRight - minX;
|
||||||
|
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 );
|
||||||
}
|
}
|
||||||
|
@ -604,10 +598,8 @@ int ALIGN_DISTRIBUTE_TOOL::DistributeVertically( const TOOL_EVENT& aEvent )
|
||||||
if( selection.Size() <= 1 )
|
if( selection.Size() <= 1 )
|
||||||
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 );
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
|
|
Loading…
Reference in New Issue