/* * This program source code file is part of KiCad, a free EDA CAD application. * * Copyright (C) 2014 CERN * Copyright (C) 2020-2022 KiCad Developers, see AUTHORS.txt for contributors. * @author Maciej Suminski * * 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 #include #include #include #include PCB_BASE_EDIT_FRAME::PCB_BASE_EDIT_FRAME( KIWAY* aKiway, wxWindow* aParent, FRAME_T aFrameType, const wxString& aTitle, const wxPoint& aPos, const wxSize& aSize, long aStyle, const wxString& aFrameName ) : PCB_BASE_FRAME( aKiway, aParent, aFrameType, aTitle, aPos, aSize, aStyle, aFrameName ), m_undoRedoBlocked( false ), m_selectionFilterPanel( nullptr ), m_appearancePanel( nullptr ) { Bind( wxEVT_IDLE, [this]( wxIdleEvent& aEvent ) { // Handle cursor adjustments. While we can get motion and key events through // wxWidgets, we can't get modifier-key-up events. if( m_toolManager ) { PCB_SELECTION_TOOL* selTool = m_toolManager->GetTool(); if( selTool ) selTool->OnIdle( aEvent ); } } ); } PCB_BASE_EDIT_FRAME::~PCB_BASE_EDIT_FRAME() { GetCanvas()->GetView()->Clear(); } void PCB_BASE_EDIT_FRAME::doCloseWindow() { SETTINGS_MANAGER* mgr = GetSettingsManager(); wxFileName projectName( Prj().GetProjectFullName() ); if( mgr->IsProjectOpen() && wxFileName::IsDirWritable( projectName.GetPath() ) && projectName.Exists() ) { GFootprintList.WriteCacheToFile( Prj().GetProjectPath() + wxT( "fp-info-cache" ) ); } // Close the project if we are standalone, so it gets cleaned up properly if( mgr->IsProjectOpen() && Kiface().IsSingle() ) mgr->UnloadProject( &Prj(), false ); } bool PCB_BASE_EDIT_FRAME::TryBefore( wxEvent& aEvent ) { static bool s_presetSwitcherShown = false; static bool s_viewportSwitcherShown = false; #ifdef __WXMAC__ wxKeyCode presetSwitchKey = WXK_RAW_CONTROL; wxKeyCode viewSwitchKey = WXK_ALT; #else wxKeyCode presetSwitchKey = WXK_RAW_CONTROL; wxKeyCode viewSwitchKey = WXK_WINDOWS_LEFT; #endif if( aEvent.GetEventType() != wxEVT_CHAR && aEvent.GetEventType() != wxEVT_CHAR_HOOK ) return PCB_BASE_FRAME::TryBefore( aEvent ); if( !s_presetSwitcherShown && wxGetKeyState( presetSwitchKey ) && wxGetKeyState( WXK_TAB ) ) { if( m_appearancePanel && this->IsActive() ) { const wxArrayString& mru = m_appearancePanel->GetLayerPresetsMRU(); if( mru.size() > 0 ) { EDA_VIEW_SWITCHER switcher( this, mru, presetSwitchKey ); s_presetSwitcherShown = true; switcher.ShowModal(); s_presetSwitcherShown = false; int idx = switcher.GetSelection(); if( idx >= 0 && idx < (int) mru.size() ) m_appearancePanel->ApplyLayerPreset( mru[idx] ); return true; } } } else if( !s_viewportSwitcherShown && wxGetKeyState( viewSwitchKey ) && wxGetKeyState( WXK_TAB ) ) { if( m_appearancePanel && this->IsActive() ) { const wxArrayString& mru = m_appearancePanel->GetViewportsMRU(); if( mru.size() > 0 ) { EDA_VIEW_SWITCHER switcher( this, mru, viewSwitchKey ); s_viewportSwitcherShown = true; switcher.ShowModal(); s_viewportSwitcherShown = false; int idx = switcher.GetSelection(); if( idx >= 0 && idx < (int) mru.size() ) m_appearancePanel->ApplyViewport( mru[idx] ); return true; } } } return PCB_BASE_FRAME::TryBefore( aEvent ); } EDA_ANGLE PCB_BASE_EDIT_FRAME::GetRotationAngle() const { // Return a default angle (90 degrees) used for rotate operations. return ANGLE_90; } void PCB_BASE_EDIT_FRAME::ActivateGalCanvas() { PCB_BASE_FRAME::ActivateGalCanvas(); GetCanvas()->SyncLayersVisibility( m_pcb ); } void PCB_BASE_EDIT_FRAME::SetBoard( BOARD* aBoard, PROGRESS_REPORTER* aReporter ) { bool new_board = ( aBoard != m_pcb ); if( new_board ) { if( m_toolManager ) m_toolManager->ResetTools( TOOL_BASE::MODEL_RELOAD ); GetCanvas()->GetView()->Clear(); GetCanvas()->GetView()->InitPreview(); } PCB_BASE_FRAME::SetBoard( aBoard, aReporter ); GetCanvas()->GetGAL()->SetGridOrigin( VECTOR2D( aBoard->GetDesignSettings().GetGridOrigin() ) ); if( new_board ) { BOARD_DESIGN_SETTINGS& bds = aBoard->GetDesignSettings(); bds.m_DRCEngine = std::make_shared( aBoard, &bds ); } // update the tool manager with the new board and its view. if( m_toolManager ) { GetCanvas()->DisplayBoard( aBoard, aReporter ); GetCanvas()->UpdateColors(); m_toolManager->SetEnvironment( aBoard, GetCanvas()->GetView(), GetCanvas()->GetViewControls(), config(), this ); if( new_board ) m_toolManager->ResetTools( TOOL_BASE::MODEL_RELOAD ); } } void PCB_BASE_EDIT_FRAME::unitsChangeRefresh() { PCB_BASE_FRAME::unitsChangeRefresh(); if( BOARD* board = GetBoard() ) { EDA_UNITS units = GetUserUnits(); KIGFX::VIEW* view = GetCanvas()->GetView(); bool selectedItemModified = false; INSPECTOR_FUNC inspector = [units, view, &selectedItemModified]( EDA_ITEM* aItem, void* aTestData ) { PCB_DIMENSION_BASE* dimension = static_cast( aItem ); if( dimension->GetUnitsMode() == DIM_UNITS_MODE::AUTOMATIC ) { dimension->SetUnits( units ); dimension->Update(); if( dimension->IsSelected() ) selectedItemModified = true; view->Update( dimension ); } return INSPECT_RESULT::CONTINUE; }; board->Visit( inspector, nullptr, { PCB_DIM_ALIGNED_T, PCB_DIM_LEADER_T, PCB_DIM_ORTHOGONAL_T, PCB_DIM_CENTER_T, PCB_DIM_RADIAL_T, PCB_FP_DIM_ALIGNED_T, PCB_FP_DIM_LEADER_T, PCB_FP_DIM_ORTHOGONAL_T, PCB_FP_DIM_CENTER_T, PCB_FP_DIM_RADIAL_T } ); if( selectedItemModified ) m_toolManager->PostEvent( EVENTS::SelectedItemsModified ); } ReCreateAuxiliaryToolbar(); } void PCB_BASE_EDIT_FRAME::SetGridVisibility( bool aVisible ) { PCB_BASE_FRAME::SetGridVisibility( aVisible ); // Update the grid checkbox in the layer widget if( m_appearancePanel ) m_appearancePanel->SetObjectVisible( LAYER_GRID, aVisible ); } void PCB_BASE_EDIT_FRAME::SetObjectVisible( GAL_LAYER_ID aLayer, bool aVisible ) { if( m_appearancePanel ) m_appearancePanel->SetObjectVisible( aLayer, aVisible ); } COLOR_SETTINGS* PCB_BASE_EDIT_FRAME::GetColorSettings( bool aForceRefresh ) const { return Pgm().GetSettingsManager().GetColorSettings( GetPcbNewSettings()->m_ColorTheme ); } wxString PCB_BASE_EDIT_FRAME::GetDesignRulesPath() { if( !GetBoard() ) return wxEmptyString; wxFileName fn = GetBoard()->GetFileName(); fn.SetExt( DesignRulesFileExtension ); return Prj().AbsolutePath( fn.GetFullName() ); } void PCB_BASE_EDIT_FRAME::handleActivateEvent( wxActivateEvent& aEvent ) { PCB_BASE_FRAME::handleActivateEvent( aEvent ); // The text in the collapsible pane headers need to be updated if( m_appearancePanel ) m_appearancePanel->RefreshCollapsiblePanes(); }