1139 lines
34 KiB
C++
1139 lines
34 KiB
C++
/*
|
|
* This program source code file is part of KiCad, a free EDA CAD application.
|
|
*
|
|
* Copyright (C) 1992-2016 KiCad Developers, see AUTHORS.txt for contributors.
|
|
* Copyright (C) 2017 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 footprint_editor_utils.cpp
|
|
*/
|
|
|
|
#include <fctsys.h>
|
|
#include <kiface_i.h>
|
|
#include <kiway.h>
|
|
#include <class_drawpanel.h>
|
|
#include <pcb_draw_panel_gal.h>
|
|
#include <confirm.h>
|
|
#include <gestfich.h>
|
|
#include <pgm_base.h>
|
|
#include <trigo.h>
|
|
#include <3d_viewer/eda_3d_viewer.h>
|
|
#include <kicad_device_context.h>
|
|
#include <macros.h>
|
|
#include <invoke_pcb_dialog.h>
|
|
#include <pcb_layer_widget.h>
|
|
#include <board_commit.h>
|
|
#include <view/view.h>
|
|
#include <class_board.h>
|
|
#include <class_module.h>
|
|
#include <class_edge_mod.h>
|
|
|
|
#include <ratsnest_data.h>
|
|
#include <pcbnew.h>
|
|
#include <protos.h>
|
|
#include <pcbnew_id.h>
|
|
#include <footprint_edit_frame.h>
|
|
#include <footprint_viewer_frame.h>
|
|
#include <footprint_tree_pane.h>
|
|
#include <fp_lib_table.h>
|
|
#include <widgets/lib_tree.h>
|
|
#include <collectors.h>
|
|
#include <tool/tool_manager.h>
|
|
#include <tools/pcb_actions.h>
|
|
|
|
#include <dialog_edit_footprint_for_fp_editor.h>
|
|
#include <dialog_move_exact.h>
|
|
#include <dialog_create_array.h>
|
|
#include <wildcards_and_files_ext.h>
|
|
#include <menus_helpers.h>
|
|
#include <footprint_wizard_frame.h>
|
|
#include <config_params.h>
|
|
|
|
#include <functional>
|
|
using namespace std::placeholders;
|
|
|
|
|
|
// Functions defined in block_module_editor, but used here
|
|
// These 3 functions are used in modedit to rotate, mirror or move the
|
|
// whole footprint so they are called with force_all = true
|
|
void MirrorMarkedItems( MODULE* module, wxPoint offset, bool force_all = false );
|
|
void RotateMarkedItems( MODULE* module, wxPoint offset, bool force_all = false );
|
|
void MoveMarkedItemsExactly( MODULE* module, const wxPoint& centre,
|
|
const wxPoint& translation, double rotation,
|
|
bool force_all = false );
|
|
|
|
|
|
BOARD_ITEM* FOOTPRINT_EDIT_FRAME::ModeditLocateAndDisplay( int aHotKeyCode )
|
|
{
|
|
BOARD_ITEM* item = GetCurItem();
|
|
|
|
if( GetBoard()->m_Modules == NULL )
|
|
return NULL;
|
|
|
|
GENERAL_COLLECTORS_GUIDE guide = GetCollectorsGuide();
|
|
|
|
// Assign to scanList the proper item types desired based on tool type
|
|
// or hotkey that is in play.
|
|
|
|
const KICAD_T* scanList = NULL;
|
|
|
|
if( aHotKeyCode )
|
|
{
|
|
// @todo: add switch here and add calls to PcbGeneralLocateAndDisplay(
|
|
// int aHotKeyCode ) when searching is needed from a hotkey handler
|
|
}
|
|
else
|
|
{
|
|
scanList = GENERAL_COLLECTOR::ModulesAndTheirItems;
|
|
}
|
|
|
|
m_Collector->Collect( GetBoard(), scanList, RefPos( true ), guide );
|
|
|
|
// Remove redundancies: when an item is found, we can remove the module from list
|
|
if( m_Collector->GetCount() > 1 )
|
|
{
|
|
for( int ii = 0; ii < m_Collector->GetCount(); ii++ )
|
|
{
|
|
item = (*m_Collector)[ii];
|
|
|
|
if( item->Type() != PCB_MODULE_T )
|
|
continue;
|
|
|
|
m_Collector->Remove( ii );
|
|
ii--;
|
|
}
|
|
}
|
|
|
|
if( m_Collector->GetCount() <= 1 )
|
|
{
|
|
item = (*m_Collector)[0];
|
|
SetCurItem( item );
|
|
}
|
|
else // we can't figure out which item user wants, do popup menu so user can choose
|
|
{
|
|
wxMenu itemMenu;
|
|
|
|
// Give a title to the selection menu. It also allows one to close the popup menu without any action
|
|
AddMenuItem( &itemMenu, wxID_NONE, _( "Clarify Selection" ),
|
|
KiBitmap( info_xpm ) );
|
|
itemMenu.AppendSeparator();
|
|
|
|
int limit = std::min( MAX_ITEMS_IN_PICKER, m_Collector->GetCount() );
|
|
|
|
for( int ii = 0; ii<limit; ++ii )
|
|
{
|
|
item = (*m_Collector)[ii];
|
|
|
|
wxString text = item->GetSelectMenuText( GetUserUnits() );
|
|
BITMAP_DEF xpm = item->GetMenuImage();
|
|
|
|
AddMenuItem( &itemMenu,
|
|
ID_POPUP_PCB_ITEM_SELECTION_START + ii,
|
|
text,
|
|
KiBitmap( xpm ) );
|
|
}
|
|
|
|
// this menu's handler is void
|
|
// PCB_BASE_FRAME::ProcessItemSelection()
|
|
// and it calls SetCurItem() which in turn calls DisplayInfo() on the
|
|
// item.
|
|
m_canvas->SetAbortRequest( true ); // changed in false if an item is selected
|
|
PopupMenu( &itemMenu ); // m_AbortRequest = false if an item is selected
|
|
|
|
m_canvas->MoveCursorToCrossHair();
|
|
m_canvas->SetIgnoreMouseEvents( false );
|
|
|
|
// The function ProcessItemSelection() has set the current item, return it.
|
|
item = GetCurItem();
|
|
}
|
|
|
|
if( item )
|
|
{
|
|
SetMsgPanel( item );
|
|
}
|
|
|
|
return item;
|
|
}
|
|
|
|
|
|
void FOOTPRINT_EDIT_FRAME::LoadModuleFromBoard( wxCommandEvent& event )
|
|
{
|
|
if( GetScreen()->IsModify() )
|
|
{
|
|
if( !IsOK( this,
|
|
_( "Current footprint changes will be lost and this operation cannot be undone. Continue?" ) ) )
|
|
return;
|
|
}
|
|
|
|
if( ! Load_Module_From_BOARD( NULL ) )
|
|
return;
|
|
|
|
GetScreen()->ClearUndoRedoList();
|
|
GetScreen()->ClrModify();
|
|
|
|
Update3DView();
|
|
}
|
|
|
|
|
|
void FOOTPRINT_EDIT_FRAME::Process_Special_Functions( wxCommandEvent& event )
|
|
{
|
|
int id = event.GetId();
|
|
wxPoint pos;
|
|
|
|
INSTALL_UNBUFFERED_DC( dc, m_canvas );
|
|
|
|
wxGetMousePosition( &pos.x, &pos.y );
|
|
|
|
pos.y += 20;
|
|
|
|
switch( id )
|
|
{
|
|
case wxID_CUT:
|
|
case wxID_COPY:
|
|
case ID_TOOLBARH_PCB_SELECT_LAYER:
|
|
case ID_MODEDIT_PAD_SETTINGS:
|
|
case ID_PCB_USER_GRID_SETUP:
|
|
case ID_POPUP_PCB_ROTATE_TEXTEPCB:
|
|
case ID_POPUP_PCB_EDIT_TEXTEPCB:
|
|
case ID_POPUP_PCB_ROTATE_TEXTMODULE:
|
|
case ID_POPUP_PCB_ROTATE_MODULE_CLOCKWISE:
|
|
case ID_POPUP_PCB_ROTATE_MODULE_COUNTERCLOCKWISE:
|
|
case ID_POPUP_PCB_EDIT_TEXTMODULE:
|
|
case ID_POPUP_PCB_APPLY_PAD_SETTINGS:
|
|
case ID_POPUP_PCB_COPY_PAD_SETTINGS:
|
|
case ID_POPUP_PCB_GLOBAL_IMPORT_PAD_SETTINGS:
|
|
case ID_POPUP_PCB_STOP_CURRENT_DRAWING:
|
|
case ID_POPUP_MODEDIT_EDIT_BODY_ITEM:
|
|
case ID_POPUP_MODEDIT_EDIT_WIDTH_ALL_EDGE:
|
|
case ID_POPUP_MODEDIT_EDIT_LAYER_ALL_EDGE:
|
|
case ID_POPUP_PCB_DELETE_EDGE:
|
|
case ID_POPUP_PCB_DELETE_TEXTMODULE:
|
|
case ID_POPUP_PCB_DELETE_PAD:
|
|
case ID_POPUP_DELETE_BLOCK:
|
|
case ID_POPUP_PLACE_BLOCK:
|
|
case ID_POPUP_ZOOM_BLOCK:
|
|
case ID_POPUP_MIRROR_X_BLOCK:
|
|
case ID_POPUP_ROTATE_BLOCK:
|
|
case ID_POPUP_DUPLICATE_BLOCK:
|
|
break;
|
|
|
|
case ID_POPUP_CANCEL_CURRENT_COMMAND:
|
|
default:
|
|
if( m_canvas->IsMouseCaptured() )
|
|
{
|
|
// for all other commands: stop the move in progress
|
|
m_canvas->CallEndMouseCapture( &dc );
|
|
}
|
|
|
|
if( id != ID_POPUP_CANCEL_CURRENT_COMMAND )
|
|
SetNoToolSelected();
|
|
|
|
break;
|
|
}
|
|
|
|
switch( id )
|
|
{
|
|
case ID_EXIT:
|
|
Close( true );
|
|
break;
|
|
|
|
case ID_OPEN_MODULE_VIEWER:
|
|
{
|
|
FOOTPRINT_VIEWER_FRAME* viewer = (FOOTPRINT_VIEWER_FRAME*) Kiway().Player( FRAME_PCB_MODULE_VIEWER, false );
|
|
|
|
if( !viewer )
|
|
{
|
|
viewer = (FOOTPRINT_VIEWER_FRAME*) Kiway().Player( FRAME_PCB_MODULE_VIEWER, true );
|
|
viewer->Show( true );
|
|
viewer->Zoom_Automatique( false );
|
|
}
|
|
else
|
|
{
|
|
// On Windows, Raise() does not bring the window on screen, when iconized
|
|
if( viewer->IsIconized() )
|
|
viewer->Iconize( false );
|
|
|
|
viewer->Raise();
|
|
|
|
// Raising the window does not set the focus on Linux. This should work on
|
|
// any platform.
|
|
if( wxWindow::FindFocus() != viewer )
|
|
viewer->SetFocus();
|
|
}
|
|
}
|
|
break;
|
|
|
|
case ID_MODEDIT_DELETE_PART:
|
|
if( DeleteModuleFromLibrary( LoadFootprint( getTargetLibId() ) ) )
|
|
{
|
|
if( getTargetLibId() == GetCurrentLibId() )
|
|
Clear_Pcb( false );
|
|
|
|
SyncLibraryTree( true );
|
|
}
|
|
break;
|
|
|
|
case ID_MODEDIT_NEW_MODULE:
|
|
{
|
|
LIB_ID selected = m_treePane->GetLibTree()->GetSelectedLibId();
|
|
|
|
if( GetScreen()->IsModify() && !GetBoard()->IsEmpty() )
|
|
{
|
|
KIDIALOG dlg( this, _( "The current footprint contains unsaved changes." ),
|
|
_( "Confirmation" ), wxOK | wxCANCEL | wxICON_WARNING );
|
|
dlg.SetOKLabel( _( "Discard Changes" ) );
|
|
dlg.DoNotShowCheckbox();
|
|
|
|
if( dlg.ShowModal() == wxID_CANCEL )
|
|
break;
|
|
}
|
|
|
|
MODULE* module = CreateNewModule( wxEmptyString );
|
|
|
|
if( module ) // i.e. if create module command not aborted
|
|
{
|
|
Clear_Pcb( false );
|
|
|
|
SetCrossHairPosition( wxPoint( 0, 0 ) );
|
|
AddModuleToBoard( module );
|
|
|
|
// Initialize data relative to nets and netclasses (for a new
|
|
// module the defaults are used)
|
|
// This is mandatory to handle and draw pads
|
|
GetBoard()->BuildListOfNets();
|
|
module->SetPosition( wxPoint( 0, 0 ) );
|
|
|
|
if( GetBoard()->m_Modules )
|
|
GetBoard()->m_Modules->ClearFlags();
|
|
|
|
Zoom_Automatique( false );
|
|
GetScreen()->SetModify();
|
|
|
|
// If selected from the library tree then go ahead and save it there
|
|
if( !selected.GetLibNickname().empty() )
|
|
{
|
|
LIB_ID fpid = module->GetFPID();
|
|
fpid.SetLibNickname( selected.GetLibNickname() );
|
|
module->SetFPID( fpid );
|
|
SaveFootprint( module );
|
|
GetScreen()->ClrModify();
|
|
}
|
|
}
|
|
|
|
updateView();
|
|
m_canvas->Refresh();
|
|
Update3DView();
|
|
|
|
SyncLibraryTree( false );
|
|
}
|
|
break;
|
|
|
|
case ID_MODEDIT_NEW_MODULE_FROM_WIZARD:
|
|
{
|
|
LIB_ID selected = m_treePane->GetLibTree()->GetSelectedLibId();
|
|
|
|
if( GetScreen()->IsModify() && !GetBoard()->IsEmpty() )
|
|
{
|
|
KIDIALOG dlg( this, _( "The current footprint contains unsaved changes." ),
|
|
_( "Confirmation" ), wxOK | wxCANCEL | wxICON_WARNING );
|
|
dlg.SetOKLabel( _( "Discard Changes" ) );
|
|
dlg.DoNotShowCheckbox();
|
|
|
|
if( dlg.ShowModal() == wxID_CANCEL )
|
|
break;
|
|
}
|
|
|
|
FOOTPRINT_WIZARD_FRAME* wizard = (FOOTPRINT_WIZARD_FRAME*) Kiway().Player(
|
|
FRAME_PCB_FOOTPRINT_WIZARD, true, this );
|
|
|
|
if( wizard->ShowModal( NULL, this ) )
|
|
{
|
|
// Creates the new footprint from python script wizard
|
|
MODULE* module = wizard->GetBuiltFootprint();
|
|
|
|
if( module == NULL ) // i.e. if create module command aborted
|
|
break;
|
|
|
|
Clear_Pcb( false );
|
|
|
|
SetCrossHairPosition( wxPoint( 0, 0 ) );
|
|
|
|
// Add the new object to board
|
|
GetBoard()->Add( module, ADD_APPEND );
|
|
|
|
// Initialize data relative to nets and netclasses (for a new
|
|
// module the defaults are used)
|
|
// This is mandatory to handle and draw pads
|
|
GetBoard()->BuildListOfNets();
|
|
module->SetPosition( wxPoint( 0, 0 ) );
|
|
module->ClearFlags();
|
|
|
|
Zoom_Automatique( false );
|
|
GetScreen()->SetModify();
|
|
|
|
// If selected from the library tree then go ahead and save it there
|
|
if( !selected.GetLibNickname().empty() )
|
|
{
|
|
LIB_ID fpid = module->GetFPID();
|
|
fpid.SetLibNickname( selected.GetLibNickname() );
|
|
module->SetFPID( fpid );
|
|
SaveFootprint( module );
|
|
GetScreen()->ClrModify();
|
|
}
|
|
|
|
updateView();
|
|
m_canvas->Refresh();
|
|
Update3DView();
|
|
|
|
SyncLibraryTree( false );
|
|
}
|
|
|
|
wizard->Destroy();
|
|
}
|
|
break;
|
|
|
|
case ID_MODEDIT_SAVE:
|
|
if( getTargetLibId() == GetCurrentLibId() )
|
|
{
|
|
if( SaveFootprint( GetBoard()->m_Modules ) )
|
|
{
|
|
m_toolManager->GetView()->Update( GetBoard()->m_Modules );
|
|
|
|
if( IsGalCanvasActive() && GetGalCanvas() )
|
|
GetGalCanvas()->ForceRefresh();
|
|
else
|
|
GetCanvas()->Refresh();
|
|
|
|
GetScreen()->ClrModify();
|
|
}
|
|
}
|
|
|
|
m_treePane->GetLibTree()->Refresh();
|
|
break;
|
|
|
|
case ID_MODEDIT_SAVE_AS:
|
|
if( getTargetLibId().GetLibItemName().empty() )
|
|
{
|
|
// Save Library As
|
|
const wxString& libName = getTargetLibId().GetLibNickname();
|
|
if( SaveLibraryAs( Prj().PcbFootprintLibs()->FindRow( libName )->GetFullURI() ) )
|
|
SyncLibraryTree( true );
|
|
}
|
|
else
|
|
{
|
|
// Save Footprint As
|
|
MODULE* footprint = LoadFootprint( getTargetLibId() );
|
|
if( footprint && SaveFootprintAs( footprint ) )
|
|
{
|
|
SyncLibraryTree( false );
|
|
|
|
if( getTargetLibId() == GetCurrentLibId() )
|
|
{
|
|
m_toolManager->GetView()->Update( GetBoard()->m_Modules );
|
|
|
|
if( IsGalCanvasActive() && GetGalCanvas() )
|
|
GetGalCanvas()->ForceRefresh();
|
|
else
|
|
GetCanvas()->Refresh();
|
|
|
|
GetScreen()->ClrModify();
|
|
}
|
|
}
|
|
}
|
|
|
|
m_treePane->GetLibTree()->Refresh();
|
|
break;
|
|
|
|
case ID_MODEDIT_INSERT_MODULE_IN_BOARD:
|
|
case ID_MODEDIT_UPDATE_MODULE_IN_BOARD:
|
|
{
|
|
// update module in the current board,
|
|
// not just add it to the board with total disregard for the netlist...
|
|
PCB_EDIT_FRAME* pcbframe = (PCB_EDIT_FRAME*) Kiway().Player( FRAME_PCB, false );
|
|
|
|
if( pcbframe == NULL ) // happens when the board editor is not active (or closed)
|
|
{
|
|
DisplayErrorMessage( this, _("No board currently open." ) );
|
|
break;
|
|
}
|
|
|
|
BOARD* mainpcb = pcbframe->GetBoard();
|
|
MODULE* source_module = NULL;
|
|
MODULE* module_in_edit = GetBoard()->m_Modules;
|
|
|
|
// Search the old module (source) if exists
|
|
// Because this source could be deleted when editing the main board...
|
|
if( module_in_edit->GetLink() ) // this is not a new module ...
|
|
{
|
|
source_module = mainpcb->m_Modules;
|
|
|
|
for( ; source_module != NULL; source_module = source_module->Next() )
|
|
{
|
|
if( module_in_edit->GetLink() == source_module->GetTimeStamp() )
|
|
break;
|
|
}
|
|
}
|
|
|
|
if( ( source_module == NULL )
|
|
&& ( id == ID_MODEDIT_UPDATE_MODULE_IN_BOARD ) ) // source not found
|
|
{
|
|
wxString msg;
|
|
msg.Printf( _( "Unable to find the footprint source on the main board" ) );
|
|
msg << _( "\nCannot update the footprint" );
|
|
DisplayError( this, msg );
|
|
break;
|
|
}
|
|
|
|
if( ( source_module != NULL )
|
|
&& ( id == ID_MODEDIT_INSERT_MODULE_IN_BOARD ) ) // source not found
|
|
{
|
|
wxString msg;
|
|
msg.Printf( _( "A footprint source was found on the main board" ) );
|
|
msg << _( "\nCannot insert this footprint" );
|
|
DisplayError( this, msg );
|
|
break;
|
|
}
|
|
|
|
m_toolManager->RunAction( PCB_ACTIONS::selectionClear, true );
|
|
pcbframe->GetToolManager()->RunAction( PCB_ACTIONS::selectionClear, true );
|
|
BOARD_COMMIT commit( pcbframe );
|
|
|
|
// Create the "new" module
|
|
MODULE* newmodule = new MODULE( *module_in_edit );
|
|
newmodule->SetParent( mainpcb );
|
|
newmodule->SetLink( 0 );
|
|
|
|
if( source_module ) // this is an update command
|
|
{
|
|
// In the main board,
|
|
// the new module replace the old module (pos, orient, ref, value
|
|
// and connexions are kept)
|
|
// and the source_module (old module) is deleted
|
|
pcbframe->Exchange_Module( source_module, newmodule, commit );
|
|
newmodule->SetTimeStamp( module_in_edit->GetLink() );
|
|
commit.Push( wxT( "Update module" ) );
|
|
}
|
|
else // This is an insert command
|
|
{
|
|
wxPoint cursor_pos = pcbframe->GetCrossHairPosition();
|
|
|
|
commit.Add( newmodule );
|
|
pcbframe->SetCrossHairPosition( wxPoint( 0, 0 ) );
|
|
pcbframe->PlaceModule( newmodule, NULL );
|
|
newmodule->SetPosition( wxPoint( 0, 0 ) );
|
|
pcbframe->SetCrossHairPosition( cursor_pos );
|
|
newmodule->SetTimeStamp( GetNewTimeStamp() );
|
|
commit.Push( wxT( "Insert module" ) );
|
|
}
|
|
|
|
newmodule->ClearFlags();
|
|
GetScreen()->ClrModify();
|
|
pcbframe->SetCurItem( NULL );
|
|
// @todo LEGACY should be unnecessary
|
|
mainpcb->m_Status_Pcb = 0;
|
|
}
|
|
break;
|
|
|
|
case ID_MODEDIT_IMPORT_PART:
|
|
if( ! Clear_Pcb( true ) )
|
|
break; // //this command is aborted
|
|
|
|
SetCrossHairPosition( wxPoint( 0, 0 ) );
|
|
Import_Module();
|
|
|
|
if( GetBoard()->m_Modules )
|
|
GetBoard()->m_Modules->ClearFlags();
|
|
|
|
GetScreen()->ClrModify();
|
|
Zoom_Automatique( false );
|
|
m_canvas->Refresh();
|
|
Update3DView();
|
|
|
|
break;
|
|
|
|
case ID_MODEDIT_EXPORT_PART:
|
|
Export_Module( LoadFootprint( getTargetLibId() ) );
|
|
break;
|
|
|
|
case ID_MODEDIT_CREATE_NEW_LIB:
|
|
{
|
|
wxFileName fn( CreateNewLibrary() );
|
|
wxString name = fn.GetName();
|
|
|
|
if( !name.IsEmpty() )
|
|
{
|
|
LIB_ID newLib( name, wxEmptyString );
|
|
|
|
SyncLibraryTree( false );
|
|
m_treePane->GetLibTree()->SelectLibId( newLib );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case ID_MODEDIT_SHEET_SET:
|
|
break;
|
|
|
|
case ID_MODEDIT_EDIT_MODULE:
|
|
{
|
|
if( GetScreen()->IsModify() && !GetBoard()->IsEmpty() )
|
|
{
|
|
KIDIALOG dlg( this, _( "The current footprint contains unsaved changes." ),
|
|
_( "Confirmation" ), wxOK | wxCANCEL | wxICON_WARNING );
|
|
dlg.SetOKLabel( _( "Discard Changes" ) );
|
|
dlg.DoNotShowCheckbox();
|
|
|
|
if( dlg.ShowModal() == wxID_CANCEL )
|
|
break;
|
|
}
|
|
|
|
LIB_ID partId = m_treePane->GetLibTree()->GetSelectedLibId();
|
|
|
|
MODULE* module = LoadFootprint( partId );
|
|
|
|
if( !module )
|
|
break;
|
|
|
|
Clear_Pcb( false );
|
|
|
|
SetCrossHairPosition( wxPoint( 0, 0 ) );
|
|
AddModuleToBoard( module );
|
|
|
|
if( GetBoard()->m_Modules )
|
|
{
|
|
GetBoard()->m_Modules->ClearFlags();
|
|
|
|
// if either m_Reference or m_Value are gone, reinstall them -
|
|
// otherwise you cannot see what you are doing on board
|
|
TEXTE_MODULE* ref = &GetBoard()->m_Modules->Reference();
|
|
TEXTE_MODULE* val = &GetBoard()->m_Modules->Value();
|
|
|
|
if( val && ref )
|
|
{
|
|
ref->SetType( TEXTE_MODULE::TEXT_is_REFERENCE ); // just in case ...
|
|
|
|
if( ref->GetLength() == 0 )
|
|
ref->SetText( wxT( "Ref**" ) );
|
|
|
|
val->SetType( TEXTE_MODULE::TEXT_is_VALUE ); // just in case ...
|
|
|
|
if( val->GetLength() == 0 )
|
|
val->SetText( wxT( "Val**" ) );
|
|
}
|
|
}
|
|
|
|
Zoom_Automatique( false );
|
|
|
|
Update3DView();
|
|
|
|
GetScreen()->ClrModify();
|
|
|
|
updateView();
|
|
m_canvas->Refresh();
|
|
m_treePane->GetLibTree()->Refresh();
|
|
}
|
|
break;
|
|
|
|
case ID_MODEDIT_PAD_SETTINGS:
|
|
InstallPadOptionsFrame( NULL );
|
|
break;
|
|
|
|
case ID_MODEDIT_CHECK:
|
|
// Currently: not implemented
|
|
break;
|
|
|
|
case ID_MODEDIT_EDIT_MODULE_PROPERTIES:
|
|
if( GetBoard()->m_Modules )
|
|
{
|
|
SetCurItem( GetBoard()->m_Modules );
|
|
|
|
DIALOG_FOOTPRINT_FP_EDITOR dialog( this, (MODULE*) GetScreen()->GetCurItem() );
|
|
|
|
dialog.ShowModal();
|
|
GetScreen()->GetCurItem()->ClearFlags();
|
|
|
|
m_canvas->Refresh();
|
|
}
|
|
break;
|
|
|
|
case ID_POPUP_CLOSE_CURRENT_TOOL:
|
|
break;
|
|
|
|
case ID_POPUP_CANCEL_CURRENT_COMMAND:
|
|
break;
|
|
|
|
case ID_POPUP_PCB_ROTATE_MODULE_COUNTERCLOCKWISE:
|
|
m_canvas->MoveCursorToCrossHair();
|
|
Rotate_Module( NULL, (MODULE*) GetScreen()->GetCurItem(), 900, true );
|
|
m_canvas->Refresh();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_ROTATE_MODULE_CLOCKWISE:
|
|
m_canvas->MoveCursorToCrossHair();
|
|
Rotate_Module( NULL, (MODULE*) GetScreen()->GetCurItem(), -900, true );
|
|
m_canvas->Refresh();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_EDIT_MODULE_PRMS:
|
|
{
|
|
DIALOG_FOOTPRINT_FP_EDITOR dialog( this, (MODULE*) GetScreen()->GetCurItem() );
|
|
dialog.ShowModal();
|
|
GetScreen()->GetCurItem()->ClearFlags();
|
|
m_canvas->MoveCursorToCrossHair();
|
|
m_canvas->Refresh();
|
|
}
|
|
break;
|
|
|
|
case ID_POPUP_PCB_MOVE_PAD_REQUEST:
|
|
m_canvas->MoveCursorToCrossHair();
|
|
StartMovePad( (D_PAD*) GetScreen()->GetCurItem(), &dc, false );
|
|
break;
|
|
|
|
case ID_POPUP_PCB_EDIT_PAD:
|
|
InstallPadOptionsFrame( (D_PAD*) GetScreen()->GetCurItem() );
|
|
m_canvas->MoveCursorToCrossHair();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_DELETE_PAD:
|
|
SaveCopyInUndoList( GetBoard()->m_Modules, UR_CHANGED );
|
|
DeletePad( (D_PAD*) GetScreen()->GetCurItem(), false );
|
|
SetCurItem( NULL );
|
|
m_canvas->MoveCursorToCrossHair();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_DUPLICATE_ITEM:
|
|
duplicateItems( false );
|
|
break;
|
|
|
|
case ID_POPUP_PCB_DUPLICATE_ITEM_AND_INCREMENT:
|
|
duplicateItems( true );
|
|
break;
|
|
|
|
case ID_POPUP_PCB_MOVE_EXACT:
|
|
moveExact();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_CREATE_ARRAY:
|
|
createArray();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_APPLY_PAD_SETTINGS:
|
|
SaveCopyInUndoList( GetBoard()->m_Modules, UR_CHANGED );
|
|
m_canvas->MoveCursorToCrossHair();
|
|
Import_Pad_Settings( (D_PAD*) GetScreen()->GetCurItem(), true );
|
|
break;
|
|
|
|
case ID_POPUP_PCB_GLOBAL_IMPORT_PAD_SETTINGS:
|
|
SaveCopyInUndoList( GetBoard()->m_Modules, UR_CHANGED );
|
|
// Calls the global change dialog:
|
|
PushPadProperties((D_PAD*) GetScreen()->GetCurItem());
|
|
m_canvas->MoveCursorToCrossHair();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_COPY_PAD_SETTINGS:
|
|
m_canvas->MoveCursorToCrossHair();
|
|
Export_Pad_Settings( (D_PAD*) GetScreen()->GetCurItem() );
|
|
break;
|
|
|
|
case ID_POPUP_PCB_EDIT_TEXTMODULE:
|
|
InstallTextOptionsFrame( GetScreen()->GetCurItem(), &dc );
|
|
break;
|
|
|
|
case ID_POPUP_PCB_MOVE_TEXTMODULE_REQUEST:
|
|
m_canvas->MoveCursorToCrossHair();
|
|
StartMoveTexteModule( static_cast<TEXTE_MODULE*>( GetScreen()->GetCurItem() ), &dc );
|
|
break;
|
|
|
|
case ID_POPUP_PCB_ROTATE_TEXTMODULE:
|
|
RotateTextModule( static_cast<TEXTE_MODULE*>( GetScreen()->GetCurItem() ), &dc );
|
|
m_canvas->MoveCursorToCrossHair();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_DELETE_TEXTMODULE:
|
|
SaveCopyInUndoList( GetBoard()->m_Modules, UR_CHANGED );
|
|
DeleteTextModule( static_cast<TEXTE_MODULE*>( GetScreen()->GetCurItem() ) );
|
|
SetCurItem( NULL );
|
|
m_canvas->MoveCursorToCrossHair();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_MOVE_EDGE:
|
|
Start_Move_EdgeMod( static_cast<EDGE_MODULE*>( GetScreen()->GetCurItem() ), &dc );
|
|
m_canvas->MoveCursorToCrossHair();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_STOP_CURRENT_DRAWING:
|
|
m_canvas->MoveCursorToCrossHair();
|
|
|
|
if( GetScreen()->GetCurItem()->IsNew() )
|
|
{
|
|
End_Edge_Module( (EDGE_MODULE*) GetScreen()->GetCurItem() );
|
|
SetCurItem( NULL );
|
|
}
|
|
break;
|
|
|
|
case ID_POPUP_MODEDIT_EDIT_BODY_ITEM :
|
|
InstallGraphicItemPropertiesDialog( GetScreen()->GetCurItem(), nullptr );
|
|
break;
|
|
|
|
case ID_POPUP_MODEDIT_EDIT_WIDTH_ALL_EDGE:
|
|
m_canvas->MoveCursorToCrossHair();
|
|
Edit_Edge_Width( NULL );
|
|
m_canvas->Refresh();
|
|
break;
|
|
|
|
case ID_POPUP_MODEDIT_EDIT_LAYER_ALL_EDGE:
|
|
m_canvas->MoveCursorToCrossHair();
|
|
Edit_Edge_Layer( NULL );
|
|
m_canvas->Refresh();
|
|
break;
|
|
|
|
case ID_POPUP_PCB_DELETE_EDGE:
|
|
SaveCopyInUndoList( GetBoard()->m_Modules, UR_CHANGED );
|
|
m_canvas->MoveCursorToCrossHair();
|
|
RemoveStruct( GetScreen()->GetCurItem() );
|
|
SetCurItem( NULL );
|
|
break;
|
|
|
|
case ID_MODEDIT_MODULE_ROTATE:
|
|
case ID_MODEDIT_MODULE_MIRROR:
|
|
case ID_MODEDIT_MODULE_MOVE_EXACT:
|
|
SaveCopyInUndoList( GetBoard()->m_Modules, UR_CHANGED );
|
|
Transform( (MODULE*) GetScreen()->GetCurItem(), id );
|
|
m_canvas->Refresh();
|
|
break;
|
|
|
|
case ID_PCB_USER_GRID_SETUP:
|
|
InvokeDialogGrid();
|
|
break;
|
|
|
|
case ID_POPUP_PLACE_BLOCK:
|
|
GetScreen()->m_BlockLocate.SetCommand( BLOCK_MOVE );
|
|
m_canvas->SetAutoPanRequest( false );
|
|
HandleBlockPlace( &dc );
|
|
break;
|
|
|
|
case ID_POPUP_DUPLICATE_BLOCK:
|
|
GetScreen()->m_BlockLocate.SetCommand( BLOCK_DUPLICATE );
|
|
GetScreen()->m_BlockLocate.SetMessageBlock( this );
|
|
m_canvas->SetAutoPanRequest( false );
|
|
HandleBlockPlace( &dc );
|
|
break;
|
|
|
|
case ID_POPUP_ZOOM_BLOCK:
|
|
GetScreen()->m_BlockLocate.SetCommand( BLOCK_ZOOM );
|
|
GetScreen()->m_BlockLocate.SetMessageBlock( this );
|
|
HandleBlockEnd( &dc );
|
|
break;
|
|
|
|
case ID_POPUP_DELETE_BLOCK:
|
|
GetScreen()->m_BlockLocate.SetCommand( BLOCK_DELETE );
|
|
GetScreen()->m_BlockLocate.SetMessageBlock( this );
|
|
HandleBlockEnd( &dc );
|
|
break;
|
|
|
|
case ID_POPUP_ROTATE_BLOCK:
|
|
GetScreen()->m_BlockLocate.SetCommand( BLOCK_ROTATE );
|
|
GetScreen()->m_BlockLocate.SetMessageBlock( this );
|
|
HandleBlockEnd( &dc );
|
|
break;
|
|
|
|
case ID_POPUP_MIRROR_X_BLOCK:
|
|
GetScreen()->m_BlockLocate.SetCommand( BLOCK_MIRROR_X );
|
|
GetScreen()->m_BlockLocate.SetMessageBlock( this );
|
|
HandleBlockEnd( &dc );
|
|
break;
|
|
|
|
case ID_POPUP_MOVE_BLOCK_EXACT:
|
|
GetScreen()->m_BlockLocate.SetCommand( BLOCK_MOVE_EXACT );
|
|
GetScreen()->m_BlockLocate.SetMessageBlock( this );
|
|
HandleBlockEnd( &dc );
|
|
break;
|
|
|
|
case ID_GEN_IMPORT_DXF_FILE:
|
|
if( GetBoard()->m_Modules )
|
|
{
|
|
InvokeDXFDialogModuleImport( this, GetBoard()->m_Modules );
|
|
m_canvas->Refresh();
|
|
}
|
|
break;
|
|
|
|
default:
|
|
DisplayError( this,
|
|
wxT( "FOOTPRINT_EDIT_FRAME::Process_Special_Functions error" ) );
|
|
break;
|
|
}
|
|
}
|
|
|
|
|
|
void FOOTPRINT_EDIT_FRAME::moveExact()
|
|
{
|
|
wxPoint translation;
|
|
double rotation;
|
|
ROTATION_ANCHOR rotationAnchor = ROTATE_AROUND_ITEM_ANCHOR;
|
|
|
|
DIALOG_MOVE_EXACT dialog( this, translation, rotation, rotationAnchor );
|
|
int ret = dialog.ShowModal();
|
|
|
|
if( ret == wxID_OK )
|
|
{
|
|
SaveCopyInUndoList( GetBoard()->m_Modules, UR_CHANGED );
|
|
|
|
BOARD_ITEM* item = GetScreen()->GetCurItem();
|
|
|
|
item->Move( translation );
|
|
|
|
switch( rotationAnchor )
|
|
{
|
|
case ROTATE_AROUND_ITEM_ANCHOR:
|
|
item->Rotate( item->GetPosition(), rotation );
|
|
break;
|
|
case ROTATE_AROUND_USER_ORIGIN:
|
|
item->Rotate( GetScreen()->m_O_Curseur, rotation );
|
|
break;
|
|
default:
|
|
wxFAIL_MSG( "Rotation choice shouldn't have been available in this context." );
|
|
}
|
|
|
|
|
|
item->Rotate( item->GetPosition(), rotation );
|
|
m_canvas->Refresh();
|
|
}
|
|
|
|
m_canvas->MoveCursorToCrossHair();
|
|
}
|
|
|
|
|
|
void FOOTPRINT_EDIT_FRAME::duplicateItems( bool aIncrement )
|
|
{
|
|
BOARD_ITEM* item = GetScreen()->GetCurItem();
|
|
|
|
PCB_BASE_EDIT_FRAME::duplicateItem( item, aIncrement );
|
|
}
|
|
|
|
|
|
void FOOTPRINT_EDIT_FRAME::Transform( MODULE* module, int transform )
|
|
{
|
|
switch( transform )
|
|
{
|
|
case ID_MODEDIT_MODULE_ROTATE:
|
|
RotateMarkedItems( module, wxPoint(0,0), true );
|
|
break;
|
|
|
|
case ID_MODEDIT_MODULE_MIRROR:
|
|
MirrorMarkedItems( module, wxPoint(0,0), true );
|
|
break;
|
|
|
|
case ID_MODEDIT_MODULE_MOVE_EXACT:
|
|
{
|
|
wxPoint translation;
|
|
double rotation;
|
|
ROTATION_ANCHOR rotationAnchor = ROTATE_AROUND_ITEM_ANCHOR;
|
|
|
|
DIALOG_MOVE_EXACT dialog( this, translation, rotation, rotationAnchor );
|
|
|
|
if( dialog.ShowModal() == wxID_OK )
|
|
{
|
|
switch( rotationAnchor )
|
|
{
|
|
case ROTATE_AROUND_ITEM_ANCHOR:
|
|
MoveMarkedItemsExactly( module, module->GetPosition() + translation, translation, rotation, true );
|
|
break;
|
|
case ROTATE_AROUND_USER_ORIGIN:
|
|
MoveMarkedItemsExactly( module, GetScreen()->m_O_Curseur, translation, rotation, true );
|
|
break;
|
|
default:
|
|
wxFAIL_MSG( "Rotation choice shouldn't have been available in this context." );
|
|
}
|
|
}
|
|
|
|
break;
|
|
}
|
|
|
|
default:
|
|
DisplayInfoMessage( this, wxT( "Not available" ) );
|
|
break;
|
|
}
|
|
|
|
module->CalculateBoundingBox();
|
|
OnModify();
|
|
}
|
|
|
|
|
|
void FOOTPRINT_EDIT_FRAME::OnVerticalToolbar( wxCommandEvent& aEvent )
|
|
{
|
|
int id = aEvent.GetId();
|
|
int lastToolID = GetToolId();
|
|
|
|
// Stop the current command and deselect the current tool.
|
|
SetNoToolSelected();
|
|
|
|
switch( id )
|
|
{
|
|
case ID_NO_TOOL_SELECTED:
|
|
break;
|
|
|
|
case ID_ZOOM_SELECTION:
|
|
// This tool is located on the main toolbar: switch it on or off on click on it
|
|
if( lastToolID != ID_ZOOM_SELECTION )
|
|
SetToolID( ID_ZOOM_SELECTION, wxCURSOR_MAGNIFIER, _( "Zoom to selection" ) );
|
|
else
|
|
SetNoToolSelected();
|
|
break;
|
|
|
|
case ID_MODEDIT_LINE_TOOL:
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Add line" ) );
|
|
break;
|
|
|
|
case ID_MODEDIT_ARC_TOOL:
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Add arc" ) );
|
|
break;
|
|
|
|
case ID_MODEDIT_CIRCLE_TOOL:
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Add circle" ) );
|
|
break;
|
|
|
|
case ID_MODEDIT_TEXT_TOOL:
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Add text" ) );
|
|
break;
|
|
|
|
case ID_MODEDIT_ANCHOR_TOOL:
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Place anchor" ) );
|
|
break;
|
|
|
|
case ID_MODEDIT_PLACE_GRID_COORD:
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Set grid origin" ) );
|
|
break;
|
|
|
|
case ID_MODEDIT_PAD_TOOL:
|
|
if( GetBoard()->m_Modules )
|
|
{
|
|
SetToolID( id, wxCURSOR_PENCIL, _( "Add pad" ) );
|
|
}
|
|
else
|
|
{
|
|
SetToolID( id, wxCURSOR_ARROW, _( "Pad properties" ) );
|
|
InstallPadOptionsFrame( NULL );
|
|
SetNoToolSelected();
|
|
}
|
|
break;
|
|
|
|
case ID_MODEDIT_DELETE_TOOL:
|
|
SetToolID( id, wxCURSOR_BULLSEYE, _( "Delete item" ) );
|
|
break;
|
|
|
|
case ID_MODEDIT_MEASUREMENT_TOOL:
|
|
DisplayError( this, wxT( "Measurement Tool not available in Legacy Toolset" ) );
|
|
SetNoToolSelected();
|
|
break;
|
|
|
|
default:
|
|
wxFAIL_MSG( wxT( "Unknown command id." ) );
|
|
SetNoToolSelected();
|
|
}
|
|
}
|
|
|
|
|
|
void FOOTPRINT_EDIT_FRAME::RemoveStruct( EDA_ITEM* Item )
|
|
{
|
|
if( Item == NULL )
|
|
return;
|
|
|
|
switch( Item->Type() )
|
|
{
|
|
case PCB_PAD_T:
|
|
DeletePad( (D_PAD*) Item, false );
|
|
break;
|
|
|
|
case PCB_MODULE_TEXT_T:
|
|
{
|
|
TEXTE_MODULE* text = static_cast<TEXTE_MODULE*>( Item );
|
|
|
|
switch( text->GetType() )
|
|
{
|
|
case TEXTE_MODULE::TEXT_is_REFERENCE:
|
|
DisplayError( this, _( "Cannot delete REFERENCE!" ) );
|
|
break;
|
|
|
|
case TEXTE_MODULE::TEXT_is_VALUE:
|
|
DisplayError( this, _( "Cannot delete VALUE!" ) );
|
|
break;
|
|
|
|
case TEXTE_MODULE::TEXT_is_DIVERS:
|
|
DeleteTextModule( text );
|
|
}
|
|
}
|
|
break;
|
|
|
|
case PCB_MODULE_EDGE_T:
|
|
Delete_Edge_Module( (EDGE_MODULE*) Item );
|
|
m_canvas->Refresh();
|
|
break;
|
|
|
|
case PCB_MODULE_T:
|
|
break;
|
|
|
|
default:
|
|
{
|
|
wxString Line;
|
|
Line.Printf( wxT( " RemoveStruct: item type %d unknown." ), Item->Type() );
|
|
wxMessageBox( Line );
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
|
|
COLOR4D FOOTPRINT_EDIT_FRAME::GetGridColor()
|
|
{
|
|
return Settings().Colors().GetItemColor( LAYER_GRID );
|
|
}
|
|
|
|
|
|
void FOOTPRINT_EDIT_FRAME::SetActiveLayer( PCB_LAYER_ID aLayer )
|
|
{
|
|
PCB_BASE_FRAME::SetActiveLayer( aLayer );
|
|
|
|
m_Layers->SelectLayer( GetActiveLayer() );
|
|
m_Layers->OnLayerSelected();
|
|
|
|
if( IsGalCanvasActive() )
|
|
{
|
|
GetGalCanvas()->SetHighContrastLayer( aLayer );
|
|
GetGalCanvas()->Refresh();
|
|
}
|
|
}
|
|
|
|
bool FOOTPRINT_EDIT_FRAME::OpenProjectFiles( const std::vector<wxString>& aFileSet, int aCtl )
|
|
{
|
|
if( ! Clear_Pcb( true ) )
|
|
return false; // //this command is aborted
|
|
|
|
SetCrossHairPosition( wxPoint( 0, 0 ) );
|
|
Import_Module( aFileSet[0] );
|
|
|
|
if( GetBoard()->m_Modules )
|
|
GetBoard()->m_Modules->ClearFlags();
|
|
|
|
GetScreen()->ClrModify();
|
|
Zoom_Automatique( false );
|
|
m_canvas->Refresh();
|
|
|
|
return true;
|
|
}
|