/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 1992-2018 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 #include #include #include #include #include #include #include #include #include #include <3d_viewer/eda_3d_viewer.h> #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include 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; iiGetSelectMenuText( 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 ) { Load_Module_From_BOARD( NULL ); } void FOOTPRINT_EDIT_FRAME::LoadModuleFromLibrary( LIB_ID aFPID) { bool is_last_fp_from_brd = IsCurrentFPFromBoard(); MODULE* module = LoadFootprint( aFPID ); if( !module ) return; if( !Clear_Pcb( true ) ) return; 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(); // Update the bitmap of the ID_MODEDIT_SAVE tool if needed. if( is_last_fp_from_brd ) ReCreateHToolbar(); m_treePane->GetLibTree()->ExpandLibId( aFPID ); m_treePane->GetLibTree()->CenterLibId( aFPID ); m_treePane->GetLibTree()->Refresh(); // update highlighting } 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_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( getTargetFPID(), true ) ) { if( getTargetFPID() == GetLoadedFPID() ) Clear_Pcb( false ); SyncLibraryTree( true ); } break; case ID_MODEDIT_NEW_MODULE: { LIB_ID selected = m_treePane->GetLibTree()->GetSelectedLibId(); MODULE* module = CreateNewModule( wxEmptyString ); if( !module ) break; if( !Clear_Pcb( true ) ) break; 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() ) { if( !HandleUnsavedChanges( this, _( "The current footprint has been modified. Save changes?" ), [&]()->bool { return SaveFootprint( GetBoard()->m_Modules ); } ) ) { 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 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 ) ); 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( getTargetFPID() == GetLoadedFPID() ) { 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( getTargetFPID().GetLibItemName().empty() ) { // Save Library As const wxString& src_libNickname = getTargetFPID().GetLibNickname(); wxString src_libFullName = Prj().PcbFootprintLibs()->GetFullURI( src_libNickname ); if( SaveLibraryAs( src_libFullName ) ) SyncLibraryTree( true ); } else if( getTargetFPID() == GetLoadedFPID() ) { // Save Board Footprint As MODULE* footprint = GetBoard()->m_Modules; if( footprint && SaveFootprintAs( footprint ) ) { m_footprintNameWhenLoaded = footprint->GetFPID().GetLibItemName(); m_toolManager->GetView()->Update( GetBoard()->m_Modules ); GetScreen()->ClrModify(); if( IsGalCanvasActive() && GetGalCanvas() ) GetGalCanvas()->ForceRefresh(); else GetCanvas()->Refresh(); SyncLibraryTree( true ); } } else { // Save Selected Footprint As MODULE* footprint = LoadFootprint( getTargetFPID() ); if( footprint && SaveFootprintAs( footprint ) ) SyncLibraryTree( true ); } m_treePane->GetLibTree()->Refresh(); break; case ID_MODEDIT_REVERT_PART: RevertFootprint(); break; case ID_MODEDIT_CUT_PART: case ID_MODEDIT_COPY_PART: if( getTargetFPID().IsValid() ) { LIB_ID fpID = getTargetFPID(); if( fpID == GetLoadedFPID() ) m_copiedModule.reset( new MODULE( *GetBoard()->m_Modules.GetFirst() ) ); else m_copiedModule.reset( LoadFootprint( fpID ) ); if( id == ID_MODEDIT_CUT_PART ) { if( fpID == GetLoadedFPID() ) Clear_Pcb( false ); DeleteModuleFromLibrary( fpID, false ); } SyncLibraryTree( true ); } break; case ID_MODEDIT_PASTE_PART: if( m_copiedModule && !getTargetFPID().GetLibNickname().empty() ) { wxString newLib = getTargetFPID().GetLibNickname(); MODULE* newModule( m_copiedModule.get() ); wxString newName = newModule->GetFPID().GetLibItemName(); while( Prj().PcbFootprintLibs()->FootprintExists( newLib, newName ) ) newName += _( "_copy" ); newModule->SetFPID( LIB_ID( newLib, newName ) ); saveFootprintInLibrary( newModule, newLib ); SyncLibraryTree( true ); m_treePane->GetLibTree()->SelectLibId( newModule->GetFPID() ); } break; case ID_ADD_FOOTPRINT_TO_BOARD: SaveFootprintToBoard( true ); 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()->SetModify(); Zoom_Automatique( false ); m_canvas->Refresh(); Update3DView(); break; case ID_MODEDIT_EXPORT_PART: if( getTargetFPID() == GetLoadedFPID() ) Export_Module( GetBoard()->m_Modules ); else Export_Module( LoadFootprint( getTargetFPID() ) ); break; case ID_MODEDIT_CREATE_NEW_LIB: { wxFileName fn( CreateNewLibrary() ); wxString name = fn.GetName(); if( !name.IsEmpty() ) { LIB_ID newLib( name, wxEmptyString ); SyncLibraryTree( true ); m_treePane->GetLibTree()->SelectLibId( newLib ); } } break; case ID_MODEDIT_ADD_LIBRARY: AddLibrary(); break; case ID_MODEDIT_SHEET_SET: break; case ID_MODEDIT_EDIT_MODULE: LoadModuleFromLibrary( m_treePane->GetLibTree()->GetSelectedLibId() ); 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 ); editFootprintProperties( (MODULE*) GetScreen()->GetCurItem() ); 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: editFootprintProperties( (MODULE*) GetScreen()->GetCurItem() ); 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( GetScreen()->GetCurItem() ), &dc ); break; case ID_POPUP_PCB_ROTATE_TEXTMODULE: RotateTextModule( static_cast( GetScreen()->GetCurItem() ), &dc ); m_canvas->MoveCursorToCrossHair(); break; case ID_POPUP_PCB_DELETE_TEXTMODULE: SaveCopyInUndoList( GetBoard()->m_Modules, UR_CHANGED ); DeleteTextModule( static_cast( GetScreen()->GetCurItem() ) ); SetCurItem( NULL ); m_canvas->MoveCursorToCrossHair(); break; case ID_POPUP_PCB_MOVE_EDGE: Start_Move_EdgeMod( static_cast( 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() ); 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_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_GRAPHICS_FILE: if( GetBoard()->m_Modules ) { InvokeDialogImportGfxModule( this, GetBoard()->m_Modules ); m_canvas->Refresh(); } break; default: wxLogDebug( wxT( "FOOTPRINT_EDIT_FRAME::Process_Special_Functions error" ) ); break; } } void FOOTPRINT_EDIT_FRAME::editFootprintProperties( MODULE* aModule ) { LIB_ID oldFPID = aModule->GetFPID(); DIALOG_FOOTPRINT_FP_EDITOR dialog( this, aModule ); dialog.ShowModal(); GetScreen()->GetCurItem()->ClearFlags(); updateTitle(); // in case of a name change... } 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( 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& 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; } void FOOTPRINT_EDIT_FRAME::KiwayMailIn( KIWAY_EXPRESS& mail ) { const std::string& payload = mail.GetPayload(); switch( mail.Command() ) { case MAIL_FP_EDIT: if( !payload.empty() ) { wxFileName fpFileName( payload ); wxString libNickname; wxString msg; FP_LIB_TABLE* libTable = Prj().PcbFootprintLibs(); const LIB_TABLE_ROW* libTableRow = libTable->FindRowByURI( fpFileName.GetPath() ); if( !libTableRow ) { msg.Printf( _( "The current configuration does not include the footprint library\n" "\"%s\".\nUse Manage Footprint Libraries to edit the configuration." ), fpFileName.GetPath() ); DisplayErrorMessage( this, _( "Library not found in footprint library table." ), msg ); break; } libNickname = libTableRow->GetNickName(); if( !libTable->HasLibrary( libNickname, true ) ) { msg.Printf( _( "The library with the nickname \"%s\" is not enabled\n" "in the current configuration. Use Manage Footprint Libraries to\n" "edit the configuration." ), libNickname ); DisplayErrorMessage( this, _( "Footprint library not enabled." ), msg ); break; } LIB_ID fpId( libNickname, fpFileName.GetName() ); if( m_treePane ) { m_treePane->GetLibTree()->SelectLibId( fpId ); wxCommandEvent event( COMPONENT_SELECTED ); wxPostEvent( m_treePane, event ); } } break; default: ; } }