Use a more robust storage of items added to drag.

We kept having various issues of the pointer list going stale.  This
will prevent all permutations of that.

Also while I was in there I fixed the double-move thing (see bug
number).

Fixes https://gitlab.com/kicad/code/kicad/issues/7910
This commit is contained in:
Jeff Young 2021-03-22 12:51:32 +00:00
parent cfaca0e4bb
commit cf4cbea8a1
4 changed files with 54 additions and 24 deletions

View File

@ -24,6 +24,7 @@
#include <bitmaps.h> #include <bitmaps.h>
#include <core/typeinfo.h> #include <core/typeinfo.h>
#include <core/kicad_algo.h>
#include <ee_actions.h> #include <ee_actions.h>
#include <ee_collectors.h> #include <ee_collectors.h>
#include <ee_selection_tool.h> #include <ee_selection_tool.h>
@ -51,7 +52,6 @@
#include <view/view.h> #include <view/view.h>
#include <view/view_controls.h> #include <view/view_controls.h>
SELECTION_CONDITION EE_CONDITIONS::SingleSymbol = []( const SELECTION& aSel ) SELECTION_CONDITION EE_CONDITIONS::SingleSymbol = []( const SELECTION& aSel )
{ {
if( aSel.GetSize() == 1 ) if( aSel.GetSize() == 1 )
@ -1483,6 +1483,20 @@ void EE_SELECTION_TOOL::RemoveItemsFromSel( EDA_ITEMS* aList, bool aQuietMode )
} }
void EE_SELECTION_TOOL::RemoveItemsFromSel( std::vector<KIID>* aList, bool aQuietMode )
{
EDA_ITEMS removeItems;
for( EDA_ITEM* item : m_selection )
{
if( alg::contains( *aList, item->m_Uuid ) )
removeItems.push_back( item );
}
RemoveItemsFromSel( &removeItems, aQuietMode );
}
void EE_SELECTION_TOOL::BrightenItem( EDA_ITEM* aItem ) void EE_SELECTION_TOOL::BrightenItem( EDA_ITEM* aItem )
{ {
highlight( aItem, BRIGHTENED ); highlight( aItem, BRIGHTENED );

View File

@ -114,6 +114,12 @@ public:
int RemoveItemsFromSel( const TOOL_EVENT& aEvent ); int RemoveItemsFromSel( const TOOL_EVENT& aEvent );
void RemoveItemsFromSel( EDA_ITEMS* aList, bool aQuietMode = false ); void RemoveItemsFromSel( EDA_ITEMS* aList, bool aQuietMode = false );
/**
* A safer version of RemoveItemsFromSel( EDA_ITEMS ) which doesn't require the items to
* still exist.
*/
void RemoveItemsFromSel( std::vector<KIID>* aList, bool aQuietMode = false );
void BrightenItem( EDA_ITEM* aItem ); void BrightenItem( EDA_ITEM* aItem );
void UnbrightenItem( EDA_ITEM* aItem ); void UnbrightenItem( EDA_ITEM* aItem );

View File

@ -47,7 +47,7 @@
SCH_MOVE_TOOL::SCH_MOVE_TOOL() : SCH_MOVE_TOOL::SCH_MOVE_TOOL() :
EE_TOOL_BASE<SCH_EDIT_FRAME>( "eeschema.InteractiveMove" ), EE_TOOL_BASE<SCH_EDIT_FRAME>( "eeschema.InteractiveMove" ),
m_moveInProgress( false ), m_moveInProgress( false ),
m_isDragOperation( false ), m_isDrag( false ),
m_moveOffset( 0, 0 ) m_moveOffset( 0, 0 )
{ {
} }
@ -93,36 +93,40 @@ int SCH_MOVE_TOOL::Main( const TOOL_EVENT& aEvent )
EESCHEMA_SETTINGS* cfg = Pgm().GetSettingsManager().GetAppSettings<EESCHEMA_SETTINGS>(); EESCHEMA_SETTINGS* cfg = Pgm().GetSettingsManager().GetAppSettings<EESCHEMA_SETTINGS>();
KIGFX::VIEW_CONTROLS* controls = getViewControls(); KIGFX::VIEW_CONTROLS* controls = getViewControls();
EE_GRID_HELPER grid( m_toolMgr ); EE_GRID_HELPER grid( m_toolMgr );
bool wasDragging = m_moveInProgress && m_isDrag;
m_anchorPos.reset(); m_anchorPos.reset();
if( aEvent.IsAction( &EE_ACTIONS::move ) ) if( aEvent.IsAction( &EE_ACTIONS::move ) )
m_isDragOperation = false; m_isDrag = false;
else if( aEvent.IsAction( &EE_ACTIONS::drag ) ) else if( aEvent.IsAction( &EE_ACTIONS::drag ) )
m_isDragOperation = true; m_isDrag = true;
else if( aEvent.IsAction( &EE_ACTIONS::moveActivate ) ) else if( aEvent.IsAction( &EE_ACTIONS::moveActivate ) )
m_isDragOperation = !cfg->m_Input.drag_is_move; m_isDrag = !cfg->m_Input.drag_is_move;
else else
return 0; return 0;
if( m_moveInProgress ) if( m_moveInProgress )
{
if( m_isDrag != wasDragging )
{ {
auto sel = m_selectionTool->GetSelection().Front(); auto sel = m_selectionTool->GetSelection().Front();
if( sel && !sel->IsNew() ) if( sel && !sel->IsNew() )
{ {
// User must have switched from move to drag or vice-versa. Reset the selected // Reset the selected items so we can start again with the current m_isDrag
// items so we can start again with the current m_isDragOperation. // state.
m_frame->RollbackSchematicFromUndo(); m_frame->RollbackSchematicFromUndo();
m_selectionTool->RemoveItemsFromSel( &m_dragAdditions, QUIET_MODE ); m_selectionTool->RemoveItemsFromSel( &m_dragAdditions, QUIET_MODE );
m_anchorPos = m_cursor - m_moveOffset; m_anchorPos = m_cursor - m_moveOffset;
m_moveInProgress = false; m_moveInProgress = false;
controls->SetAutoPan( false ); controls->SetAutoPan( false );
// And give it a kick so it doesn't have to wait for the first mouse movement to // And give it a kick so it doesn't have to wait for the first mouse movement
// refresh. // to refresh.
m_toolMgr->RunAction( EE_ACTIONS::restartMove ); m_toolMgr->RunAction( EE_ACTIONS::restartMove );
} }
}
return 0; return 0;
} }
@ -196,8 +200,10 @@ int SCH_MOVE_TOOL::Main( const TOOL_EVENT& aEvent )
it->SetFlags( STARTPOINT | ENDPOINT ); it->SetFlags( STARTPOINT | ENDPOINT );
} }
if( m_isDragOperation ) if( m_isDrag )
{ {
EDA_ITEMS connectedDragItems;
// Add connections to the selection for a drag. // Add connections to the selection for a drag.
// //
for( EDA_ITEM* edaItem : selection ) for( EDA_ITEM* edaItem : selection )
@ -211,10 +217,14 @@ int SCH_MOVE_TOOL::Main( const TOOL_EVENT& aEvent )
connections = item->GetConnectionPoints(); connections = item->GetConnectionPoints();
for( wxPoint point : connections ) for( wxPoint point : connections )
getConnectedDragItems( item, point, m_dragAdditions ); getConnectedDragItems( item, point, connectedDragItems );
} }
m_selectionTool->AddItemsToSel( &m_dragAdditions, QUIET_MODE ); for( EDA_ITEM* item : connectedDragItems )
{
m_dragAdditions.push_back( item->m_Uuid );
m_selectionTool->AddItemToSel( item, QUIET_MODE );
}
} }
else else
{ {
@ -246,7 +256,7 @@ int SCH_MOVE_TOOL::Main( const TOOL_EVENT& aEvent )
if( item->IsNew() ) if( item->IsNew() )
{ {
if( item->HasFlag( TEMP_SELECTED ) && m_isDragOperation ) if( item->HasFlag( TEMP_SELECTED ) && m_isDrag )
{ {
// Item was added in getConnectedDragItems // Item was added in getConnectedDragItems
saveCopyInUndoList( (SCH_ITEM*) item, UNDO_REDO::NEWITEM, appendUndo ); saveCopyInUndoList( (SCH_ITEM*) item, UNDO_REDO::NEWITEM, appendUndo );

View File

@ -76,10 +76,10 @@ private:
private: private:
///< Flag determining if anything is being dragged right now ///< Flag determining if anything is being dragged right now
bool m_moveInProgress; bool m_moveInProgress;
bool m_isDragOperation; bool m_isDrag;
///< Items (such as wires) which were added to the selection for a drag ///< Items (such as wires) which were added to the selection for a drag
EDA_ITEMS m_dragAdditions; std::vector<KIID> m_dragAdditions;
///< Used for chaining commands ///< Used for chaining commands
VECTOR2I m_moveOffset; VECTOR2I m_moveOffset;