/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2014-2016 CERN * Copyright (C) 2019-2021 KiCad Developers, see AUTHORS.txt for contributors. * @author Maciej Suminski <maciej.suminski@cern.ch> * * 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 <tools/edit_tool.h> #include <router/router_tool.h> #include <pgm_base.h> #include <tools/pcb_actions.h> #include <tools/pcb_control.h> #include <tools/pcb_picker_tool.h> #include <tools/pcb_selection_tool.h> #include <tools/board_reannotate_tool.h> #include <3d_viewer/eda_3d_viewer_frame.h> #include <bitmaps.h> #include <board_commit.h> #include <board.h> #include <board_design_settings.h> #include <board_item.h> #include <dialogs/dialog_paste_special.h> #include <pcb_dimension.h> #include <footprint.h> #include <pcb_group.h> #include <pcb_track.h> #include <zone.h> #include <fp_shape.h> #include <confirm.h> #include <connectivity/connectivity_data.h> #include <core/kicad_algo.h> #include <kicad_clipboard.h> #include <origin_viewitem.h> #include <pcb_edit_frame.h> #include <pcb_painter.h> #include <properties.h> #include <settings/color_settings.h> #include <tool/tool_manager.h> #include <footprint_viewer_frame.h> #include <footprint_edit_frame.h> #include <widgets/wx_progress_reporters.h> #include <widgets/infobar.h> #include <wx/hyperlink.h> using namespace std::placeholders; // files.cpp extern bool AskLoadBoardFileName( PCB_EDIT_FRAME* aParent, int* aCtl, wxString* aFileName, bool aKicadFilesOnly = false ); extern IO_MGR::PCB_FILE_T plugin_type( const wxString& aFileName, int aCtl ); PCB_CONTROL::PCB_CONTROL() : PCB_TOOL_BASE( "pcbnew.Control" ), m_frame( nullptr ), m_pickerItem( nullptr ) { m_gridOrigin.reset( new KIGFX::ORIGIN_VIEWITEM() ); } PCB_CONTROL::~PCB_CONTROL() { } void PCB_CONTROL::Reset( RESET_REASON aReason ) { m_frame = getEditFrame<PCB_BASE_FRAME>(); if( aReason == MODEL_RELOAD || aReason == GAL_SWITCH ) { m_gridOrigin->SetPosition( board()->GetDesignSettings().GetGridOrigin() ); m_gridOrigin->SetColor( m_frame->GetGridColor() ); getView()->Remove( m_gridOrigin.get() ); getView()->Add( m_gridOrigin.get() ); } } int PCB_CONTROL::AddLibrary( const TOOL_EVENT& aEvent ) { if( m_frame->IsType( FRAME_FOOTPRINT_EDITOR ) || m_frame->IsType( FRAME_PCB_EDITOR ) ) { if( aEvent.IsAction( &ACTIONS::newLibrary ) ) static_cast<PCB_BASE_EDIT_FRAME*>( m_frame )->CreateNewLibrary(); else if( aEvent.IsAction( &ACTIONS::addLibrary ) ) static_cast<PCB_BASE_EDIT_FRAME*>( m_frame )->AddLibrary(); } return 0; } int PCB_CONTROL::Quit( const TOOL_EVENT& aEvent ) { m_frame->Close( false ); return 0; } template<class T> void Flip( T& aValue ) { aValue = !aValue; } int PCB_CONTROL::TrackDisplayMode( const TOOL_EVENT& aEvent ) { PCB_DISPLAY_OPTIONS opts = displayOptions(); Flip( opts.m_DisplayPcbTrackFill ); m_frame->SetDisplayOptions( opts ); for( PCB_TRACK* track : board()->Tracks() ) { if( track->Type() == PCB_TRACE_T || track->Type() == PCB_ARC_T ) view()->Update( track, KIGFX::REPAINT ); } canvas()->Refresh(); return 0; } int PCB_CONTROL::ToggleRatsnest( const TOOL_EVENT& aEvent ) { PCB_DISPLAY_OPTIONS opts = displayOptions(); if( aEvent.IsAction( &PCB_ACTIONS::showRatsnest ) ) { // N.B. Do not disable the Ratsnest layer here. We use it for local ratsnest Flip( opts.m_ShowGlobalRatsnest ); m_frame->SetDisplayOptions( opts ); getEditFrame<PCB_EDIT_FRAME>()->SetElementVisibility( LAYER_RATSNEST, opts.m_ShowGlobalRatsnest ); } else if( aEvent.IsAction( &PCB_ACTIONS::ratsnestLineMode ) ) { Flip( opts.m_DisplayRatsnestLinesCurved ); m_frame->SetDisplayOptions( opts ); } canvas()->RedrawRatsnest(); canvas()->Refresh(); return 0; } int PCB_CONTROL::ViaDisplayMode( const TOOL_EVENT& aEvent ) { PCB_DISPLAY_OPTIONS opts = displayOptions(); Flip( opts.m_DisplayViaFill ); m_frame->SetDisplayOptions( opts ); for( PCB_TRACK* track : board()->Tracks() ) { if( track->Type() == PCB_TRACE_T || track->Type() == PCB_VIA_T ) view()->Update( track, KIGFX::REPAINT ); } canvas()->Refresh(); return 0; } /** * We have bug reports indicating that some new users confuse zone filling/unfilling with the * display modes. This will put up a warning if they show zone fills when one or more zones * are unfilled. */ void PCB_CONTROL::unfilledZoneCheck() { if( Pgm().GetCommonSettings()->m_DoNotShowAgain.zone_fill_warning ) return; bool unfilledZones = false; for( const ZONE* zone : board()->Zones() ) { if( !zone->GetIsRuleArea() && !zone->IsFilled() ) { unfilledZones = true; break; } } if( unfilledZones ) { WX_INFOBAR* infobar = frame()->GetInfoBar(); wxHyperlinkCtrl* button = new wxHyperlinkCtrl( infobar, wxID_ANY, _( "Don't show again" ), wxEmptyString ); button->Bind( wxEVT_COMMAND_HYPERLINK, std::function<void( wxHyperlinkEvent& aEvent )>( [&]( wxHyperlinkEvent& aEvent ) { Pgm().GetCommonSettings()->m_DoNotShowAgain.zone_fill_warning = true; frame()->GetInfoBar()->Dismiss(); } ) ); infobar->RemoveAllButtons(); infobar->AddButton( button ); wxString msg; msg.Printf( _( "Not all zones are filled. Use Edit > Fill All Zones (%s) " "if you wish to see all fills." ), KeyNameFromKeyCode( PCB_ACTIONS::zoneFillAll.GetHotKey() ) ); infobar->ShowMessageFor( msg, 5000, wxICON_WARNING ); } } int PCB_CONTROL::ZoneDisplayMode( const TOOL_EVENT& aEvent ) { PCB_DISPLAY_OPTIONS opts = displayOptions(); // Apply new display options to the GAL canvas if( aEvent.IsAction( &PCB_ACTIONS::zoneDisplayFilled ) ) { unfilledZoneCheck(); opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_FILLED; } else if( aEvent.IsAction( &PCB_ACTIONS::zoneDisplayOutline ) ) { opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_ZONE_OUTLINE; } else if( aEvent.IsAction( &PCB_ACTIONS::zoneDisplayFractured ) ) { opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_FRACTURE_BORDERS; } else if( aEvent.IsAction( &PCB_ACTIONS::zoneDisplayTriangulated ) ) { opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_TRIANGULATION; } else if( aEvent.IsAction( &PCB_ACTIONS::zoneDisplayToggle ) ) { if( opts.m_ZoneDisplayMode == ZONE_DISPLAY_MODE::SHOW_FILLED ) opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_ZONE_OUTLINE; else opts.m_ZoneDisplayMode = ZONE_DISPLAY_MODE::SHOW_FILLED; } else { wxFAIL; } m_frame->SetDisplayOptions( opts ); for( ZONE* zone : board()->Zones() ) view()->Update( zone, KIGFX::REPAINT ); canvas()->Refresh(); return 0; } int PCB_CONTROL::HighContrastMode( const TOOL_EVENT& aEvent ) { PCB_DISPLAY_OPTIONS opts = displayOptions(); opts.m_ContrastModeDisplay = opts.m_ContrastModeDisplay == HIGH_CONTRAST_MODE::NORMAL ? HIGH_CONTRAST_MODE::DIMMED : HIGH_CONTRAST_MODE::NORMAL; m_frame->SetDisplayOptions( opts ); return 0; } int PCB_CONTROL::HighContrastModeCycle( const TOOL_EVENT& aEvent ) { PCB_DISPLAY_OPTIONS opts = displayOptions(); switch( opts.m_ContrastModeDisplay ) { case HIGH_CONTRAST_MODE::NORMAL: opts.m_ContrastModeDisplay = HIGH_CONTRAST_MODE::DIMMED; break; case HIGH_CONTRAST_MODE::DIMMED: opts.m_ContrastModeDisplay = HIGH_CONTRAST_MODE::HIDDEN; break; case HIGH_CONTRAST_MODE::HIDDEN: opts.m_ContrastModeDisplay = HIGH_CONTRAST_MODE::NORMAL; break; } m_frame->SetDisplayOptions( opts ); return 0; } int PCB_CONTROL::NetColorModeCycle( const TOOL_EVENT& aEvent ) { PCB_DISPLAY_OPTIONS opts = displayOptions(); switch( opts.m_NetColorMode ) { case NET_COLOR_MODE::ALL: opts.m_NetColorMode = NET_COLOR_MODE::RATSNEST; break; case NET_COLOR_MODE::RATSNEST: opts.m_NetColorMode = NET_COLOR_MODE::OFF; break; case NET_COLOR_MODE::OFF: opts.m_NetColorMode = NET_COLOR_MODE::ALL; break; } m_frame->SetDisplayOptions( opts ); return 0; } int PCB_CONTROL::RatsnestModeCycle( const TOOL_EVENT& aEvent ) { PCB_DISPLAY_OPTIONS opts = displayOptions(); if( !opts.m_ShowGlobalRatsnest ) { opts.m_ShowGlobalRatsnest = true; opts.m_RatsnestMode = RATSNEST_MODE::ALL; } else if( opts.m_RatsnestMode == RATSNEST_MODE::ALL ) { opts.m_RatsnestMode = RATSNEST_MODE::VISIBLE; } else { opts.m_ShowGlobalRatsnest = false; } m_frame->SetDisplayOptions( opts ); return 0; } int PCB_CONTROL::LayerSwitch( const TOOL_EVENT& aEvent ) { m_frame->SwitchLayer( aEvent.Parameter<PCB_LAYER_ID>() ); return 0; } int PCB_CONTROL::LayerNext( const TOOL_EVENT& aEvent ) { PCB_BASE_FRAME* editFrame = m_frame; BOARD* brd = board(); LAYER_NUM layer = editFrame->GetActiveLayer(); LAYER_NUM startLayer = layer; if( layer < F_Cu || layer > B_Cu ) return 0; while( startLayer != ++layer ) { if( brd->IsLayerVisible( static_cast<PCB_LAYER_ID>( layer ) ) && IsCopperLayer( layer ) ) break; if( layer >= B_Cu ) layer = F_Cu - 1; } wxCHECK( IsCopperLayer( layer ), 0 ); editFrame->SwitchLayer( ToLAYER_ID( layer ) ); return 0; } int PCB_CONTROL::LayerPrev( const TOOL_EVENT& aEvent ) { PCB_BASE_FRAME* editFrame = m_frame; BOARD* brd = board(); LAYER_NUM layer = editFrame->GetActiveLayer(); LAYER_NUM startLayer = layer; if( layer < F_Cu || layer > B_Cu ) return 0; while( startLayer != --layer ) { if( IsCopperLayer( layer ) // also test for valid layer id (layer >= F_Cu) && brd->IsLayerVisible( static_cast<PCB_LAYER_ID>( layer ) ) ) { break; } if( layer <= F_Cu ) layer = B_Cu + 1; } wxCHECK( IsCopperLayer( layer ), 0 ); editFrame->SwitchLayer( ToLAYER_ID( layer ) ); return 0; } int PCB_CONTROL::LayerToggle( const TOOL_EVENT& aEvent ) { LAYER_NUM currentLayer = m_frame->GetActiveLayer(); PCB_SCREEN* screen = m_frame->GetScreen(); if( currentLayer == screen->m_Route_Layer_TOP ) m_frame->SwitchLayer( screen->m_Route_Layer_BOTTOM ); else m_frame->SwitchLayer( screen->m_Route_Layer_TOP ); return 0; } // It'd be nice to share the min/max with the DIALOG_COLOR_PICKER, but those are // set in wxFormBuilder. #define ALPHA_MIN 0.20 #define ALPHA_MAX 1.00 #define ALPHA_STEP 0.05 int PCB_CONTROL::LayerAlphaInc( const TOOL_EVENT& aEvent ) { COLOR_SETTINGS* settings = m_frame->GetColorSettings(); LAYER_NUM currentLayer = m_frame->GetActiveLayer(); KIGFX::COLOR4D currentColor = settings->GetColor( currentLayer ); if( currentColor.a <= ALPHA_MAX - ALPHA_STEP ) { currentColor.a += ALPHA_STEP; settings->SetColor( currentLayer, currentColor ); m_frame->GetCanvas()->UpdateColors(); KIGFX::VIEW* view = m_frame->GetCanvas()->GetView(); view->UpdateLayerColor( currentLayer ); view->UpdateLayerColor( GetNetnameLayer( currentLayer ) ); if( IsCopperLayer( currentLayer ) ) view->UpdateLayerColor( ZONE_LAYER_FOR( currentLayer ) ); static_cast<PCB_BASE_EDIT_FRAME*>( m_frame )->OnLayerAlphaChanged(); } else { wxBell(); } return 0; } int PCB_CONTROL::LayerAlphaDec( const TOOL_EVENT& aEvent ) { COLOR_SETTINGS* settings = m_frame->GetColorSettings(); LAYER_NUM currentLayer = m_frame->GetActiveLayer(); KIGFX::COLOR4D currentColor = settings->GetColor( currentLayer ); if( currentColor.a >= ALPHA_MIN + ALPHA_STEP ) { currentColor.a -= ALPHA_STEP; settings->SetColor( currentLayer, currentColor ); m_frame->GetCanvas()->UpdateColors(); KIGFX::VIEW* view = m_frame->GetCanvas()->GetView(); view->UpdateLayerColor( currentLayer ); view->UpdateLayerColor( GetNetnameLayer( currentLayer ) ); if( IsCopperLayer( currentLayer ) ) view->UpdateLayerColor( ZONE_LAYER_FOR( currentLayer ) ); static_cast<PCB_BASE_EDIT_FRAME*>( m_frame )->OnLayerAlphaChanged(); } else { wxBell(); } return 0; } void PCB_CONTROL::DoSetGridOrigin( KIGFX::VIEW* aView, PCB_BASE_FRAME* aFrame, EDA_ITEM* originViewItem, const VECTOR2D& aPoint ) { aFrame->GetDesignSettings().SetGridOrigin( wxPoint( aPoint ) ); aView->GetGAL()->SetGridOrigin( aPoint ); originViewItem->SetPosition( (wxPoint) aPoint ); aView->MarkDirty(); aFrame->OnModify(); } int PCB_CONTROL::GridSetOrigin( const TOOL_EVENT& aEvent ) { VECTOR2D* origin = aEvent.Parameter<VECTOR2D*>(); if( origin ) { // We can't undo the other grid dialog settings, so no sense undoing just the origin DoSetGridOrigin( getView(), m_frame, m_gridOrigin.get(), *origin ); delete origin; } else { if( m_isFootprintEditor && !getEditFrame<PCB_BASE_EDIT_FRAME>()->GetModel() ) return 0; std::string tool = aEvent.GetCommandStr().get(); PCB_PICKER_TOOL* picker = m_toolMgr->GetTool<PCB_PICKER_TOOL>(); if( !picker ) // Happens in footprint wizard return 0; // Deactivate other tools; particularly important if another PICKER is currently running Activate(); picker->SetClickHandler( [this]( const VECTOR2D& pt ) -> bool { m_frame->SaveCopyInUndoList( m_gridOrigin.get(), UNDO_REDO::GRIDORIGIN ); DoSetGridOrigin( getView(), m_frame, m_gridOrigin.get(), pt ); return false; // drill origin is a one-shot; don't continue with tool } ); m_toolMgr->RunAction( ACTIONS::pickerTool, true, &tool ); } return 0; } int PCB_CONTROL::GridResetOrigin( const TOOL_EVENT& aEvent ) { m_frame->SaveCopyInUndoList( m_gridOrigin.get(), UNDO_REDO::GRIDORIGIN ); DoSetGridOrigin( getView(), m_frame, m_gridOrigin.get(), VECTOR2D( 0, 0 ) ); return 0; } #define HITTEST_THRESHOLD_PIXELS 5 int PCB_CONTROL::DeleteItemCursor( const TOOL_EVENT& aEvent ) { if( m_isFootprintEditor && !m_frame->GetBoard()->GetFirstFootprint() ) return 0; std::string tool = aEvent.GetCommandStr().get(); PCB_PICKER_TOOL* picker = m_toolMgr->GetTool<PCB_PICKER_TOOL>(); m_pickerItem = nullptr; m_toolMgr->RunAction( PCB_ACTIONS::selectionClear, true ); // Deactivate other tools; particularly important if another PICKER is currently running Activate(); picker->SetCursor( KICURSOR::REMOVE ); picker->SetClickHandler( [this]( const VECTOR2D& aPosition ) -> bool { if( m_pickerItem ) { if( m_pickerItem && m_pickerItem->IsLocked() ) { m_statusPopup.reset( new STATUS_TEXT_POPUP( m_frame ) ); m_statusPopup->SetText( _( "Item locked." ) ); m_statusPopup->PopupFor( 2000 ); m_statusPopup->Move( wxGetMousePosition() + wxPoint( 20, 20 ) ); return true; } PCB_SELECTION_TOOL* selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>(); selectionTool->UnbrightenItem( m_pickerItem ); selectionTool->AddItemToSel( m_pickerItem, true /*quiet mode*/ ); m_toolMgr->RunAction( ACTIONS::doDelete, true ); m_pickerItem = nullptr; } return true; } ); picker->SetMotionHandler( [this]( const VECTOR2D& aPos ) { BOARD* board = m_frame->GetBoard(); PCB_SELECTION_TOOL* selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>(); GENERAL_COLLECTORS_GUIDE guide = m_frame->GetCollectorsGuide(); GENERAL_COLLECTOR collector; collector.m_Threshold = KiROUND( getView()->ToWorld( HITTEST_THRESHOLD_PIXELS ) ); if( m_isFootprintEditor ) collector.Collect( board, GENERAL_COLLECTOR::FootprintItems, (wxPoint) aPos, guide ); else collector.Collect( board, GENERAL_COLLECTOR::BoardLevelItems, (wxPoint) aPos, guide ); // Remove unselectable items for( int i = collector.GetCount() - 1; i >= 0; --i ) { if( !selectionTool->Selectable( collector[ i ] ) ) collector.Remove( i ); } if( collector.GetCount() > 1 ) selectionTool->GuessSelectionCandidates( collector, aPos ); BOARD_ITEM* item = collector.GetCount() == 1 ? collector[ 0 ] : nullptr; if( m_pickerItem != item ) { if( m_pickerItem ) selectionTool->UnbrightenItem( m_pickerItem ); m_pickerItem = item; if( m_pickerItem ) selectionTool->BrightenItem( m_pickerItem ); } } ); picker->SetFinalizeHandler( [this]( const int& aFinalState ) { if( m_pickerItem ) m_toolMgr->GetTool<PCB_SELECTION_TOOL>()->UnbrightenItem( m_pickerItem ); m_statusPopup.reset(); // Ensure the cursor gets changed&updated m_frame->GetCanvas()->SetCurrentCursor( KICURSOR::ARROW ); m_frame->GetCanvas()->Refresh(); } ); m_toolMgr->RunAction( ACTIONS::pickerTool, true, &tool ); return 0; } static void pasteFootprintItemsToFootprintEditor( FOOTPRINT* aClipFootprint, BOARD* aBoard, std::vector<BOARD_ITEM*>& aPastedItems ) { FOOTPRINT* editorFootprint = aBoard->GetFirstFootprint(); aClipFootprint->SetParent( aBoard ); for( PAD* pad : aClipFootprint->Pads() ) { pad->SetParent( editorFootprint ); aPastedItems.push_back( pad ); } aClipFootprint->Pads().clear(); // Not all graphic items can be added to the current footprint: // Reference and value are already existing in the current footprint, and // must be unique. // So they will be skipped for( BOARD_ITEM* item : aClipFootprint->GraphicalItems() ) { if( item->Type() == PCB_FP_SHAPE_T ) { FP_SHAPE* shape = static_cast<FP_SHAPE*>( item ); shape->SetParent( nullptr ); shape->SetLocalCoord(); } else if( item->Type() == PCB_FP_TEXT_T ) { FP_TEXT* text = static_cast<FP_TEXT*>( item ); if( text->GetType() != FP_TEXT::TEXT_is_DIVERS ) continue; text->SetTextAngle( text->GetTextAngle() + aClipFootprint->GetOrientation() ); text->SetParent( nullptr ); text->SetLocalCoord(); } item->SetParent( editorFootprint ); aPastedItems.push_back( item ); } aClipFootprint->GraphicalItems().clear(); for( FP_ZONE* zone : aClipFootprint->Zones() ) { zone->SetParent( editorFootprint ); aPastedItems.push_back( zone ); } aClipFootprint->Zones().clear(); for( PCB_GROUP* group : aClipFootprint->Groups() ) { group->SetParent( editorFootprint ); aPastedItems.push_back( group ); } aClipFootprint->Groups().clear(); } int PCB_CONTROL::Paste( const TOOL_EVENT& aEvent ) { CLIPBOARD_IO pi; BOARD_ITEM* clipItem = pi.Parse(); if( !clipItem ) return 0; // The viewer frames cannot paste if( !frame()->IsType( FRAME_FOOTPRINT_EDITOR ) && !frame()->IsType( FRAME_PCB_EDITOR ) ) return 0; PASTE_MODE pasteMode = PASTE_MODE::KEEP_ANNOTATIONS; const wxString defaultRef = wxT( "REF**" ); if( aEvent.IsAction( &ACTIONS::pasteSpecial ) ) { DIALOG_PASTE_SPECIAL dlg( m_frame, &pasteMode, defaultRef ); if( dlg.ShowModal() == wxID_CANCEL ) return 0; } bool isFootprintEditor = m_isFootprintEditor || frame()->IsType( FRAME_FOOTPRINT_EDITOR ); if( clipItem->Type() == PCB_T ) { if( isFootprintEditor ) { for( BOARD_CONNECTED_ITEM* item : static_cast<BOARD*>( clipItem )->AllConnectedItems() ) item->SetNet( NETINFO_LIST::OrphanedItem() ); } else { static_cast<BOARD*>( clipItem )->MapNets( m_frame->GetBoard() ); } } // The clipboard can contain two different things, an entire kicad_pcb or a single footprint if( isFootprintEditor && ( !board() || !footprint() ) ) { return 0; } switch( clipItem->Type() ) { case PCB_T: { BOARD* clipBoard = static_cast<BOARD*>( clipItem ); if( isFootprintEditor ) { FOOTPRINT* editorFootprint = board()->GetFirstFootprint(); std::vector<BOARD_ITEM*> pastedItems; for( FOOTPRINT* clipFootprint : clipBoard->Footprints() ) pasteFootprintItemsToFootprintEditor( clipFootprint, board(), pastedItems ); for( BOARD_ITEM* clipDrawItem : clipBoard->Drawings() ) { if( clipDrawItem->Type() == PCB_SHAPE_T ) { PCB_SHAPE* clipShape = static_cast<PCB_SHAPE*>( clipDrawItem ); // Convert to PCB_FP_SHAPE_T FP_SHAPE* pastedShape = new FP_SHAPE( editorFootprint ); static_cast<PCB_SHAPE*>( pastedShape )->SwapData( clipShape ); pastedShape->SetLocalCoord(); // Replace parent nuked by above call to SwapData() pastedShape->SetParent( editorFootprint ); pastedItems.push_back( pastedShape ); } else if( clipDrawItem->Type() == PCB_TEXT_T ) { PCB_TEXT* clipTextItem = static_cast<PCB_TEXT*>( clipDrawItem ); // Convert to PCB_FP_TEXT_T FP_TEXT* pastedTextItem = new FP_TEXT( editorFootprint ); static_cast<EDA_TEXT*>( pastedTextItem )->SwapText( *clipTextItem ); static_cast<EDA_TEXT*>( pastedTextItem )->SwapEffects( *clipTextItem ); pastedTextItem->SetParent( editorFootprint ); pastedItems.push_back( pastedTextItem ); } } delete clipBoard; placeBoardItems( pastedItems, true, true, pasteMode == PASTE_MODE::UNIQUE_ANNOTATIONS ); } else { clipBoard->SetElementVisibility( LAYER_RATSNEST, board()->IsElementVisible( LAYER_RATSNEST ) ); if( pasteMode == PASTE_MODE::REMOVE_ANNOTATIONS ) { for( FOOTPRINT* clipFootprint : clipBoard->Footprints() ) clipFootprint->SetReference( defaultRef ); } placeBoardItems( clipBoard, true, pasteMode == PASTE_MODE::UNIQUE_ANNOTATIONS ); m_frame->GetBoard()->BuildConnectivity(); m_frame->Compile_Ratsnest( true ); } break; } case PCB_FOOTPRINT_T: { FOOTPRINT* clipFootprint = static_cast<FOOTPRINT*>( clipItem ); std::vector<BOARD_ITEM*> pastedItems; if( isFootprintEditor ) { pasteFootprintItemsToFootprintEditor( clipFootprint, board(), pastedItems ); delete clipFootprint; } else { for( PAD* pad : clipFootprint->Pads() ) pad->SetLocalRatsnestVisible( board()->IsElementVisible( LAYER_RATSNEST ) ); if( pasteMode == PASTE_MODE::REMOVE_ANNOTATIONS ) clipFootprint->SetReference( defaultRef ); clipFootprint->SetParent( board() ); pastedItems.push_back( clipFootprint ); } placeBoardItems( pastedItems, true, true, pasteMode == PASTE_MODE::UNIQUE_ANNOTATIONS ); break; } default: m_frame->DisplayToolMsg( _( "Invalid clipboard contents" ) ); break; } return 1; } int PCB_CONTROL::AppendBoardFromFile( const TOOL_EVENT& aEvent ) { int open_ctl; wxString fileName; PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame ); if( !editFrame ) return 1; // Pick a file to append if( !AskLoadBoardFileName( editFrame, &open_ctl, &fileName, true ) ) return 1; IO_MGR::PCB_FILE_T pluginType = plugin_type( fileName, open_ctl ); PLUGIN::RELEASER pi( IO_MGR::PluginFind( pluginType ) ); return AppendBoard( *pi, fileName ); } // Helper function for PCB_CONTROL::placeBoardItems() template<typename T> static void moveUnflaggedItems( std::deque<T>& aList, std::vector<BOARD_ITEM*>& aTarget, bool aIsNew ) { std::copy_if( aList.begin(), aList.end(), std::back_inserter( aTarget ), [aIsNew]( T aItem ) { bool doCopy = ( aItem->GetFlags() & SKIP_STRUCT ) == 0; aItem->ClearFlags( SKIP_STRUCT ); aItem->SetFlags( aIsNew ? IS_NEW : 0 ); return doCopy; } ); if( aIsNew ) aList.clear(); } static void moveUnflaggedItems( ZONES& aList, std::vector<BOARD_ITEM*>& aTarget, bool aIsNew ) { if( aList.size() == 0 ) return; auto obj = aList.front(); int idx = 0; if( aIsNew ) { obj = aList.back(); aList.pop_back(); } for( ; obj ; ) { if( obj->HasFlag( SKIP_STRUCT ) ) obj->ClearFlags( SKIP_STRUCT ); else aTarget.push_back( obj ); if( aIsNew ) { if( aList.size() ) { obj = aList.back(); aList.pop_back(); } else { obj = nullptr; } } else { obj = idx < int(aList.size()-1) ? aList[++idx] : nullptr; } } } int PCB_CONTROL::placeBoardItems( BOARD* aBoard, bool aAnchorAtOrigin, bool aReannotateDuplicates ) { // items are new if the current board is not the board source bool isNew = board() != aBoard; std::vector<BOARD_ITEM*> items; moveUnflaggedItems( aBoard->Tracks(), items, isNew ); moveUnflaggedItems( aBoard->Footprints(), items, isNew ); moveUnflaggedItems( aBoard->Drawings(), items, isNew ); moveUnflaggedItems( aBoard->Zones(), items, isNew ); // Subtlety: When selecting a group via the mouse, // PCB_SELECTION_TOOL::highlightInternal runs, which does a SetSelected() on all // descendants. In PCB_CONTROL::placeBoardItems, below, we skip that and // mark items non-recursively. That works because the saving of the // selection created aBoard that has the group and all descendants in it. moveUnflaggedItems( aBoard->Groups(), items, isNew ); return placeBoardItems( items, isNew, aAnchorAtOrigin, aReannotateDuplicates ); } int PCB_CONTROL::placeBoardItems( std::vector<BOARD_ITEM*>& aItems, bool aIsNew, bool aAnchorAtOrigin, bool aReannotateDuplicates ) { m_toolMgr->RunAction( PCB_ACTIONS::selectionClear, true ); PCB_SELECTION_TOOL* selectionTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>(); EDIT_TOOL* editTool = m_toolMgr->GetTool<EDIT_TOOL>(); std::vector<BOARD_ITEM*> itemsToSel; itemsToSel.reserve( aItems.size() ); for( BOARD_ITEM* item : aItems ) { if( aIsNew ) { const_cast<KIID&>( item->m_Uuid ) = KIID(); if( selectionTool->GetEnteredGroup() && !item->GetParentGroup() ) selectionTool->GetEnteredGroup()->AddItem( item ); } // Update item attributes if needed switch( item->Type() ) { case PCB_DIMENSION_T: case PCB_DIM_ALIGNED_T: case PCB_DIM_CENTER_T: case PCB_DIM_ORTHOGONAL_T: case PCB_DIM_LEADER_T: { // Dimensions need to have their units updated if they are automatic PCB_DIMENSION_BASE* dim = static_cast<PCB_DIMENSION_BASE*>( item ); if( dim->GetUnitsMode() == DIM_UNITS_MODE::AUTOMATIC ) dim->SetUnits( frame()->GetUserUnits() ); break; } case PCB_FOOTPRINT_T: // Update the footprint path with the new KIID path if the footprint is new if( aIsNew ) static_cast<FOOTPRINT*>( item )->SetPath( KIID_PATH() ); break; default: break; } // We only need to add the items that aren't inside a group currently selected // to the selection. If an item is inside a group and that group is selected, // then the selection tool will select it for us. if( !item->GetParentGroup() || !alg::contains( aItems, item->GetParentGroup() ) ) itemsToSel.push_back( item ); } // Select the items that should be selected m_toolMgr->RunAction( PCB_ACTIONS::selectItems, true, &itemsToSel ); // Reannotate duplicate footprints (make sense only in board editor ) if( aReannotateDuplicates && m_frame->IsType( FRAME_PCB_EDITOR ) ) m_toolMgr->GetTool<BOARD_REANNOTATE_TOOL>()->ReannotateDuplicatesInSelection(); for( BOARD_ITEM* item : aItems ) { // Commit after reannotation if( aIsNew ) editTool->GetCurrentCommit()->Add( item ); else editTool->GetCurrentCommit()->Added( item ); } PCB_SELECTION& selection = selectionTool->GetSelection(); if( selection.Size() > 0 ) { if( aAnchorAtOrigin ) { selection.SetReferencePoint( VECTOR2I( 0, 0 ) ); } else { BOARD_ITEM* item = static_cast<BOARD_ITEM*>( selection.GetTopLeftItem() ); selection.SetReferencePoint( item->GetPosition() ); } getViewControls()->SetCursorPosition( getViewControls()->GetMousePosition(), false ); m_toolMgr->ProcessEvent( EVENTS::SelectedEvent ); m_toolMgr->RunAction( PCB_ACTIONS::move, true ); } return 0; } int PCB_CONTROL::AppendBoard( PLUGIN& pi, wxString& fileName ) { PCB_EDIT_FRAME* editFrame = dynamic_cast<PCB_EDIT_FRAME*>( m_frame ); if( !editFrame ) return 1; BOARD* brd = board(); if( !brd ) return 1; // Mark existing items, in order to know what are the new items so we can select only // the new items after loading for( PCB_TRACK* track : brd->Tracks() ) track->SetFlags( SKIP_STRUCT ); for( FOOTPRINT* footprint : brd->Footprints() ) footprint->SetFlags( SKIP_STRUCT ); for( PCB_GROUP* group : brd->Groups() ) group->SetFlags( SKIP_STRUCT ); for( BOARD_ITEM* drawing : brd->Drawings() ) drawing->SetFlags( SKIP_STRUCT ); for( ZONE* zone : brd->Zones() ) zone->SetFlags( SKIP_STRUCT ); std::map<wxString, wxString> oldProperties = brd->GetProperties(); std::map<wxString, wxString> newProperties; // Keep also the count of copper layers, to adjust if necessary int initialCopperLayerCount = brd->GetCopperLayerCount(); LSET initialEnabledLayers = brd->GetEnabledLayers(); // Load the data try { PROPERTIES props; char xbuf[30]; char ybuf[30]; // EAGLE_PLUGIN can use this info to center the BOARD, but it does not yet. sprintf( xbuf, "%d", editFrame->GetPageSizeIU().x ); sprintf( ybuf, "%d", editFrame->GetPageSizeIU().y ); props["page_width"] = xbuf; props["page_height"] = ybuf; WX_PROGRESS_REPORTER progressReporter( editFrame, _( "Loading PCB" ), 1 ); editFrame->GetDesignSettings().GetNetClasses().Clear(); pi.Load( fileName, brd, &props, nullptr, &progressReporter ); } catch( const IO_ERROR& ioe ) { wxString msg = wxString::Format( _( "Error loading board.\n%s" ), ioe.What() ); DisplayError( editFrame, msg ); return 0; } newProperties = brd->GetProperties(); for( const std::pair<const wxString, wxString>& prop : oldProperties ) newProperties[ prop.first ] = prop.second; brd->SetProperties( newProperties ); // rebuild nets and ratsnest before any use of nets brd->BuildListOfNets(); brd->SynchronizeNetsAndNetClasses(); brd->BuildConnectivity(); // Synchronize layers // we should not ask PLUGINs to do these items: int copperLayerCount = brd->GetCopperLayerCount(); if( copperLayerCount > initialCopperLayerCount ) brd->SetCopperLayerCount( copperLayerCount ); // Enable all used layers, and make them visible: LSET enabledLayers = brd->GetEnabledLayers(); enabledLayers |= initialEnabledLayers; brd->SetEnabledLayers( enabledLayers ); brd->SetVisibleLayers( enabledLayers ); return placeBoardItems( brd, false, false ); // Do not reannotate duplicates on Append Board } int PCB_CONTROL::Undo( const TOOL_EVENT& aEvent ) { PCB_BASE_EDIT_FRAME* editFrame = dynamic_cast<PCB_BASE_EDIT_FRAME*>( m_frame ); wxCommandEvent dummy; if( editFrame ) editFrame->RestoreCopyFromUndoList( dummy ); return 0; } int PCB_CONTROL::Redo( const TOOL_EVENT& aEvent ) { PCB_BASE_EDIT_FRAME* editFrame = dynamic_cast<PCB_BASE_EDIT_FRAME*>( m_frame ); wxCommandEvent dummy; if( editFrame ) editFrame->RestoreCopyFromRedoList( dummy ); return 0; } int PCB_CONTROL::UpdateMessagePanel( const TOOL_EVENT& aEvent ) { PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool<PCB_SELECTION_TOOL>(); ROUTER_TOOL* routerTool = m_toolMgr->GetTool<ROUTER_TOOL>(); PCB_SELECTION& selection = selTool->GetSelection(); if( routerTool && routerTool->RoutingInProgress() ) { routerTool->UpdateMessagePanel(); return 0; } if( selection.GetSize() == 1 ) { EDA_ITEM* item = selection.Front(); std::vector<MSG_PANEL_ITEM> msgItems; item->GetMsgPanelInfo( m_frame, msgItems ); m_frame->SetMsgPanel( msgItems ); } else if( selection.GetSize() > 1 ) { std::vector<MSG_PANEL_ITEM> msgItems; wxString msg = wxString::Format( wxT( "%d" ), selection.GetSize() ); msgItems.emplace_back( MSG_PANEL_ITEM( _( "Selected Items" ), msg ) ); m_frame->SetMsgPanel( msgItems ); } else if( auto editFrame = dynamic_cast<FOOTPRINT_EDIT_FRAME*>( m_frame ) ) { FOOTPRINT* footprint = static_cast<FOOTPRINT*>( editFrame->GetModel() ); if( !footprint ) return 0; std::vector<MSG_PANEL_ITEM> msgItems; wxString msg; msg = footprint->GetFPID().GetLibNickname().wx_str(); msgItems.emplace_back( MSG_PANEL_ITEM( _( "Library" ), msg ) ); msg = footprint->GetFPID().GetLibItemName().wx_str(); msgItems.emplace_back( MSG_PANEL_ITEM( _( "Footprint Name" ), msg ) ); wxDateTime date( static_cast<time_t>( footprint->GetLastEditTime() ) ); if( footprint->GetLastEditTime() && date.IsValid() ) // Date format: see http://www.cplusplus.com/reference/ctime/strftime msg = date.Format( wxT( "%b %d, %Y" ) ); // Abbreviated_month_name Day, Year else msg = _( "Unknown" ); msgItems.emplace_back( MSG_PANEL_ITEM( _( "Last Change" ), msg ) ); msg.Printf( wxT( "%zu" ), (size_t) footprint->GetPadCount( DO_NOT_INCLUDE_NPTH ) ); msgItems.emplace_back( MSG_PANEL_ITEM( _( "Pads" ), msg ) ); wxString doc, keyword; doc.Printf( _( "Doc: %s" ), footprint->GetDescription() ); keyword.Printf( _( "Keywords: %s" ), footprint->GetKeywords() ); msgItems.emplace_back( MSG_PANEL_ITEM( doc, keyword ) ); m_frame->SetMsgPanel( msgItems ); } else { m_frame->SetMsgPanel( m_frame->GetBoard() ); } return 0; } int PCB_CONTROL::FlipPcbView( const TOOL_EVENT& aEvent ) { view()->SetMirror( !view()->IsMirroredX(), false ); view()->RecacheAllItems(); frame()->GetCanvas()->ForceRefresh(); frame()->OnDisplayOptionsChanged(); return 0; } void PCB_CONTROL::setTransitions() { Go( &PCB_CONTROL::AddLibrary, ACTIONS::newLibrary.MakeEvent() ); Go( &PCB_CONTROL::AddLibrary, ACTIONS::addLibrary.MakeEvent() ); Go( &PCB_CONTROL::Print, ACTIONS::print.MakeEvent() ); Go( &PCB_CONTROL::Quit, ACTIONS::quit.MakeEvent() ); // Display modes Go( &PCB_CONTROL::TrackDisplayMode, PCB_ACTIONS::trackDisplayMode.MakeEvent() ); Go( &PCB_CONTROL::ToggleRatsnest, PCB_ACTIONS::showRatsnest.MakeEvent() ); Go( &PCB_CONTROL::ToggleRatsnest, PCB_ACTIONS::ratsnestLineMode.MakeEvent() ); Go( &PCB_CONTROL::ViaDisplayMode, PCB_ACTIONS::viaDisplayMode.MakeEvent() ); Go( &PCB_CONTROL::ZoneDisplayMode, PCB_ACTIONS::zoneDisplayFilled.MakeEvent() ); Go( &PCB_CONTROL::ZoneDisplayMode, PCB_ACTIONS::zoneDisplayOutline.MakeEvent() ); Go( &PCB_CONTROL::ZoneDisplayMode, PCB_ACTIONS::zoneDisplayFractured.MakeEvent() ); Go( &PCB_CONTROL::ZoneDisplayMode, PCB_ACTIONS::zoneDisplayTriangulated.MakeEvent() ); Go( &PCB_CONTROL::ZoneDisplayMode, PCB_ACTIONS::zoneDisplayToggle.MakeEvent() ); Go( &PCB_CONTROL::HighContrastMode, ACTIONS::highContrastMode.MakeEvent() ); Go( &PCB_CONTROL::HighContrastModeCycle, ACTIONS::highContrastModeCycle.MakeEvent() ); Go( &PCB_CONTROL::NetColorModeCycle, PCB_ACTIONS::netColorModeCycle.MakeEvent() ); Go( &PCB_CONTROL::RatsnestModeCycle, PCB_ACTIONS::ratsnestModeCycle.MakeEvent() ); Go( &PCB_CONTROL::FlipPcbView, PCB_ACTIONS::flipBoard.MakeEvent() ); // Layer control Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerTop.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner1.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner2.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner3.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner4.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner5.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner6.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner7.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner8.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner9.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner10.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner11.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner12.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner13.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner14.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner15.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner16.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner17.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner18.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner19.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner20.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner21.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner22.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner23.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner24.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner25.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner26.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner27.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner28.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner29.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerInner30.MakeEvent() ); Go( &PCB_CONTROL::LayerSwitch, PCB_ACTIONS::layerBottom.MakeEvent() ); Go( &PCB_CONTROL::LayerNext, PCB_ACTIONS::layerNext.MakeEvent() ); Go( &PCB_CONTROL::LayerPrev, PCB_ACTIONS::layerPrev.MakeEvent() ); Go( &PCB_CONTROL::LayerToggle, PCB_ACTIONS::layerToggle.MakeEvent() ); Go( &PCB_CONTROL::LayerAlphaInc, PCB_ACTIONS::layerAlphaInc.MakeEvent() ); Go( &PCB_CONTROL::LayerAlphaDec, PCB_ACTIONS::layerAlphaDec.MakeEvent() ); // Grid control Go( &PCB_CONTROL::GridSetOrigin, ACTIONS::gridSetOrigin.MakeEvent() ); Go( &PCB_CONTROL::GridResetOrigin, ACTIONS::gridResetOrigin.MakeEvent() ); Go( &PCB_CONTROL::Undo, ACTIONS::undo.MakeEvent() ); Go( &PCB_CONTROL::Redo, ACTIONS::redo.MakeEvent() ); // Miscellaneous Go( &PCB_CONTROL::DeleteItemCursor, ACTIONS::deleteTool.MakeEvent() ); // Append control Go( &PCB_CONTROL::AppendBoardFromFile, PCB_ACTIONS::appendBoard.MakeEvent() ); Go( &PCB_CONTROL::Paste, ACTIONS::paste.MakeEvent() ); Go( &PCB_CONTROL::Paste, ACTIONS::pasteSpecial.MakeEvent() ); Go( &PCB_CONTROL::UpdateMessagePanel, EVENTS::SelectedEvent ); Go( &PCB_CONTROL::UpdateMessagePanel, EVENTS::UnselectedEvent ); Go( &PCB_CONTROL::UpdateMessagePanel, EVENTS::ClearedEvent ); Go( &PCB_CONTROL::UpdateMessagePanel, EVENTS::SelectedItemsModified ); }