2019-05-08 18:56:03 +00:00
|
|
|
/*
|
|
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
|
|
*
|
2019-08-14 08:28:07 +00:00
|
|
|
* Copyright (C) 2019 CERN
|
2019-05-08 18:56:03 +00:00
|
|
|
* Copyright (C) 2019 KiCad Developers, see AUTHORS.txt for contributors.
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License
|
|
|
|
* as published by the Free Software Foundation; either version 2
|
|
|
|
* of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
* GNU General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, you may find one here:
|
|
|
|
* http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
|
|
|
* or you may search the http://www.gnu.org website for the version 2 license,
|
|
|
|
* or you may write to the Free Software Foundation, Inc.,
|
|
|
|
* 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <tool/tool_manager.h>
|
2019-05-10 17:19:48 +00:00
|
|
|
#include <tools/ee_selection_tool.h>
|
|
|
|
#include <ee_actions.h>
|
2019-05-08 18:56:03 +00:00
|
|
|
#include <bitmaps.h>
|
|
|
|
#include <base_struct.h>
|
|
|
|
#include <lib_edit_frame.h>
|
|
|
|
#include "lib_move_tool.h"
|
|
|
|
#include "lib_pin_tool.h"
|
|
|
|
|
|
|
|
|
|
|
|
LIB_MOVE_TOOL::LIB_MOVE_TOOL() :
|
2019-06-09 22:21:53 +00:00
|
|
|
EE_TOOL_BASE( "eeschema.SymbolMoveTool" ),
|
2019-05-08 18:56:03 +00:00
|
|
|
m_moveInProgress( false ),
|
|
|
|
m_moveOffset( 0, 0 )
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool LIB_MOVE_TOOL::Init()
|
|
|
|
{
|
2019-05-12 17:19:20 +00:00
|
|
|
EE_TOOL_BASE::Init();
|
2019-08-15 18:32:19 +00:00
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
//
|
|
|
|
// Add move actions to the selection tool menu
|
|
|
|
//
|
|
|
|
CONDITIONAL_MENU& selToolMenu = m_selectionTool->GetToolMenu().GetMenu();
|
|
|
|
|
2019-05-11 14:38:05 +00:00
|
|
|
selToolMenu.AddItem( EE_ACTIONS::move, EE_CONDITIONS::IdleSelection, 150 );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void LIB_MOVE_TOOL::Reset( RESET_REASON aReason )
|
|
|
|
{
|
2019-05-12 11:49:58 +00:00
|
|
|
EE_TOOL_BASE::Reset( aReason );
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
if( aReason == MODEL_RELOAD )
|
|
|
|
{
|
|
|
|
m_moveInProgress = false;
|
|
|
|
m_moveOffset = { 0, 0 };
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int LIB_MOVE_TOOL::Main( const TOOL_EVENT& aEvent )
|
|
|
|
{
|
|
|
|
KIGFX::VIEW_CONTROLS* controls = getViewControls();
|
2019-08-09 22:45:44 +00:00
|
|
|
|
|
|
|
m_anchorPos = { 0, 0 };
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
// Be sure that there is at least one item that we can move. If there's no selection try
|
|
|
|
// looking for the stuff under mouse cursor (i.e. Kicad old-style hover selection).
|
2019-06-08 21:48:22 +00:00
|
|
|
EE_SELECTION& selection = m_selectionTool->RequestSelection();
|
|
|
|
bool unselect = selection.IsHover();
|
2019-05-08 18:56:03 +00:00
|
|
|
|
2019-08-15 14:51:27 +00:00
|
|
|
if( selection.Empty() || m_moveInProgress )
|
2019-05-08 18:56:03 +00:00
|
|
|
return 0;
|
|
|
|
|
2019-07-15 12:15:58 +00:00
|
|
|
std::string tool = aEvent.GetCommandStr().get();
|
|
|
|
m_frame->PushTool( tool );
|
2019-05-08 18:56:03 +00:00
|
|
|
Activate();
|
2019-06-18 17:56:40 +00:00
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
controls->ShowCursor( true );
|
|
|
|
controls->SetAutoPan( true );
|
|
|
|
|
2019-06-17 13:43:22 +00:00
|
|
|
bool restore_state = false;
|
|
|
|
bool chain_commands = false;
|
|
|
|
TOOL_EVENT* evt = const_cast<TOOL_EVENT*>( &aEvent );
|
|
|
|
VECTOR2I prevPos;
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
if( !selection.Front()->IsNew() )
|
2020-08-26 18:04:32 +00:00
|
|
|
saveCopyInUndoList( m_frame->GetCurPart(), UNDO_REDO::LIBEDIT );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
2019-08-09 22:45:44 +00:00
|
|
|
m_cursor = controls->GetCursorPosition();
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
// Main loop: keep receiving events
|
|
|
|
do
|
|
|
|
{
|
2019-06-27 21:33:48 +00:00
|
|
|
m_frame->GetCanvas()->SetCurrentCursor( wxCURSOR_ARROW );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
2019-05-10 17:19:48 +00:00
|
|
|
if( evt->IsAction( &EE_ACTIONS::move ) || evt->IsMotion() || evt->IsDrag( BUT_LEFT )
|
2020-04-23 22:57:34 +00:00
|
|
|
|| evt->IsAction( &ACTIONS::refreshPreview )
|
|
|
|
|| evt->IsAction( &EE_ACTIONS::symbolMoveActivate ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( !m_moveInProgress ) // Prepare to start moving/dragging
|
|
|
|
{
|
2019-08-09 22:45:44 +00:00
|
|
|
LIB_ITEM* lib_item = (LIB_ITEM*) selection.Front();
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
// Pick up any synchronized pins
|
|
|
|
//
|
2019-06-26 23:24:09 +00:00
|
|
|
// Careful when pasting. The pasted pin will be at the same location as it
|
|
|
|
// was copied from, leading us to believe it's a synchronized pin. It's not.
|
2019-08-09 22:45:44 +00:00
|
|
|
if( selection.GetSize() == 1 && lib_item->Type() == LIB_PIN_T
|
2019-06-26 23:24:09 +00:00
|
|
|
&& m_frame->SynchronizePins()
|
2019-08-09 22:45:44 +00:00
|
|
|
&& ( lib_item->GetEditFlags() & IS_PASTED ) == 0 )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-08-09 22:45:44 +00:00
|
|
|
LIB_PIN* cur_pin = (LIB_PIN*) lib_item;
|
2019-05-08 18:56:03 +00:00
|
|
|
LIB_PART* part = m_frame->GetCurPart();
|
|
|
|
|
|
|
|
for( LIB_PIN* pin = part->GetNextPin(); pin; pin = part->GetNextPin( pin ) )
|
|
|
|
{
|
|
|
|
if( pin->GetPosition() == cur_pin->GetPosition()
|
|
|
|
&& pin->GetOrientation() == cur_pin->GetOrientation()
|
|
|
|
&& pin->GetConvert() == cur_pin->GetConvert() )
|
|
|
|
{
|
2019-09-08 18:43:54 +00:00
|
|
|
m_selectionTool->AddItemToSel( pin, true /*quiet mode*/ );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Apply any initial offset in case we're coming from a previous command.
|
|
|
|
//
|
|
|
|
for( EDA_ITEM* item : selection )
|
|
|
|
moveItem( item, m_moveOffset );
|
|
|
|
|
|
|
|
// Set up the starting position and move/drag offset
|
|
|
|
//
|
|
|
|
m_cursor = controls->GetCursorPosition();
|
|
|
|
|
2019-08-27 12:12:34 +00:00
|
|
|
if( lib_item->IsNew() )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-08-09 22:45:44 +00:00
|
|
|
m_anchorPos = selection.GetReferencePoint();
|
|
|
|
VECTOR2I delta = m_cursor - mapCoords( m_anchorPos );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
// Drag items to the current cursor position
|
|
|
|
for( EDA_ITEM* item : selection )
|
|
|
|
{
|
|
|
|
moveItem( item, delta );
|
2019-05-12 11:49:58 +00:00
|
|
|
updateView( item );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
|
2019-08-09 22:45:44 +00:00
|
|
|
m_anchorPos = m_cursor;
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
2019-08-09 22:45:44 +00:00
|
|
|
else if( selection.Size() == 1 && m_frame->GetMoveWarpsCursor() )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
2019-08-15 18:32:19 +00:00
|
|
|
wxPoint itemPos = lib_item->GetPosition();
|
|
|
|
m_anchorPos = wxPoint( itemPos.x, -itemPos.y );
|
|
|
|
|
2019-08-09 22:45:44 +00:00
|
|
|
getViewControls()->WarpCursor( m_anchorPos, true, true );
|
|
|
|
m_cursor = m_anchorPos;
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-08-09 22:45:44 +00:00
|
|
|
m_cursor = getViewControls()->GetCursorPosition( true );
|
|
|
|
m_anchorPos = m_cursor;
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
controls->SetCursorPosition( m_cursor, false );
|
2019-08-09 22:45:44 +00:00
|
|
|
m_toolMgr->PostEvent( EVENTS::SelectedItemsModified );
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
prevPos = m_cursor;
|
|
|
|
controls->SetAutoPan( true );
|
|
|
|
m_moveInProgress = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// Follow the mouse
|
|
|
|
//
|
|
|
|
m_cursor = controls->GetCursorPosition();
|
|
|
|
VECTOR2I delta( m_cursor - prevPos );
|
2019-08-09 22:45:44 +00:00
|
|
|
m_anchorPos = m_cursor;
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
m_moveOffset += delta;
|
|
|
|
prevPos = m_cursor;
|
|
|
|
|
|
|
|
for( EDA_ITEM* item : selection )
|
|
|
|
{
|
|
|
|
moveItem( item, delta );
|
2019-05-12 11:49:58 +00:00
|
|
|
updateView( item );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
|
2020-08-15 18:40:49 +00:00
|
|
|
m_toolMgr->PostEvent( EVENTS::SelectedItemsMoved );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// Handle cancel
|
|
|
|
//
|
2019-07-01 21:01:33 +00:00
|
|
|
else if( evt->IsCancelInteractive() || evt->IsActivate() )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( m_moveInProgress )
|
2020-08-25 11:53:39 +00:00
|
|
|
{
|
|
|
|
evt->SetPassEvent( false );
|
2019-05-08 18:56:03 +00:00
|
|
|
restore_state = true;
|
2020-08-25 11:53:39 +00:00
|
|
|
}
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// Handle TOOL_ACTION special cases
|
|
|
|
//
|
|
|
|
else if( evt->Action() == TA_UNDO_REDO_PRE )
|
|
|
|
{
|
|
|
|
unselect = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if( evt->Category() == TC_COMMAND )
|
|
|
|
{
|
2019-07-31 12:48:00 +00:00
|
|
|
if( evt->IsAction( &ACTIONS::doDelete ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
// Exit on a remove operation; there is no further processing for removed items.
|
|
|
|
break;
|
|
|
|
}
|
2019-05-27 10:40:52 +00:00
|
|
|
else if( evt->IsAction( &ACTIONS::duplicate ) )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( selection.Front()->IsNew() )
|
|
|
|
{
|
|
|
|
// This doesn't really make sense; we'll just end up dragging a stack of
|
|
|
|
// objects so Duplicate() is going to ignore this and we'll just carry on.
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Move original back and exit. The duplicate will run in its own loop.
|
|
|
|
restore_state = true;
|
|
|
|
unselect = false;
|
|
|
|
chain_commands = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// Handle context menu
|
|
|
|
//
|
|
|
|
else if( evt->IsClick( BUT_RIGHT ) )
|
|
|
|
{
|
2019-05-19 21:04:04 +00:00
|
|
|
m_menu.ShowContextMenu( m_selectionTool->GetSelection() );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// Handle drop
|
|
|
|
//
|
|
|
|
else if( evt->IsMouseUp( BUT_LEFT ) || evt->IsClick( BUT_LEFT ) )
|
|
|
|
{
|
|
|
|
if( selection.GetSize() == 1 && selection.Front()->Type() == LIB_PIN_T )
|
|
|
|
{
|
|
|
|
LIB_PIN_TOOL* pinTool = m_toolMgr->GetTool<LIB_PIN_TOOL>();
|
|
|
|
|
|
|
|
if( !pinTool->PlacePin( (LIB_PIN*) selection.Front() ) )
|
|
|
|
restore_state = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
break; // Finish
|
|
|
|
}
|
2019-07-26 18:16:44 +00:00
|
|
|
else
|
|
|
|
evt->SetPassEvent();
|
2019-05-08 18:56:03 +00:00
|
|
|
|
2019-06-17 13:43:22 +00:00
|
|
|
} while( ( evt = Wait() ) ); // Assignment intentional; not equality test
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
controls->ForceCursorPosition( false );
|
|
|
|
controls->ShowCursor( false );
|
|
|
|
controls->SetAutoPan( false );
|
|
|
|
|
|
|
|
if( !chain_commands )
|
|
|
|
m_moveOffset = { 0, 0 };
|
|
|
|
|
2019-08-09 22:45:44 +00:00
|
|
|
m_anchorPos = { 0, 0 };
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
for( auto item : selection )
|
2019-05-11 10:06:28 +00:00
|
|
|
item->ClearEditFlags();
|
2019-05-08 18:56:03 +00:00
|
|
|
|
|
|
|
if( restore_state )
|
2019-08-09 22:45:44 +00:00
|
|
|
{
|
2019-05-08 18:56:03 +00:00
|
|
|
m_frame->RollbackPartFromUndo();
|
2019-08-09 22:45:44 +00:00
|
|
|
|
|
|
|
if( unselect )
|
|
|
|
m_toolMgr->RunAction( EE_ACTIONS::clearSelection, true );
|
|
|
|
else
|
|
|
|
m_toolMgr->ProcessEvent( EVENTS::SelectedEvent );
|
|
|
|
}
|
2019-05-08 18:56:03 +00:00
|
|
|
else
|
2019-08-09 22:45:44 +00:00
|
|
|
{
|
|
|
|
if( unselect )
|
|
|
|
m_toolMgr->RunAction( EE_ACTIONS::clearSelection, true );
|
|
|
|
|
2019-05-08 18:56:03 +00:00
|
|
|
m_frame->OnModify();
|
2019-08-09 22:45:44 +00:00
|
|
|
}
|
2019-05-08 18:56:03 +00:00
|
|
|
|
2019-06-15 00:29:42 +00:00
|
|
|
m_moveInProgress = false;
|
2019-07-15 12:15:58 +00:00
|
|
|
m_frame->PopTool( tool );
|
2019-05-08 18:56:03 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void LIB_MOVE_TOOL::moveItem( EDA_ITEM* aItem, VECTOR2I aDelta )
|
|
|
|
{
|
2019-05-09 07:57:07 +00:00
|
|
|
static_cast<LIB_ITEM*>( aItem )->Offset( mapCoords( aDelta ));
|
2019-05-08 18:56:03 +00:00
|
|
|
aItem->SetFlags( IS_MOVED );
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-08 21:48:22 +00:00
|
|
|
bool LIB_MOVE_TOOL::updateModificationPoint( EE_SELECTION& aSelection )
|
2019-05-08 18:56:03 +00:00
|
|
|
{
|
|
|
|
if( m_moveInProgress && aSelection.HasReferencePoint() )
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// When there is only one item selected, the reference point is its position...
|
|
|
|
if( aSelection.Size() == 1 )
|
|
|
|
{
|
|
|
|
LIB_ITEM* item = static_cast<LIB_ITEM*>( aSelection.Front() );
|
|
|
|
aSelection.SetReferencePoint( item->GetPosition() );
|
|
|
|
}
|
|
|
|
// ...otherwise modify items with regard to the grid-snapped cursor position
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_cursor = getViewControls()->GetCursorPosition( true );
|
|
|
|
aSelection.SetReferencePoint( m_cursor );
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void LIB_MOVE_TOOL::setTransitions()
|
|
|
|
{
|
2019-05-10 17:19:48 +00:00
|
|
|
Go( &LIB_MOVE_TOOL::Main, EE_ACTIONS::move.MakeEvent() );
|
2020-04-23 22:57:34 +00:00
|
|
|
Go( &LIB_MOVE_TOOL::Main, EE_ACTIONS::symbolMoveActivate.MakeEvent() );
|
2019-05-08 18:56:03 +00:00
|
|
|
}
|