/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2019-2020 KiCad Developers, see AUTHORS.TXT for contributors. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, you may find one here: * http://www.gnu.org/licenses/old-licenses/gpl-2.0.html * or you may search the http://www.gnu.org website for the version 2 license, * or you may write to the Free Software Foundation, Inc., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include GLOBAL_EDIT_TOOL::GLOBAL_EDIT_TOOL() : PCB_TOOL_BASE( "pcbnew.GlobalEdit" ), m_selectionTool( nullptr ) { } void GLOBAL_EDIT_TOOL::Reset( RESET_REASON aReason ) { if( aReason != RUN ) m_commit = std::make_unique( this ); } bool GLOBAL_EDIT_TOOL::Init() { // Find the selection tool, so they can cooperate m_selectionTool = m_toolMgr->GetTool(); return true; } int GLOBAL_EDIT_TOOL::ExchangeFootprints( const TOOL_EVENT& aEvent ) { PCB_SELECTION& selection = m_selectionTool->GetSelection(); FOOTPRINT* footprint = nullptr; bool updateMode = false; bool currentMode = false; if( aEvent.HasPosition() ) selection = m_selectionTool->RequestSelection( EDIT_TOOL::FootprintFilter ); if( !selection.Empty() ) footprint = selection.FirstOfKind(); if( aEvent.IsAction( &PCB_ACTIONS::updateFootprint ) ) { updateMode = true; currentMode = true; } else if( aEvent.IsAction( &PCB_ACTIONS::updateFootprints ) ) { updateMode = true; currentMode = selection.CountType( PCB_FOOTPRINT_T ) > 0; } else if( aEvent.IsAction( &PCB_ACTIONS::changeFootprint ) ) { updateMode = false; currentMode = true; } else if( aEvent.IsAction( &PCB_ACTIONS::changeFootprints ) ) { updateMode = false; currentMode = selection.CountType( PCB_FOOTPRINT_T ) > 0; } else { wxFAIL_MSG( wxT( "ExchangeFootprints: unexpected action" ) ); } // invoke the exchange dialog process { PCB_EDIT_FRAME* editFrame = getEditFrame(); DIALOG_EXCHANGE_FOOTPRINTS dialog( editFrame, footprint, updateMode, currentMode ); dialog.ShowQuasiModal(); } return 0; } bool GLOBAL_EDIT_TOOL::swapBoardItem( BOARD_ITEM* aItem, std::map& aLayerMap ) { LSET originalLayers = aItem->GetLayerSet(); LSET newLayers; for( PCB_LAYER_ID original : originalLayers.Seq() ) { if( aLayerMap.count( original ) ) newLayers.set( aLayerMap[ original ] ); else newLayers.set( original ); } if( originalLayers.Seq() != newLayers.Seq() ) { m_commit->Modify( aItem ); aItem->SetLayerSet( newLayers ); frame()->GetCanvas()->GetView()->Update( aItem, KIGFX::GEOMETRY ); return true; } return false; } int GLOBAL_EDIT_TOOL::SwapLayers( const TOOL_EVENT& aEvent ) { std::map layerMap; DIALOG_SWAP_LAYERS dlg( frame(), layerMap ); if( dlg.ShowModal() != wxID_OK ) return 0; bool hasChanges = false; // Change tracks. for( PCB_TRACK* segm : frame()->GetBoard()->Tracks() ) { if( segm->Type() == PCB_VIA_T ) { PCB_VIA* via = static_cast( segm ); PCB_LAYER_ID top_layer, bottom_layer; if( via->GetViaType() == VIATYPE::THROUGH ) continue; via->LayerPair( &top_layer, &bottom_layer ); if( layerMap[bottom_layer] != bottom_layer || layerMap[top_layer] != top_layer ) { m_commit->Modify( via ); via->SetLayerPair( layerMap[top_layer], layerMap[bottom_layer] ); frame()->GetCanvas()->GetView()->Update( via, KIGFX::GEOMETRY ); hasChanges = true; } } else { hasChanges |= swapBoardItem( segm, layerMap ); } } for( BOARD_ITEM* zone : frame()->GetBoard()->Zones() ) hasChanges |= swapBoardItem( zone, layerMap ); for( BOARD_ITEM* drawing : frame()->GetBoard()->Drawings() ) hasChanges |= swapBoardItem( drawing, layerMap ); if( hasChanges ) { frame()->OnModify(); m_commit->Push( wxT( "Swap Layers" ) ); frame()->GetCanvas()->Refresh(); } return 0; } int GLOBAL_EDIT_TOOL::CleanupTracksAndVias( const TOOL_EVENT& aEvent ) { PCB_EDIT_FRAME* editFrame = getEditFrame(); DIALOG_CLEANUP_TRACKS_AND_VIAS dlg( editFrame ); dlg.ShowModal(); return 0; } int GLOBAL_EDIT_TOOL::CleanupGraphics( const TOOL_EVENT& aEvent ) { PCB_EDIT_FRAME* editFrame = getEditFrame(); DIALOG_CLEANUP_GRAPHICS dlg( editFrame, false ); dlg.ShowModal(); return 0; } int GLOBAL_EDIT_TOOL::RemoveUnusedPads( const TOOL_EVENT& aEvent ) { PCB_EDIT_FRAME* editFrame = getEditFrame(); PCB_SELECTION& selection = m_selectionTool->RequestSelection( []( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector, PCB_SELECTION_TOOL* sTool ) { sTool->FilterCollectorForHierarchy( aCollector, true ); } ); DIALOG_UNUSED_PAD_LAYERS dlg( editFrame, selection, *m_commit ); dlg.ShowModal(); return 0; } int GLOBAL_EDIT_TOOL::ZonesManager( const TOOL_EVENT& aEvent ) { PCB_EDIT_FRAME* editFrame = getEditFrame(); BOARD_COMMIT commit( editFrame ); BOARD* board = editFrame->GetBoard(); for( ZONE* zone : board->Zones() ) commit.Modify( zone ); ZONE_SETTINGS zoneInfo = board->GetDesignSettings().GetDefaultZoneSettings(); DIALOG_ZONE_MANAGER dlg( editFrame, &zoneInfo ); int dialogResult = dlg.ShowQuasiModal(); if( dialogResult == wxID_OK && ZONE_MANAGER_PREFERENCE::GetRepourOnClose() ) dialogResult = ZONE_MANAGER_REPOUR; if( dialogResult == wxID_CANCEL ) return 0; wxBusyCursor dummy; // Undraw old zone outlines for( ZONE* zone : board->Zones() ) editFrame->GetCanvas()->GetView()->Update( zone ); board->GetDesignSettings().SetDefaultZoneSettings( zoneInfo ); commit.Push( _( "Modify zones properties with zone manager" ), SKIP_CONNECTIVITY ); editFrame->OnModify(); //rebuildConnectivity board->BuildConnectivity(); if( TOOL_MANAGER* manger = GetManager() ) { manger->PostEvent( EVENTS::ConnectivityChangedEvent ); } editFrame->GetCanvas()->RedrawRatsnest(); if( dialogResult == ZONE_MANAGER_REPOUR ) { if( TOOL_MANAGER* manger = GetManager() ) { manger->PostAction( PCB_ACTIONS::zoneFillAll ); } } return 0; } void GLOBAL_EDIT_TOOL::setTransitions() { Go( &GLOBAL_EDIT_TOOL::ExchangeFootprints, PCB_ACTIONS::updateFootprint.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::ExchangeFootprints, PCB_ACTIONS::updateFootprints.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::ExchangeFootprints, PCB_ACTIONS::changeFootprint.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::ExchangeFootprints, PCB_ACTIONS::changeFootprints.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::SwapLayers, PCB_ACTIONS::swapLayers.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::EditTracksAndVias, PCB_ACTIONS::editTracksAndVias.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::EditTextAndGraphics, PCB_ACTIONS::editTextAndGraphics.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::EditTeardrops, PCB_ACTIONS::editTeardrops.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::GlobalDeletions, PCB_ACTIONS::globalDeletions.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::CleanupTracksAndVias, PCB_ACTIONS::cleanupTracksAndVias.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::CleanupGraphics, PCB_ACTIONS::cleanupGraphics.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::RemoveUnusedPads, PCB_ACTIONS::removeUnusedPads.MakeEvent() ); Go( &GLOBAL_EDIT_TOOL::ZonesManager, PCB_ACTIONS::zonesManager.MakeEvent() ); }