494 lines
14 KiB
C++
494 lines
14 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 2015 Jean-Pierre Charras, jean-pierre.charras@ujf-grenoble.fr
|
|
* Copyright (C) 2012 SoftPLC Corporation, Dick Hollenbeck <dick@softplc.com>
|
|
* Copyright (C) 2012 Wayne Stambaugh <stambaughw@verizon.net>
|
|
* Copyright (C) 1992-2015 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
|
|
*/
|
|
|
|
/**
|
|
* @file modules.cpp
|
|
*/
|
|
|
|
#include <fctsys.h>
|
|
#include <gr_basic.h>
|
|
#include <class_drawpanel.h>
|
|
#include <confirm.h>
|
|
#include <pcb_edit_frame.h>
|
|
#include <trigo.h>
|
|
#include <macros.h>
|
|
|
|
#include <class_board.h>
|
|
#include <class_module.h>
|
|
|
|
#include <pcbnew.h>
|
|
#include <drag.h>
|
|
#include <dialog_get_footprint_by_name.h>
|
|
|
|
#include <connectivity_data.h>
|
|
|
|
static void MoveFootprint( EDA_DRAW_PANEL* aPanel, wxDC* aDC,
|
|
const wxPoint& aPosition, bool aErase );
|
|
static void Abort_MoveOrCopyModule( EDA_DRAW_PANEL* Panel, wxDC* DC );
|
|
|
|
|
|
static MODULE* s_ModuleInitialCopy = NULL; /* Copy of module for
|
|
* abort/undo command
|
|
*/
|
|
static PICKED_ITEMS_LIST s_PickedList; /* a picked list to
|
|
* save initial module
|
|
* and dragged tracks
|
|
*/
|
|
|
|
|
|
MODULE* PCB_BASE_FRAME::GetFootprintFromBoardByReference()
|
|
{
|
|
wxString moduleName;
|
|
MODULE* module = NULL;
|
|
wxArrayString fplist;
|
|
|
|
// Build list of available fp references, to display them in dialog
|
|
for( MODULE* fp = GetBoard()->m_Modules; fp; fp = fp->Next() )
|
|
fplist.Add( fp->GetReference() + wxT(" ( ") + fp->GetValue() + wxT(" )") );
|
|
|
|
fplist.Sort();
|
|
|
|
DIALOG_GET_FOOTPRINT_BY_NAME dlg( this, fplist );
|
|
|
|
if( dlg.ShowModal() != wxID_OK ) //Aborted by user
|
|
return NULL;
|
|
|
|
moduleName = dlg.GetValue();
|
|
moduleName.Trim( true );
|
|
moduleName.Trim( false );
|
|
|
|
if( !moduleName.IsEmpty() )
|
|
{
|
|
module = GetBoard()->m_Modules;
|
|
|
|
while( module )
|
|
{
|
|
if( module->GetReference().CmpNoCase( moduleName ) == 0 )
|
|
break;
|
|
|
|
module = module->Next();
|
|
}
|
|
}
|
|
|
|
return module;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::StartMoveModule( MODULE* aModule, wxDC* aDC,
|
|
bool aDragConnectedTracks )
|
|
{
|
|
if( aModule == NULL )
|
|
return;
|
|
|
|
if( s_ModuleInitialCopy )
|
|
delete s_ModuleInitialCopy;
|
|
|
|
s_PickedList.ClearItemsList(); // Should be empty, but...
|
|
|
|
// Creates a copy of the current module, for abort and undo commands
|
|
s_ModuleInitialCopy = (MODULE*)aModule->Clone();
|
|
s_ModuleInitialCopy->SetParent( GetBoard() );
|
|
s_ModuleInitialCopy->ClearFlags();
|
|
|
|
SetCurItem( aModule );
|
|
GetBoard()->m_Status_Pcb &= ~RATSNEST_ITEM_LOCAL_OK;
|
|
aModule->SetFlags( IS_MOVED );
|
|
|
|
/* Show ratsnest. */
|
|
if( GetBoard()->IsElementVisible( LAYER_RATSNEST ) )
|
|
DrawGeneralRatsnest( aDC );
|
|
|
|
EraseDragList();
|
|
|
|
if( aDragConnectedTracks )
|
|
{
|
|
DRAG_LIST drglist( GetBoard() );
|
|
drglist.BuildDragListe( aModule );
|
|
|
|
ITEM_PICKER itemWrapper( NULL, UR_CHANGED );
|
|
|
|
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ )
|
|
{
|
|
TRACK* segm = g_DragSegmentList[ii].m_Track;
|
|
itemWrapper.SetItem( segm );
|
|
itemWrapper.SetLink( segm->Clone() );
|
|
itemWrapper.GetLink()->SetState( IN_EDIT, false );
|
|
s_PickedList.PushItem( itemWrapper );
|
|
}
|
|
|
|
UndrawAndMarkSegmentsToDrag( m_canvas, aDC );
|
|
}
|
|
|
|
GetBoard()->m_Status_Pcb |= DO_NOT_SHOW_GENERAL_RASTNEST;
|
|
m_canvas->SetMouseCapture( MoveFootprint, Abort_MoveOrCopyModule );
|
|
m_canvas->SetAutoPanRequest( true );
|
|
|
|
// Erase the module.
|
|
if( aDC )
|
|
{
|
|
aModule->SetFlags( DO_NOT_DRAW );
|
|
m_canvas->RefreshDrawingRect( aModule->GetBoundingBox() );
|
|
aModule->ClearFlags( DO_NOT_DRAW );
|
|
}
|
|
|
|
m_canvas->CallMouseCapture( aDC, wxDefaultPosition, false );
|
|
}
|
|
|
|
|
|
/* Called on a move or copy module command abort
|
|
*/
|
|
void Abort_MoveOrCopyModule( EDA_DRAW_PANEL* Panel, wxDC* DC )
|
|
{
|
|
TRACK* pt_segm;
|
|
MODULE* module;
|
|
PCB_EDIT_FRAME* pcbframe = (PCB_EDIT_FRAME*) Panel->GetParent();
|
|
|
|
module = (MODULE*) pcbframe->GetScreen()->GetCurItem();
|
|
pcbframe->GetBoard()->m_Status_Pcb &= ~RATSNEST_ITEM_LOCAL_OK;
|
|
Panel->SetMouseCapture( NULL, NULL );
|
|
|
|
if( module )
|
|
{
|
|
// Erase the current footprint on screen
|
|
module->DrawOutlinesWhenMoving( Panel, DC, g_Offset_Module );
|
|
|
|
/* If a move command: return to old position
|
|
* If a copy command, delete the new footprint
|
|
*/
|
|
if( module->IsMoving() )
|
|
{
|
|
/* Restore old position for dragged tracks */
|
|
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ )
|
|
{
|
|
pt_segm = g_DragSegmentList[ii].m_Track;
|
|
pt_segm->Draw( Panel, DC, GR_XOR );
|
|
pt_segm->SetState( IN_EDIT, false );
|
|
pt_segm->ClearFlags();
|
|
g_DragSegmentList[ii].RestoreInitialValues();
|
|
pt_segm->Draw( Panel, DC, GR_OR );
|
|
}
|
|
|
|
EraseDragList();
|
|
module->ClearFlags( IS_MOVED );
|
|
}
|
|
|
|
if( module->IsNew() ) // Copy command: delete new footprint
|
|
{
|
|
module->DeleteStructure();
|
|
module = NULL;
|
|
pcbframe->GetBoard()->m_Status_Pcb = 0;
|
|
pcbframe->GetBoard()->BuildListOfNets();
|
|
}
|
|
}
|
|
|
|
/* Redraw the module. */
|
|
if( module && s_ModuleInitialCopy )
|
|
{
|
|
if( s_ModuleInitialCopy->GetOrientation() != module->GetOrientation() )
|
|
pcbframe->Rotate_Module( NULL, module, s_ModuleInitialCopy->GetOrientation(), false );
|
|
|
|
if( s_ModuleInitialCopy->GetLayer() != module->GetLayer() )
|
|
pcbframe->Change_Side_Module( module, NULL );
|
|
|
|
module->Draw( Panel, DC, GR_OR );
|
|
}
|
|
|
|
pcbframe->SetCurItem( NULL );
|
|
|
|
delete s_ModuleInitialCopy;
|
|
s_ModuleInitialCopy = NULL;
|
|
s_PickedList.ClearListAndDeleteItems();
|
|
|
|
// Display ratsnest is allowed
|
|
pcbframe->GetBoard()->m_Status_Pcb &= ~DO_NOT_SHOW_GENERAL_RASTNEST;
|
|
|
|
if( pcbframe->GetBoard()->IsElementVisible( LAYER_RATSNEST ) )
|
|
pcbframe->DrawGeneralRatsnest( DC );
|
|
|
|
#ifdef __WXMAC__
|
|
Panel->Refresh();
|
|
#endif
|
|
}
|
|
|
|
|
|
/* Redraw the footprint when moving the mouse.
|
|
*/
|
|
void MoveFootprint( EDA_DRAW_PANEL* aPanel, wxDC* aDC, const wxPoint& aPosition, bool aErase )
|
|
{
|
|
MODULE* module = (MODULE*) aPanel->GetScreen()->GetCurItem();
|
|
|
|
if( module == NULL )
|
|
return;
|
|
|
|
/* Erase current footprint. */
|
|
if( aErase )
|
|
{
|
|
module->DrawOutlinesWhenMoving( aPanel, aDC, g_Offset_Module );
|
|
}
|
|
|
|
/* Redraw the module at the new position. */
|
|
g_Offset_Module = module->GetPosition() - aPanel->GetParent()->GetCrossHairPosition();
|
|
module->DrawOutlinesWhenMoving( aPanel, aDC, g_Offset_Module );
|
|
|
|
DrawSegmentWhileMovingFootprint( aPanel, aDC );
|
|
}
|
|
|
|
|
|
bool PCB_EDIT_FRAME::Delete_Module( MODULE* aModule, wxDC* aDC )
|
|
{
|
|
wxString msg;
|
|
|
|
if( aModule == NULL )
|
|
return false;
|
|
|
|
SetMsgPanel( aModule );
|
|
|
|
/* Remove module from list, and put it in undo command list */
|
|
m_Pcb->Remove( aModule );
|
|
aModule->SetState( IS_DELETED, true );
|
|
SaveCopyInUndoList( aModule, UR_DELETED );
|
|
|
|
if( aDC && GetBoard()->IsElementVisible( LAYER_RATSNEST ) )
|
|
Compile_Ratsnest( aDC, true );
|
|
|
|
// Redraw the full screen to ensure perfect display of board and ratsnest.
|
|
if( aDC )
|
|
m_canvas->Refresh();
|
|
|
|
OnModify();
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
void PCB_EDIT_FRAME::Change_Side_Module( MODULE* Module, wxDC* DC )
|
|
{
|
|
if( Module == NULL )
|
|
return;
|
|
|
|
if( ( Module->GetLayer() != F_Cu ) && ( Module->GetLayer() != B_Cu ) )
|
|
return;
|
|
|
|
OnModify();
|
|
|
|
if( !Module->IsMoving() ) /* This is a simple flip, no other edition in progress */
|
|
{
|
|
|
|
if( DC )
|
|
{
|
|
Module->SetFlags( DO_NOT_DRAW );
|
|
m_canvas->RefreshDrawingRect( Module->GetBoundingBox() );
|
|
Module->ClearFlags( DO_NOT_DRAW );
|
|
}
|
|
|
|
/* Show ratsnest if necessary. */
|
|
if( DC && GetBoard()->IsElementVisible( LAYER_RATSNEST ) )
|
|
DrawGeneralRatsnest( DC );
|
|
|
|
g_Offset_Module.x = 0;
|
|
g_Offset_Module.y = 0;
|
|
}
|
|
else // Module is being moved.
|
|
{
|
|
/* Erase footprint and draw outline if it has been already drawn. */
|
|
if( DC )
|
|
{
|
|
Module->DrawOutlinesWhenMoving( m_canvas, DC, g_Offset_Module );
|
|
DrawSegmentWhileMovingFootprint( m_canvas, DC );
|
|
}
|
|
}
|
|
|
|
/* Flip the module */
|
|
Module->Flip( Module->GetPosition() );
|
|
m_Pcb->GetConnectivity()->Update( Module );
|
|
SetMsgPanel( Module );
|
|
|
|
if( !Module->IsMoving() ) /* Inversion simple */
|
|
{
|
|
if( DC )
|
|
{
|
|
Module->Draw( m_canvas, DC, GR_OR );
|
|
|
|
if( GetBoard()->IsElementVisible( LAYER_RATSNEST ) )
|
|
Compile_Ratsnest( DC, true );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if( DC )
|
|
{
|
|
Module->DrawOutlinesWhenMoving( m_canvas, DC, g_Offset_Module );
|
|
DrawSegmentWhileMovingFootprint( m_canvas, DC );
|
|
}
|
|
|
|
}
|
|
m_Pcb->GetConnectivity()->Update( Module );
|
|
}
|
|
|
|
|
|
void PCB_BASE_FRAME::PlaceModule( MODULE* aModule, wxDC* aDC, bool aRecreateRatsnest )
|
|
{
|
|
wxPoint newpos;
|
|
|
|
if( aModule == 0 )
|
|
return;
|
|
|
|
OnModify();
|
|
|
|
|
|
if( aModule->IsNew() )
|
|
{
|
|
SaveCopyInUndoList( aModule, UR_NEW );
|
|
}
|
|
else if( aModule->IsMoving() )
|
|
{
|
|
ITEM_PICKER picker( aModule, UR_CHANGED );
|
|
picker.SetLink( s_ModuleInitialCopy );
|
|
s_PickedList.PushItem( picker );
|
|
s_ModuleInitialCopy = NULL; // the picker is now owner of s_ModuleInitialCopy.
|
|
}
|
|
|
|
if( s_PickedList.GetCount() )
|
|
{
|
|
SaveCopyInUndoList( s_PickedList, UR_UNSPECIFIED );
|
|
|
|
// Clear list, but DO NOT delete items, because they are owned by the saved undo
|
|
// list and they therefore in use
|
|
s_PickedList.ClearItemsList();
|
|
}
|
|
|
|
auto displ_opts = (PCB_DISPLAY_OPTIONS*)GetDisplayOptions();
|
|
|
|
if( displ_opts->m_Show_Module_Ratsnest && aDC )
|
|
TraceModuleRatsNest( aDC );
|
|
|
|
newpos = GetCrossHairPosition();
|
|
aModule->SetPosition( newpos );
|
|
aModule->ClearFlags();
|
|
|
|
delete s_ModuleInitialCopy;
|
|
s_ModuleInitialCopy = NULL;
|
|
|
|
if( aDC )
|
|
aModule->Draw( m_canvas, aDC, GR_OR );
|
|
|
|
// Redraw dragged track segments, if any
|
|
for( unsigned ii = 0; ii < g_DragSegmentList.size(); ii++ )
|
|
{
|
|
TRACK * track = g_DragSegmentList[ii].m_Track;
|
|
track->SetState( IN_EDIT, false );
|
|
track->ClearFlags();
|
|
|
|
if( aDC )
|
|
track->Draw( m_canvas, aDC, GR_OR );
|
|
}
|
|
|
|
// Delete drag list
|
|
EraseDragList();
|
|
|
|
m_canvas->SetMouseCapture( NULL, NULL );
|
|
|
|
if( aRecreateRatsnest )
|
|
m_Pcb->GetConnectivity()->Update( aModule );
|
|
|
|
if( ( GetBoard()->IsElementVisible( LAYER_RATSNEST ) || displ_opts->m_Show_Module_Ratsnest )
|
|
&& aRecreateRatsnest )
|
|
Compile_Ratsnest( aDC, true );
|
|
|
|
if( aDC )
|
|
m_canvas->Refresh();
|
|
|
|
SetMsgPanel( aModule );
|
|
}
|
|
|
|
|
|
/*
|
|
* Rotate the footprint angle degrees in the direction < 0.
|
|
* If incremental == true, the rotation is made from the last orientation,
|
|
* If the module is placed in the absolute orientation angle.
|
|
* If DC == NULL, the component does not redraw.
|
|
* Otherwise, it erases and redraws turns
|
|
*/
|
|
void PCB_BASE_FRAME::Rotate_Module( wxDC* DC, MODULE* module, double angle, bool incremental )
|
|
{
|
|
if( module == NULL )
|
|
return;
|
|
|
|
OnModify();
|
|
|
|
if( !module->IsMoving() ) /* This is a simple rotation, no other
|
|
* edition in progress */
|
|
{
|
|
if( DC ) // Erase footprint to screen
|
|
{
|
|
module->SetFlags( DO_NOT_DRAW );
|
|
m_canvas->RefreshDrawingRect( module->GetBoundingBox() );
|
|
module->ClearFlags( DO_NOT_DRAW );
|
|
|
|
if( GetBoard()->IsElementVisible( LAYER_RATSNEST ) )
|
|
DrawGeneralRatsnest( DC );
|
|
}
|
|
}
|
|
else
|
|
{
|
|
if( DC )
|
|
{
|
|
module->DrawOutlinesWhenMoving( m_canvas, DC, g_Offset_Module );
|
|
DrawSegmentWhileMovingFootprint( m_canvas, DC );
|
|
}
|
|
}
|
|
|
|
|
|
if( incremental )
|
|
module->SetOrientation( module->GetOrientation() + angle );
|
|
else
|
|
module->SetOrientation( angle );
|
|
|
|
SetMsgPanel( module );
|
|
m_Pcb->GetConnectivity()->Update( module );
|
|
|
|
if( DC )
|
|
{
|
|
if( !module->IsMoving() )
|
|
{
|
|
// not beiing moved: redraw the module and update ratsnest
|
|
module->Draw( m_canvas, DC, GR_OR );
|
|
|
|
if( GetBoard()->IsElementVisible( LAYER_RATSNEST ) )
|
|
Compile_Ratsnest( DC, true );
|
|
}
|
|
else
|
|
{
|
|
// Beiing moved: just redraw it
|
|
module->DrawOutlinesWhenMoving( m_canvas, DC, g_Offset_Module );
|
|
DrawSegmentWhileMovingFootprint( m_canvas, DC );
|
|
}
|
|
|
|
if( module->GetFlags() == 0 ) // module not in edit: redraw full screen
|
|
m_canvas->Refresh();
|
|
}
|
|
}
|