/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2023 Alex Shvartzkop * Copyright (C) 2023 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 "generator_tool.h" #include #include #include #include #include #include GENERATOR_TOOL::GENERATOR_TOOL() : GENERATOR_TOOL_PNS_PROXY( "pcbnew.Generators" ), m_mgrDialog( nullptr ) { PROPERTY_MANAGER::Instance().RegisterListener( TYPE_HASH( BOARD_ITEM ), [&]( INSPECTABLE* aItem, PROPERTY_BASE* aProperty, COMMIT* aCommit ) { // Special case: propagate lock from generated items to parent generator BOARD_ITEM* item = dynamic_cast( aItem ); if( item && aProperty->Name() == _HKI( "Locked" ) ) { if( PCB_GENERATOR* generator = dynamic_cast( item->GetParentGroup() ) ) { if( aCommit->GetStatus( generator ) != CHT_MODIFY ) aCommit->Modify( generator ); // Must set generator to unlocked first or item->IsLocked() will just // return the parent's locked state. generator->SetLocked( false ); generator->SetLocked( item->IsLocked() ); } } } ); PROPERTY_MANAGER::Instance().RegisterListener( TYPE_HASH( PCB_GENERATOR ), [&]( INSPECTABLE* aItem, PROPERTY_BASE* aProperty, COMMIT* aCommit ) { // Special case: regenerator generators when their properties change if( PCB_GENERATOR* generator = dynamic_cast( aItem ) ) { BOARD_COMMIT* commit = static_cast( aCommit ); generator->EditStart( this, board(), commit ); generator->Update( this, board(), commit ); generator->EditPush( this, board(), commit ); } } ); } GENERATOR_TOOL::~GENERATOR_TOOL() { } void GENERATOR_TOOL::Reset( RESET_REASON aReason ) { GENERATOR_TOOL_PNS_PROXY::Reset( aReason ); } bool GENERATOR_TOOL::Init() { auto tuningPatternCondition = []( const SELECTION& aSel ) { for( EDA_ITEM* item : aSel ) { if( PCB_GENERATOR* generator = dynamic_cast( item ) ) { if( generator->GetGeneratorType() == wxS( "tuning_pattern" ) ) return true; } } return false; }; // Add the generator control menus to relevant other tools PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool(); if( selTool ) { TOOL_MENU& toolMenu = selTool->GetToolMenu(); CONDITIONAL_MENU& menu = toolMenu.GetMenu(); menu.AddItem( PCB_ACTIONS::regenerateAllTuning, tuningPatternCondition, 100 ); } ROUTER_TOOL* routerTool = m_toolMgr->GetTool(); if( routerTool ) { TOOL_MENU& toolMenu = routerTool->GetToolMenu(); CONDITIONAL_MENU& menu = toolMenu.GetMenu(); menu.AddItem( PCB_ACTIONS::regenerateAllTuning, SELECTION_CONDITIONS::ShowAlways, 100 ); } return true; } void GENERATOR_TOOL::DestroyManagerDialog() { if( m_mgrDialog ) { m_mgrDialog->Destroy(); m_mgrDialog = nullptr; } } int GENERATOR_TOOL::ShowGeneratorsManager( const TOOL_EVENT& aEvent ) { PCB_EDIT_FRAME* pcbFrame = static_cast( frame() ); if( !pcbFrame ) return 0; if( !m_mgrDialog ) { m_mgrDialog = new DIALOG_GENERATORS( pcbFrame, pcbFrame ); } else { m_mgrDialog->RebuildModels(); } m_mgrDialog->Show( true ); return 0; } int GENERATOR_TOOL::RegenerateAllOfType( const TOOL_EVENT& aEvent ) { wxString generatorType = aEvent.Parameter(); BOARD_COMMIT commit( this ); wxString commitMsg; int commitFlags = 0; if( generatorType == wxS( "*" ) ) commitMsg = _( "Regenerate All" ); for( PCB_GENERATOR* generator : board()->Generators() ) { if( generatorType == wxS( "*" ) || generator->GetGeneratorType() == generatorType ) { if( commitMsg.IsEmpty() ) commitMsg.Printf( _( "Update %s" ), generator->GetPluralName() ); generator->EditStart( this, board(), &commit ); generator->Update( this, board(), &commit ); generator->EditPush( this, board(), &commit, commitMsg, commitFlags ); commitFlags |= APPEND_UNDO; } } frame()->RefreshCanvas(); return 0; } int GENERATOR_TOOL::RegenerateSelected( const TOOL_EVENT& aEvent ) { BOARD_COMMIT commit( this ); int commitFlags = 0; PCB_SELECTION_TOOL* selTool = m_toolMgr->GetTool(); PCB_SELECTION sel = selTool->RequestSelection( []( const VECTOR2I& aPt, GENERAL_COLLECTOR& aCollector, PCB_SELECTION_TOOL* sTool ) { // Iterate from the back so we don't have to worry about removals. for( int i = aCollector.GetCount() - 1; i >= 0; --i ) { BOARD_ITEM* item = aCollector[i]; if( item->Type() != PCB_GENERATOR_T ) aCollector.Remove( item ); } } ); GENERATORS generators; for( EDA_ITEM* item : sel ) { if( PCB_GENERATOR* gen = dynamic_cast( item ) ) generators.push_back( gen ); } #ifdef GENERATOR_ORDER std::sort( generators.begin(), generators.end(), []( const PCB_GENERATOR* a, const PCB_GENERATOR* b ) -> bool { return a->GetUpdateOrder() < b->GetUpdateOrder(); } ); #endif for( PCB_GENERATOR* gen : generators ) { gen->EditStart( this, board(), &commit ); gen->Update( this, board(), &commit ); gen->EditPush( this, board(), &commit, _( "Regenerate Selected" ), commitFlags ); commitFlags |= APPEND_UNDO; } frame()->RefreshCanvas(); return 0; } int GENERATOR_TOOL::RegenerateItem( const TOOL_EVENT& aEvent ) { BOARD_COMMIT commit( this ); int commitFlags = 0; PCB_GENERATOR* gen = aEvent.Parameter(); gen->EditStart( this, board(), &commit ); gen->Update( this, board(), &commit ); gen->EditPush( this, board(), &commit, _( "Regenerate Item" ), commitFlags ); frame()->RefreshCanvas(); return 0; } int GENERATOR_TOOL::GenEditAction( const TOOL_EVENT& aEvent ) { BOARD_COMMIT* commit = dynamic_cast( aEvent.Commit() ); wxCHECK( commit, 0 ); PCB_GENERATOR* gen = aEvent.Parameter(); if( aEvent.IsAction( &PCB_ACTIONS::genStartEdit ) ) { gen->EditStart( this, board(), commit ); } else if( aEvent.IsAction( &PCB_ACTIONS::genUpdateEdit ) ) { gen->Update( this, board(), commit ); } else if( aEvent.IsAction( &PCB_ACTIONS::genPushEdit ) ) { gen->EditPush( this, board(), commit, wxEmptyString ); wxASSERT( commit->Empty() ); } else if( aEvent.IsAction( &PCB_ACTIONS::genRevertEdit ) ) { gen->EditRevert( this, board(), commit ); wxASSERT( commit->Empty() ); } else if( aEvent.IsAction( &PCB_ACTIONS::genRemove ) ) { gen->Remove( this, board(), commit ); } return 0; } void GENERATOR_TOOL::setTransitions() { // Generator actions Go( &GENERATOR_TOOL::ShowGeneratorsManager, PCB_ACTIONS::generatorsShowManager.MakeEvent() ); Go( &GENERATOR_TOOL::RegenerateAllOfType, PCB_ACTIONS::regenerateAllTuning.MakeEvent() ); Go( &GENERATOR_TOOL::RegenerateAllOfType, PCB_ACTIONS::regenerateAll.MakeEvent() ); Go( &GENERATOR_TOOL::RegenerateSelected, PCB_ACTIONS::regenerateSelected.MakeEvent() ); Go( &GENERATOR_TOOL::GenEditAction, PCB_ACTIONS::genStartEdit.MakeEvent() ); Go( &GENERATOR_TOOL::GenEditAction, PCB_ACTIONS::genUpdateEdit.MakeEvent() ); Go( &GENERATOR_TOOL::GenEditAction, PCB_ACTIONS::genPushEdit.MakeEvent() ); Go( &GENERATOR_TOOL::GenEditAction, PCB_ACTIONS::genRevertEdit.MakeEvent() ); Go( &GENERATOR_TOOL::GenEditAction, PCB_ACTIONS::genRemove.MakeEvent() ); }